Transactions
A transaction represents a payment instance created in Order or Checkout. It holds a list of events that make up the payment process. Each event has a type that describes the action taken on the transaction. You can see the complete list of events in the TransactionEventTypeEnum
.
Besides the events, a transaction also contains other payment information, like the amount or currency.
Amount rounding
If the provided amount uses more decimal places than used currency, it will be rounded to the nearest value.
For example:
19.999 USD
will become20.00 USD
10.2 JPY
will become10 JPY
Creating transactions
Transaction stores details of a payment transaction attached to an order or a checkout:
The transactionCreate
mutation takes the following arguments:
id
: The ID of the checkout or order.transaction
: Input data required to create a new transaction object.transactionEvent
: Data that defines a transaction event. It can be used to provide more context about the current state of the transaction.
The transactionCreate
can only be called by staff users or apps with the HANDLE_PAYMENTS
permission.
The following example shows how you can use the transactionCreate
mutation to create a new transaction.
The transaction was authorized, and the payment was made with a credit card. The actions that can be called from Saleor are: CANCEL
and CHARGE
.
The authorized amount is $99.
Saleor 3.13+
mutation {
transactionCreate(
id: "Q2hlY2tvdXQ6MWQzNmU5YzctYWEwYS00NzM5LTk0MGQtNzdjNmU4Mjc5YmQ0"
transaction: {
name: "Credit card"
message: "Authorized"
pspReference: "PSP-ref123"
availableActions: [CANCEL, CHARGE]
amountAuthorized: { currency: "USD", amount: 99 }
externalUrl: "https://saleor.io/payment-id/123"
}
) {
transaction {
id
}
}
}
The response:
{
"data": {
"transactionCreate": {
"transaction": {
"id": "VHJhbnNhY3Rpb25JdGVtOjE="
}
}
},
"extensions": {
"cost": {
"requestedQueryCost": 0,
"maximumAvailable": 50000
}
}
}
- Transactions attached to the checkout are accessible via the
checkout.transactions
field. - Transactions attached to order are accessible via the
order.transactions
field.
Updating transactions
The transactionUpdate
mutation allows updating the transaction details.
It takes the following arguments:
id
: The ID of the transaction.transaction
: Input data that will be used to update the transaction object.transactionEvent
: Data that defines a transaction event. It can be used to provide more context about the current state of the transaction.
The transactionUpdate
can only be called by staff users with the
HANDLE_PAYMENTS permission
or by the App that created the the transaction and has HANDLE_PAYMENTS
permission.
The following example shows how you can use the transactionUpdate
mutation to update the transaction.
The available action is REFUND
. The authorized funds are charged, so amountAuthorized
is $0 and amountCharged
is $99.
mutation {
transactionUpdate(
id: "VHJhbnNhY3Rpb25JdGVtOjE="
transaction: {
name: "Credit card"
message: "Authorized"
pspReference: "PSP-ref123"
availableActions: [REFUND]
amountAuthorized: { currency: "USD", amount: 0 }
amountCharged: { currency: "USD", amount: 99 }
}
transactionEvent: {
message: "Payment charged"
pspReference: "PSP-ref123.charge"
}
) {
transaction {
id
}
}
}
The response:
{
"data": {
"transactionUpdate": {
"transaction": {
"id": "VHJhbnNhY3Rpb25JdGVtOjE="
}
}
},
"extensions": {
"cost": {
"requestedQueryCost": 0,
"maximumAvailable": 50000
}
}
}
During the update of transactions, all funds that go to a new state should be subtracted from the previous state.
Assuming we have a transaction with authorizedAmount
equal to 100 USD. Moving the authorizedAmount
to chargedAmount
requires setting the authorizedAmount
to 0.
This complexity is handled automatically when Payment Apps are used instead of a custom app.
mutation {
transactionUpdate(
id: "VHJhbnNhY3Rpb25JdGVtOjE="
transaction: {
status: "Charged"
availableActions: [REFUND]
amountAuthorized: { currency: "USD", amount: 0 }
amountCharged: { currency: "USD", amount: 100 }
}
transactionEvent: {
status: SUCCESS
name: "Charged credit card"
reference: "PSP-ref123.charge"
}
) {
transaction {
id
}
}
}
Reporting actions for transactions
The transactionEventReport
is used to
report a new transaction event. The newly created event will be used to recalculate the transaction's amounts.
The mutation should be used for handling action requests or reporting any
changes that happened on the payment provider side (eg. asynchronous webhooks for delayed payment methods, chargebacks, disputes etc.).
It takes the following arguments:
id
: The id of the transaction.type
: Type of the reported action.amount
: The amount of the reported action. The amount is rounded based on the given currency precision.pspReference
: The reference assigned to the action.time
: The time of the action.externalUrl
: The URL for the staff user to check the details of the action on the payment provider's page. This URL will be available in the Saleor Dashboard.message
: Message related to the action.availableActions
: Current list of actions available for the transaction.
The transactionEventReport
can only be called by staff users with
HANDLE_PAYMENTS
permission
or by the App that created the transaction and has HANDLE_PAYMENTS
permission.
The following example shows how the transactionEventReport mutation is used to report an event
that happened for a given transaction.
The report is a success charge action, with 20 as an amount. The currency is the same as declared
for the transaction. Available action that can proceed for a transaction is REFUND
.
The provided data will be used to create a new TransactionEvent object that will be included in the
recalculation process.
mutation TransactionEventReport {
transactionEventReport(
id: "VHJhbnNhY3Rpb25JdGVtOjE="
type: CHARGE_SUCCESS
amount: 20
pspReference: "psp-123"
time: "2022-01-01"
externalUrl: "https://saleor.io/event-details/123"
message: "Charge completed"
availableActions: [REFUND]
) {
errors {
field
code
}
alreadyProcessed
transaction {
id
}
transactionEvent {
id
}
}
}
In the response, Saleor returns:
alreadyProcessed
- Defines if the reported event hasn't been processed earlier. If there is an event with the samepspReference
,amount
, andtype
as the ones provided in the input mutation, Saleor will return it instead of creating a new one, and the flag will be set totrue
. If the event with providedpspReference
andtype
was already reported but with a different amount, the error with code INCORRECT_DETAILS will be raised.transaction
- Transaction that has been updated based on the received report.transactionEvent
- TransactionEvent that has been created based on the received report.
Handling action requests for transactions
An action request is called when a staff user or an app requests an action for a given transaction.
Saleor 3.13+
Two mutations can trigger the action on the app side:
-
transactionRequestAction
: will also create a newTransactionEvent
with one of the request type (AUTHORIZATION_REQUEST
,CHARGE_REQUEST
,REFUND_REQUEST
,CANCEL_REQUEST
),amount
and theowner
(User or App). Saleor will send a synchronous webhook dedicated to the actionTRANSACTION_CHARGE_REQUESTED
,TRANSACTION_CANCELATION_REQUESTED
,TRANSACTION_REFUND_REQUESTED
-
transactionRequestRefundForGrantedRefund
: will create a newTransactionEvent
withREFUND_REQUEST
type,amount
and theowner
(User or App). Saleor will send a synchronous webhookTRANSACTION_REFUND_REQUESTED
.OrderGrantedRefund
will be included in the webhook payload (if requested in a subscription query for the webhook). This mutation is useful when the payment provider requires details about lines that are related to refund action.
The response should contain at least pspReference
of the action. The pspReference
will be placed in the previously created event of …_REQUEST
type.
Optionally the response can contain the details of the completed action.
More information about request webhooks can be found in the synchronous webhooks for transactions guide.
The webhook will be sent only to the app that created the transaction.
Asynchronously processing actions
When action is processed asynchronously on the payment provider side, the app should call the transactionActionRequest
mutation once it receives a webhook notification from the payment provider.
The diagram below shows an example of processing asynchronous refund action.
Synchronously processing the action
The app immediately receives the status of the requested action. It can provide the details of the action in response to the received Saleor webhook. The following webhook events
can accept action details in the response: TRANSACTION_CHARGE_REQUESTED
,
TRANSACTION_CANCELATION_REQUESTED
,
TRANSACTION_REFUND_REQUESTED
.
The below diagram shows an example of processing synchronous refund action.
Webhooks
Follow Transactions Webhook Events guide.