Shopware 6's payment system is an integral part of the checkout process. A payment is applied to a transaction of an order. As with any order change this is done through the state machine. At its core the payment system is composed from payment handlers, these extend Shopware to support multiple different payment types. A list of all payment handlers is stored in the database.
{% hint style="info" %} If you want to skip to the implementation details go straight to our guide on implementing a payment plugin. {% endhint %}
The payment and checkout flow consists of two essential steps. Placing the order and handling the payment. These steps are outlined in the diagram below:
The diagram above shows the payment flow for headless environments, however for the single-stack scenario (i.e. when our default storefront is used) the differences are minor and described in the section below.
If you want to see a specific example of a headless payment using the Store API, head to our API documentation.
Let's start easy. The first step for a user is to select their desired payment. The current payment method is stored in the user context, which can be manipulated by calling the corresponding route or endpoint (/store-api/context
).
In this step, an order is created. It takes no required parameters, but creates the order basend on the users current context and cart. You can add additional information like tracking parameters or comments. Shopware creates the order internally together with an open transaction which acts as a placeholder for the payment.
A transaction contains information like a unique ID, the payment method or the total amount to be payed. An order can have multiple transaction, but in this step only a single one is created.
Some payment integrations already create a payment reservation or authorization at this point. This totally depends on the specific payment extension and is not standardized by Shopware in any way. However, usually some type of payment intent or transaction reference is stored in the meantime.
This step can only be executed after an order has been placed. It starts the payment by determining the correct payment handler for the selected payment method.
{% hint style="info" %} Whilst from a functional perspective steps 2 and 3 are separated, in our default storefront both are initiated in the same request. {% endhint %}
There are two types of payment handlers in Shopware
- Synchronous
- Asynchronous
Synchronous Payment
In this scenario, the payment integration usually makes a request to the payment gateway to execute or authorize the payment. The gateway immediately responds with a status and Shopware can give feedback to the user.
Asynchronous Payment
These payments include a user redirect. The redirect target is defined by the payment integration and contains information about the transaction as well as a callback URL for the payment gateway.
The frontend can also define success and error URLs that will be used for the eventual redirect after step 3.3.
In the default storefront, this redirect takes place automatically. In the headless scenario, Shopware returns the redirect URL within the API response, so the frontend can perform the redirect.
This step is only executed for asynchronous payments. After being redirected, the user can perform final checks or authorizations on the payment gateway UI. After that, the payment gateway redirects the user to the callback URL provided in step 3.1 along with a parameter indicating the outcome of the payment.
This step is only executed for asynchronous payments. It is triggered by the callback URL (which points to /payment/finalize-transaction
) that has been provided to the payment gateway in step 3.1. Depending on the payments success, Shopware updates the transaction status and redirects the user to the corresponding finish page from step 3.
{% hint style="warning" %} Disclaimer
The actual implementation of payment integrations differ between providers. For that reason our specification does not include any guidelines about payment states or specific API calls to be made. Some integrations share data between the steps or provide and call upon web hooks after the payment process has been finished. These implementations go beyond our standards. {% endhint %}
{% page-ref page="../../../guides/plugins/plugins/checkout/payment/add-payment-plugin.md" %}