This package provides a one page standard checkout with the following features:
- Concept for PaymentProviders, that can be used to implement specific Payments
- An "offline payment" provider is part of the module
Table of content
If your template does not want to ask for all the information required you can also set default values for the checkoutform (strings)
commerce:
checkout:
# to enable the offline payment provider
enableOfflinePaymentProvider: true
# checkout flow control flags:
skipStartAction: false
skipReviewAction: false
showReviewStepAfterPaymentError: false
showEmptyCartPageIfNoItems: false
redirectToCartOnInvalideCart: false
# checkout form settings:
useDeliveryForms: true
usePersonalDataForm: false
privacyPolicyRequired: true
# GraphQL place order process
placeorder:
lock:
type: "memory" # only suited for single node applications, use "redis" for multi node setup
contextstore:
type: "memory" # only suited for single node applications, use "redis" for multi node setup
This module implements a controller for the following checkout flow (the checkout process for the end customer):
- StartAction (optional)
- check if user is logged in
- yes: next action
- no: show start template (where the user can login)
- check if user is logged in
- Checkout Action
- this is the main step, and the template should render the big checkout form (or at least the parts that are interesting).
- on submit and if everything was valid:
- Action will update the cart - specifically the following information:
- Update billing (on cart level)
- Update deliveryinfos on (each) delivery in the cart (e.g. used to set shipping address)
- Select Payment Gateway and preferred Payment Method
- Optional Save the wished payment split for each item in the cart
- Optional add Vouchers/GiftCards (may already happened before)
- If Review Action is skipped:
- Start payment and place order if needed (EarlyPlaceOrder)
- Redirect to Payment Action
- If Review Step is not skipped:
- Redirect to Review Action
- Action will update the cart - specifically the following information:
- Review Action (Optional)
- Renders "review" template that can be used to review the cart
- After confirming start the payment and place order if needed (EarlyPlaceOrder)
- Payment Action
- Ask Payment Gateway about FlowStatus and handle it
- FlowStatus:
- Error / Abort by customer: Regenerate Idempotency Key of PaymentSelection, redirect to checkout and reopen cart if needed
- Success / Approved: Redirect to PalceOrderAction
- Unapproved: Render payment template and let frontend decide how to continue in flow (e.g. redirect to payment provider)
- Place Order Action
- Check if order already placed (EarlyPlaceOrder)
- If order not already placed check FlowStatus and place order
- Put order infos in flash message and redirect to Success Action
- Success Action:
- Renders order success template
When we introduced GraphQL, we rethought the checkout process from the ground up. Among other things, we decided to map the individual steps of the checkout to states of a newly created place order state machine. This should mainly make the process more robust and make it easier to roll back operations in case of errors.
Important: when you start using the new place order process please ensure that you use the "OnWrite" Flamingo session save mode:
flamingo.session.saveMode: "OnWrite"
The checkout module exposes the following Mutations and Queries:
mutation Commerce_Checkout_StartPlaceOrder
starts a place order process and returns the process' UUID. If there is already a running process, it will be replaced by the new one. The processing is started in background and continues after the return of the mutation until an action is required or a final state (error or success) is reached.mutation Commerce_Checkout_RefreshPlaceOrder
refreshes the process and tries to start the background processing again if it is not running anymore. The result is the state at the moment of the mutation. If the background process is still running, this mutation is non-operational.mutation Commerce_Checkout_RefreshPlaceOrderBlocking
refreshes the process and starts the background processing again if it is not running anymore. It waits with the return until the background process comes to a final state or need an action.mutation Commerce_Checkout_CancelPlaceOrder
cancels the running process if it is not yet in a final state.mutation Commerce_Checkout_ClearPlaceOrder
clears the last stored process. (Non-operational if there is no last process)query Commerce_Checkout_ActivePlaceOrder
checks if there is a place order process in a non-final state.query Commerce_Checkout_CurrentContext
returns the current state without restarting the background processing.
We differentiate between internal and exposed states.
Internal states implement the interface checkout/domain/placeorder/process.State
. There is a map binding on this interface using the name of the state.
This way, other modules can overwrite specific states to introduce their own implementation.
The start and failed states are defined by an annotated binding with annotations startState
and failedState
, respectively.
Exposed states implement the interface checkout/interfaces/graphql/dto.State
. To map internal states to exposed states,
we use a map binding on the dto.State
interface with the internal state names as keys and the exposed state as target.
The default implementation defines the state flow as follows (for an cart that needs payment):
Fully discounted carts don't need a payment, therefore the state flow is similar but lacks the payment creation/validation:
The place order context must be stored aside of the session, since it is manipulated by a background process.
In Memory
Important: This context store is only suited for single node applications, please use redis for multi node setup
Default context store implementation. Provides a map based in memory adapter for the ContextStore
port.
commerce.checkout.placeorder.contextstore.type: "memory"
Redis
The Redis implementation writes the context into the configured redis instance. The complete entry will be gob-encoded.
Be aware that you have to gob-register your own StateData
if you introduce some.
commerce.checkout.placeorder.contextstore:
type: "redis"
redis:
maxIdle: 25
idleTimeoutMilliseconds: 240000
network: "tcp"
address: "localhost:6379"
database: 0
To ensure that the state machine cannot be processed multiple times for one process, we have decided to introduce a process lock. At the start of each place order transaction an attempt is made to obtain a lock, if this is not possible a transaction is already running and we just wait.
The module offers the TryLock
port and currently two implementations (Memory, Redis).
In Memory
Important: This lock is only suited for single node applications, please use redis for multi node setup
Default lock implementation. Provides a mutex based in memory adapter for the TryLock
port.
commerce.checkout.placeorder.lock.type: "memory"
Redis
Provides a redis based lock implementation using the go-redsync/redsync package. Node acquires the lock and refreshes it every X second, if the node dies the lock is automatically released after the provided max duration.
commerce.checkout.placeorder.lock:
type: "redis"
redis:
maxIdle: 25
idleTimeoutMilliseconds: 240000
network: "tcp"
address: "localhost:6379"
database: 0
New GraphQL related process context store. For more details see Context store
New GraphQL related process lock. For more details see Locking