This tutorial is part of the « E-commerce website with Strapi, Nuxt.js, GraphQL and Stripe
Note: The source code is available on Github.
You must be starving by now... I am sure you want to be able to order!
Let’s create a new checkout page, and store orders in our database.
To store the orders in our database, we will create a new content type.
Order
for the Display name, and click Continue.address
in the Name field.dishes
under the Name field, then click Finish.amount
under the Name field, then click Finish.Choose the Relation field, create a many-to-one relation, user has many orders as below, then click Finish.
Finally, click Save and wait for Strapi to restart.
Now that we have created new fields for the orders collection type. We will take care of the permissions, same process as usual.
We need to make sure that authenticated users can create new orders:
In the next section, we will setup Stripe.js to get user's address and debit card information.
You are going to create a checkout page that will display your cart thanks to the Cart
component and the Stripe form.
Add Stripe to your frontend app, run:
1 yarn add vue-stripe-elements-plus
In the nuxt.config.js
file, add the Stripe script in the head
object:
1 module.exports = {
2 // Global page headers: https://go.nuxtjs.dev/config-head
3 head: {
4 // ...other properties
5 script: [{ src: 'https://js.stripe.com/v3/' }],
6 },
7 }
You can use Stripe.js’ APIs to tokenize customer information, collect sensitive payment details using customizable Stripe Elements, and accept payments with browser payment APIs like Apple Pay and the Payment Request API.
Create a new pages/checkout.vue
, open the file with your text editor, and copy/paste the following code.
1 // pages/checkout.vue
2 <template>
3 <div class="uk-container uk-container-xsmall">
4 <h1 class="uk-heading-small">Checkout</h1>
5 <div v-if="success" class="uk-alert-success" uk-alert>
6 <a class="uk-alert-close" uk-close></a>
7 <p>{{ success.message }}</p>
8 </div>
9 <div v-if="err" class="uk-alert-danger" uk-alert>
10 <a class="uk-alert-close" uk-close></a>
11 <p>{{ err.message }}</p>
12 </div>
13 <Cart :checkout="false" />
14 <div class="uk-margin">
15 <label class="uk-form-label">
16 Address
17 <input v-model="address" class="uk-input" type="email" />
18 </label>
19 </div>
20 <div class="uk-margin-top">
21 <StripeElements v-slot="{ elements }" ref="elms" :stripe-key="stripeKey">
22 <StripeElement ref="card" type="card" :elements="elements" />
23 </StripeElements>
24 <button
25 class="uk-button uk-button-secondary uk-margin-top uk-width-1-1"
26 @click="pay"
27 >
28 Pay
29 </button>
30 </div>
31 </div>
32 </template>
33 <script>
34 import { StripeElements, StripeElement } from 'vue-stripe-elements-plus'
35 import { mapGetters, mapMutations } from 'vuex'
36 export default {
37 components: {
38 StripeElements,
39 StripeElement,
40 },
41 data() {
42 return {
43 success: null,
44 err: null,
45 address: '',
46 stripeKey: process.env.stripePublishable,
47 }
48 },
49 computed: {
50 ...mapGetters({
51 username: 'auth/username',
52 token: 'auth/token',
53 }),
54 },
55 methods: {
56 async pay() {
57 // ref in template
58 const groupComponent = this.$refs.elms
59 const cardComponent = this.$refs.card
60 // Get stripe element
61 const cardElement = cardComponent.stripeElement
62 const address = this.address
63 const username = this.username
64 let token = this.token
65 this.$http.setToken(token, 'Bearer')
66 try {
67 // Access instance methods, e.g. createToken()
68 token = await groupComponent.instance.createToken(cardElement)
69 } catch (err) {
70 this.err = err.response?.data?.error
71 }
72 try {
73 await this.$http.$post('orders', {
74 data: {
75 address,
76 amount: this.$store.getters['cart/price'],
77 user: username,
78 dishes: this.$store.getters['cart/items'],
79 token,
80 },
81 })
82 // this.emptyCart()
83 this.success = {
84 message: 'Payment completed successfuly',
85 }
86 } catch (err) {
87 this.err = err.response?.data?.error
88 }
89 },
90 ...mapMutations({
91 emptyCart: 'cart/emptyList',
92 }),
93 },
94 }
95 </script>
96
97Don’t forget to create a `stripePublishable` property inside `env` object in your `nuxt.config.js` file:
98
99 export default {
100 env: {
101 stripePublishable: process.env.STRIPE_PUBLISHABLE || '##PUBLISHABLE##'
102 }
103 }
Don't forget to replace ##PUBLISHABLE##
by your public Stripe API key.
In this page, you display a form to get user's address and debit card information. You use the Stripe Elements system. When the form is submitted, you get a token from Stripe. Then, you create the order in your Strapi API.
You are now able to let users submit their order. Bon appétit! 🇫🇷
Create a Stripe account, and navigate to API Keys.
If you have already used Stripe, you may know that the credit card information does not go through your backend server. Instead, the information is directly sent to the Stripe API. Then, your frontend receives a token. The id
of the token must be sent to your backend which will create the Stripe charge.
Open the backend
terminal, and install the stripe
package:
1 // Ctrl + C
2 yarn add stripe
The Stripe Node library provides convenient access to the Stripe API from applications written in server-side JavaScript.
In order to integrate the Stripe logic, you need to update the create
charge endpoint in your Strapi API.
1 // src/api/order/controllers/order.js
2 'use strict';
3 const stripe = require('stripe')(process.env.STRIPE_KEY);
4 /**
5 * order controller
6 */
7 const { createCoreController } = require('@strapi/strapi').factories;
8
9 module.exports = createCoreController('api::order.order', ({ strapi }) => ({
10 async create(ctx) {
11 const {
12 amount,
13 address,
14 dishes,
15 token,
16 } = ctx.request.body.data;
17 try {
18 // Charge the customer
19 await stripe.charges.create({
20 amount: amount,
21 currency: 'eur',
22 description: `Order ${new Date()} by ${ctx.state.user.id}`,
23 source: token,
24 });
25
26 // Create the order
27 const entity = await strapi.service('api::order.order').create({ amount, address, dishes, user: ctx.state.user.id }).
28 const sanitizedEntity = await this.sanitizeOutput(entity, ctx);
29 return this.transformResponse(sanitizedEntity);
30 } catch (err) {
31 // return 500 error
32 ctx.response.status = 500;
33 return { error: { message: 'There was a problem creating the charge'}};
34 }
35 }
36 }));
Create a .env
file in the backend
directory, then copy/pase the following code:
1STRIPE_KEY=##Secret##
Don't forget to replace ##Secret##
by your Stripe API Key.
Restart the Strapi backend with npm run develop
.
Note: In a real-world example, the amount should be checked on the backend side and the list of dishes related to the command should be stored in a more specific Content Type called
orderDetail
.
In this section, we have created a new checkout page, which let user submit a form with their address and debit card information and used Stripe API to check the amount. 🚀 In the next (and last) section, you will learn how to deploy your Strapi & Nuxt.js applications.
Pierre created Strapi with Aurélien and Jim back in 2015. He's a strong believer in open-source, remote and people-first organizations. You can also find him regularly windsurfing or mountain-biking!