This page is for classic Adyen for Platforms integrations. If you are just starting your implementation, refer to our new integration guide instead.
Adyen for Platforms enables you to onboard account holders with point-of-sale (POS) integrations to accept in-person payments. To make this happen, you need to:
- Subscribe to store notifications.
- Add a store for the account holder.
- Assign a terminal to the store and ship it to the the account holder's store address.
- Split funds between accounts in your platform.
Requirements
Before you start processing point-of-sale payments for your account holders, you need to:
- Have a Terminal API integration with Adyen.
- Be familiar with the features of Adyen for Platforms, such as account structure, the onboarding and verification process, split payments, and notifications.
- Contact our Support Team and provide the following required information.
- Countries
- Currencies
- Merchant category codes (MCCs)
- Payment methods
You can specify different currencies, MCCs, and payment methods for each country/region. We use this information to enable you to create stores based on the type of payments that must be processed in the store.
Subscribe to store notifications
When you add a store for an account holder, we send a notification to let you know the store has been set up. To ensure you receive this notification:
-
Make a POST request to the Terminal API endpoint, specifying:
active
: true.description
: Your text to recognize the notification subscription by.- An
eventConfigs
array witheventType
: ACCOUNT_HOLDER_STORE_STATUS_CHANGE andincludeMode
: INCLUDE. notifyURL
: Endpoint on your server where you'll receive the notifications.hmacSignatureKey
: Use this parameter if you want us to protect notifications with HMAC signatures. It specifies the HMAC key that we'll use to calculate the HMAC signatures of the notifications we send to thenotifyURL
. On your end you'll need to verify the signatures of the incoming notifications.notifyUsername
andnotifyPassword
: Username and password that we need for basic authentication with the server where you'll receive the notifications.sslProtocol
: Protocol used to secure the communication, such as TLS or SSL.
/createNotificationConfiguration requestExpand viewCopy link to code blockCopy codecurl https://cal-test.adyen.com/cal/services/Notification/v6/createNotificationConfiguration \ -H 'x-api-key: ADYEN_API_KEY' \ -H 'content-type: application/json' \ -d '{ "configurationDetails":{ "active": true, "description":"YOUR_UNIQUE_DESCRIPTION", "eventConfigs":[ { "eventType":"ACCOUNT_HOLDER_STORE_STATUS_CHANGE", "includeMode":"INCLUDE" } ], "notifyURL":"https://www.merchant-domain.com/notification-handler", "hmacSignatureKey":"c6c608e9d2ec95889cb9757e6beb4956c187f44cf7bb7297b16880c0310af7ff", "notifyUsername":"YOUR_USERNAME", "notifyPassword":"YOUR_PASSWORD", "sslProtocol":"SSL" } }' ResponseExpand viewCopy link to code blockCopy code{ "pspReference": "8515681150749298", "configurationDetails": { "active": true, "description": "YOUR_UNIQUE_DESCRIPTION", "eventConfigs": [ { "eventType": "ACCOUNT_HOLDER_STORE_STATUS_CHANGE", "includeMode": "INCLUDE" } ], "notificationId": 20337, "notifyURL": "https://www.merchant-domain.com/notification-handler", "sslProtocol": "SSLInsecureCiphers" } } -
Make sure that your server accepts the notifications.
-
Use the
notificationId
you receive in the response to test the notification.
Create a store for the account holder
Make sure our Support Team has enabled creating stores. If you haven't requested this yet, refer to the requirements.
Once our Support Team has enabled creating stores, you can start adding stores for your account holders.
To provide store details, include a storeDetails
array when you create a new account holder or when you update an existing one:
-
Make a POST request to one of the following endpoints:
Situation Endpoint Specify There is no account holder for the sub-merchant yet. /createAccountHolder The usual parameters to create a business account holder, as well as businessDetails.taxId
.You want to add a store to an existing account holder. /updateAccountHolder accountHolderCode
: Your unique reference for the account holder. IncludebusinessDetails.taxId
, unless it is already present as part ofbusinessDetails
. -
In the request body, include an accountHolderDetails.storeDetails array, specifying:
- address: The
city
,country
,houseNumberOrName
,postalCode
,stateOrProvince
, andstreet
of the physical store where the account holder will process payments from. - fullPhoneNumber: The phone number of the store.
- merchantAccount: The merchant account used for accepting payments.
- merchantCategoryCode: The merchant category code (MCC) that classifies the business of the account holder.
- storeName: The name of the account holder's store. This will be included in the shopper statement.
- storeReference: Your unique reference for the account holder's store.
The next example shows a /createAccountHolder request to create a new account holder with a store.
/createAccountHolder requestExpand viewCopy link to code blockCopy codecurl https://cal-test.adyen.com/cal/services/Account/v6/createAccountHolder \ -H 'x-api-key: ADYEN_API_KEY' \ -H 'content-type: application/json' \ -d '{ "accountHolderCode": "YOUR_UNIQUE_ACCOUNT_HOLDER_CODE", "accountHolderDetails": { "address": { "country": "US" }, "businessDetails": { "legalBusinessName": "Real Good Restaurant Inc.", "taxId": "444455555", "shareholders": [ { "name": { "firstName": "Maria", "gender": "FEMALE", "lastName": "Green" }, "address": { "country": "US" } } ] }, "email": "maria@green.com", "storeDetails" : [ { "storeReference": "YOUR_SUBMERCHANT_STORE_ID", "storeName": "Store Name", "merchantAccount": "YOUR_MERCHANT_ACCOUNT", "merchantCategoryCode": "7999", "address" : { "city": "Store City", "country": "US", "houseNumberOrName": "1", "postalCode": "123", "stateOrProvince": "CA", "street": "Store Street" }, "fullPhoneNumber": "+31201234567" } ] }, "legalEntity": "Business" }' The following example shows how to add a store to an existing account holder using an /updateAccountHolder call.
/updateAccountHolder requestExpand viewCopy link to code blockCopy codecurl https://cal-test.adyen.com/cal/services/Account/v6/updateAccountHolder \ -H 'x-api-key: ADYEN_API_KEY' \ -H 'content-type: application/json' \ -d '{ "accountHolderCode": "YOUR_UNIQUE_ACCOUNT_HOLDER_CODE", "accountHolderDetails": { "storeDetails" : [ { "storeReference": "YOUR_SUBMERCHANT_STORE_ID", "storeName": "Store Name", "merchantAccount": "YOUR_MERCHANT_ACCOUNT", "merchantCategoryCode": "7999", "address": { "city": "Store City", "country": "US", "houseNumberOrName": "1", "postalCode": "123", "stateOrProvince": "CA", "street": "Store Street" }, "fullPhoneNumber": "+31201234567" } ] } }' The response contains various account holder details, including the
storeDetails
showing that the status of the store is Pending.ResponseExpand viewCopy link to code blockCopy code{ "invalidFields": [], ... "accountHolderDetails":{ ... "storeDetails" : [ { "address": { "city": "Store City", "country": "US", "houseNumberOrName": "1", "postalCode": "123", "stateOrProvince": "CA", "street": "Store Street" }, "fullPhoneNumber": "+31201234567", "merchantAccount": "YOUR_MERCHANT_ACCOUNT", "merchantCategoryCode": "7999", "status": "Pending", "store": "6db2bdbc-5e77-4388-9a6e-9e653905d3e5", "storeReference": "YOUR_SUBMERCHANT_STORE_ID", "storeName": "Store Name", } ], ... }, ... } - address: The
-
Before processing point-of-sale (POS) payments, wait until you receive an
ACCOUNT_HOLDER_STORE_STATUS_CHANGE
notification showing that the new status of the store is Active.NotificationExpand viewCopy link to code blockCopy code{ "eventType":"ACCOUNT_HOLDER_STORE_STATUS_CHANGE", "eventDate":"2018-04-23T13:13:44+02:00", "executingUserKey":"ws", "live":true, "pspReference":"12312312313", "content":{ "accountHolderCode":"YOUR_UNIQUE_ACCOUNT_HOLDER_CODE", "store": "6db2bdbc-5e77-4388-9a6e-9e653905d3e5", "storeReference":"YOUR_SUBMERCHANT_STORE_ID", "newStatus":"Active", "oldStatus":"Pending", "reason":"Test Reason" } }
Assign a terminal to the store
To process POS payments, the account holder's store needs to have one or more payment terminals. The process is as follows:
-
Select a terminal and inform the account holder of the communication requirements.
-
Order a terminal if you do not have the required terminal in your inventory.
-
Assign the terminal from your inventory to the store of the account holder in one of the following ways:
- Assign terminals using your Customer Area
or -
Assign terminals in bulk using API calls
The advantage of using the Terminal Management API over the Customer Area, is that API calls enable you to automate assigning terminals and retrieving an overview of terminal assignments.
- Assign terminals using your Customer Area
-
Configure the terminals in the Customer Area, under Point of Sale > Terminal settings.
-
Ship the terminal to the physical store address and instruct the account holder to board the terminal.
Split a payment
To split a point-of-sale payment, you can:
- Create a split configuration to automatically split payments processed through the store. For more information, refer to How split configuration works.
- Send split instructions for every transaction at time of payment or at capture.
When you use a split configuration, Adyen evaluates the rules in the split configuration for all the transactions processed through the store. If you need to change the commission fees for specific transactions, you can send split instructions in the payment or capture API request. Any split instructions that you send through the API overrides the automatic split.
Make a split payment using the Terminal API
To split a POS payment, you make a Terminal API call. In the PaymentRequest
you provide the split payment data in the SaleToAcquirerData
, in key-value pair or Base64-encoded format. This credits payments directly to an account holder's account.
To make the payment:
-
Make a POST request to a Terminal API endpoint, specifying:
-
MessageHeader
: the standardSaleToPOIRequest.MessageHeader
object. Specify:Parameter Required Description ProtocolVersion
3.0 MessageClass
Service MessageCategory
Payment MessageType
Request ServiceID
Your unique ID for this request, consisting of 1-10 alphanumeric characters. Must be unique within the last 48 hours for the terminal ( POIID
) being used.SaleID
Your unique ID for the POS system component to send this request from. POIID
The unique ID of the terminal to send this request to. Format: [device model]-[serial number].
-
PaymentRequest
: The request body. This must include:Parameter Required Description SaleData.SaleTransactionID
An object with: TransactionID
: your reference to identify a payment. We recommend using a unique value per payment. In your Customer Area and Adyen reports, this will show as the merchant reference for the transaction.TimeStamp
: date and time of the request in UTC format.
SaleData.SaleToAcquirerData
Provides the split payment data as concatenated key-value pairs separated by an ampersand (&) character, or in Base64-encoded format. See the tables below for details. PaymentTransaction.AmountsReq
An object with: Currency
: the transaction currency.RequestedAmount
: the transaction amount.
-
To provide the split payment data in
PaymentRequest.SaleData.SaleToAcquirerData
, use the following fields:Field Description Example split.api
Version of the Split API: 1. split.api=1 split.totalAmount
Amount to be split, in minor units. Must be equal to the transaction amount and to the sum of the split amounts. split.totalAmount=62000 split.currencyCode
Currency of the amount to be split. split.currencyCode=EUR split.nrOfItems
Number of times you will split the payment. Must match the number of split items in the request. split.nrOfItems=2 Specify each split as a
split.item{item#}
starting at 1:split.item1
,split.item2
, and so on. Each split item includes:Field Description Example split.item{item#}.amount
Amount of the split, in minor units. split.item1.amount=60000 and split.item2.amount=2000 split.item{item#}.type
Split type. A type of MarketPlace or VAT sends the amount to the account
specified. A type of Commission or PaymentFee sends the amount to your liable account.split.item1.type=MarketPlace and split.item2.type=Commission split.item{item#}.account
Account that will receive the split. This is the accountCode
of one of your account holder's accounts or your own liable account. You do not need to specify an account whensplit.type
is Commission.split.item1.account=8815628270908491 split.item{item#}.reference
The reference for that specific transaction split, which is returned in our reporting. Required if the split.type
is MarketPlace.
The following example shows how to split a EUR 620.00 payment into EUR 600.00 to be paid into the account holder's account with
accountCode
8815628270908491 and EUR 20.00 commission to be paid to the platform.Payment request with split instructionsExpand viewCopy link to code blockCopy code{ "SaleToPOIRequest":{ "MessageHeader":{ "ProtocolVersion":"3.0", "MessageClass":"Service", "MessageCategory":"Payment", "MessageType":"Request", "SaleID":"POSSystemID12345", "ServiceID":"0207111104", "POIID":"V400m-324688179" }, "PaymentRequest":{ "SaleData":{ "SaleTransactionID":{ "TransactionID":"27908", "TimeStamp":"2019-03-07T10:11:04+00:00" }, "SaleToAcquirerData": "split.api=1&split.nrOfItems=2&split.totalAmount=62000&split.currencyCode=EUR&split.item1.amount=60000&split.item1.type=MarketPlace&split.item1.account=8815628270908491&split.item1.reference=test1&split.item2.amount=2000&split.item2.type=Commission&split.item2.reference=TestCommission" }, "PaymentTransaction":{ "AmountsReq":{ "Currency":"EUR", "RequestedAmount":620.00 } } } } } The payment request is routed to the terminal, for the shopper to present their card and verify the payment. The payment is then sent to the Adyen payments platform for processing.
-
-
Receive the payment result.
If your integration uses asynchronous cloud communications, you must set up event notifications. We then send the Terminal API responses to your endpoint.
If the payment is successful:
- Approved is shown on the terminal display.
- You receive a payment response containing:
POIData.POITransactionID.TransactionID
: Transaction identifier for the payment in the formattenderReference.pspReference
. For example, oLkO0012498220087000.981517998282382C.PaymentResponse.Response.Result
: SuccessPaymentResponse.Response.AdditionalResponse
: A base64 string. When decoded, this is a JSON object with additional transaction data.PaymentReceipt
: Object containing data you can use to generate a receipt.
You can also view the details of a payment in your Customer Area, under Transactions > Payments.
Split at capture using the API
Some platforms do not know the final split amounts at the moment the payment request is made. If this applies to your account holder's POS payments, you need to enable manual capture for POS payments and follow up each authorised Terminal API payment with a /payments/{paymentPspReference}/captures API call. Splitting at capture is also the procedure to use for tipping (gratuity) and partial payments.
Enabling manual capture for POS payments applies to all your point-of-sale account holders.
Proceed as follows to split at capture:
-
From the
transactionID
field in the payment response, get thepspReference
of the authorization you want to capture. -
To capture and split a payment at the same time, send a POST /payments/{paymentPspReference}/captures request, where
paymentPspReference
is thepspReference
of the authorization you want to capture.In the body, include the splits array. For each item in the array, specify the following fields:
Parameter Required Description account The account that will receive (or be charged) the split amount. This is the accountCode
of your account holder's accounts or your liable balance account. You do not need to specify this field whentype
is Commission.amount.value The value of the split amount. You do not need to specify this field for transaction fees, since they are not known at the time of payment. type Defines the part of the payment you want to book to a specific account
.
Possible values:- Commission: books the commission to your platform's liable account.
- MarketPlace: books the sale amount to the specified account.
- PaymentFee: books the transaction fees to your platform's liable account.
- VAT: books the value-added tax for the sale amount to the specified account.
reference Required if type
is MarketPlaceYour reference for the split, which you can use to link the split to other operations and to reconcile payments. If the reference is not provided, the split is reported as part of the aggregated TransferBalance record in the Marketplace Payments accounting report. description Your description for that specific transaction split, which is returned in our reporting.
The example below shows how to capture a EUR 620.00 authorization with pspReference
981517998282382C, and split it into EUR 600.00 to be paid into the account holder's account with accountCode
8815628270908491 and EUR 20.00 commission to be paid to your liable account.
curl https://checkout-test.adyen.com/v71/payments/981517998282382C/captures \ -H 'x-api-key: ADYEN_API_KEY' \ -H 'content-type: application/json' \ -d '{ "merchantAccount": "YOUR_MERCHANT_ACCOUNT", "amount": { "value": 62000, "currency": "EUR" }, "reference": "YOUR_REFERENCE_NUMBER", "splits":[ { "amount":{ "value":60000 }, "type":"MarketPlace", "account":"8815628270908491", "reference":"YOUR_REFERENCE_TO_SPLIT_#1" }, { "amount":{ "value":2000 }, "type":"Commission", "reference":"YOUR_REFERENCE_TO_SPLIT_#2" } ] }'
Check the result
API response
In the capture response, note the following:
paymentPspReference
: the PSP reference of the authorizationpspReference
: the PSP reference associated with this capture request. This is different from the PSP reference of the authorization.status
: received
Webhook
Wait for the CAPTURE webhook to learn the outcome of the request.