Split Payouts on Payment Provider Protocol
Split payouts are a common requirement for payment providers used by marketplaces, since it allows marketplaces to not only process the payment for an order, but also collect their commission and pay sellers for their products. This leads to improved operational efficiency when managing payouts for a large marketplace.
The VTEX platform allows the merchant to register sellers, configure commission percentages (total order and freight values), and specific commissions by seller category. Check on Adding a Seller.
Cart Scenarios
When an order is placed, one of three scenarios are present in the cart:
- Products only from the Marketplace.
- Products from the Marketplace, and one or more sellers.
- Products only from seller(s).
Marketplace Products only
As the order contains products only from the Marketplace, there is no split of payment, calculation of commissioning, or split payload to be sent. The Marketplace will receive the full order payment value.
Marketplace and Sellers Products only
When an order contains products from the Marketplace and sellers (or only from sellers), a split payout is required. In this scenario, the VTEX gateway is responsible for:
- Calculate commissioning per seller of each order.
- Send split payload data to the payment processor.
- Process the complete or partial refund in split operations, if required.
The split payload is sent only if the payment provider supports this scenario.
An illustrative example is shown below to explain how a calculation breakdown for split is performed:
- The registered commission is: a. 16% for the marketplace when buying the product from seller X. b. 20% for the marketplace when buying products from seller Y.
- The cart is made up of products from sellerX, sellerY, and the Marketplace itself.
- An Order with the value of 199.62 is registered. See below:
_23"items": [_23      {_23        "id": "25807",_23        "name": "marketplace product",_23        "sellerId": "markeplace",_23        "value": 69.90,_23        "comission": 0_23      },_23      {_23        "id": "29052",_23        "name": "SellerX product",_23        "sellerId": "sellerX",_23        "value": 87.12,_23        "comission": 13.94_23      },_23      {_23        "id": "48760",_23        "name": "SellerY product",_23        "sellerId": "sellerY",_23        "value": 42.60,_23        "comission": 8.52_23      }_23]
- The following containers will be sent with these values: a. 92.36 for the marketplace. b. 73.18 for seller X. c. 34.08 for seller Y.
Calculation Breakdown
| ID | Item Value | Marketplace Comission | Recipient | 
|---|---|---|---|
| SellerX | 87.12 | 13.94 | 73.18 | 
| SellerY | 42.60 | 8.52 | 34.08 | 
| Marketplace | 69.90 | - | (69.90+13.94+8.52) = 92.36 | 
Split Time Settings
It is also possible to configure the following characteristics of the split process:
- 
Automatic Capture Time: The payment processor has the permission to control if the automatic capture will occur before or after approval of anti-fraud (setting the fields as delayToAutoSettleordelayToAutoSettleAfterAntifraud).
- 
Split Payload Sending Time: This moment can be set according to each method of payment associated with the transaction. For processors within the PPP (Payment Provider Protocol), it can be done through setting the field on the manifest as onAuthorize,onCaptureordisabled.If not specified by the processor, the default behavior is: - Credit card transaction: The sending of the split is in the capture call.
- Transaction with boleto: The sending of the split is in the authorization call.
 
Split Payout Solution for Payment Provider Protocol
Our Split Payout solution consists of adding an array of objects in the Payment Provider Protocol that we call recipients.
Each item of this array informs the necessary data so that the provider can perform the split between the recipients involved in that transaction (marketplace/sellers). The following fields are part of the recipient’s object:
| Field | Type | Description | 
|---|---|---|
| id | string | Recipient identifier | 
| name | string | Recipient name | 
| documentType | string | Recipient document type | 
| document | string | Recipient document number | 
| role | string | Indicates if the recipient is the seller or the marketplace | 
| chargeProcessingFee | boolean | Indicates whether or not this recipient is charged for processing fees | 
| chargebackLiable | boolean | Indicates whether or not this recipient is liable to chargebacks | 
| amount | float | Amount due to this recipient | 
| comissionAmount | float | Amount received by the Marketplace | 
Payment Protocol modifications
In this section, we will describe modifications applied to our payment protocol at each stage (e.g., authorization, capture, etc.) when the split payout is active.
Authorization Stage
For transactions where the split in authorization is enabled, we send the recipients in the authorization payload. The “recipient’s” object array contains data on the values to be passed on to each seller and the marketplace involved in the order.
See below an example of the authorization request:
_125curl --location --request POST 'https://{{providerApiEndpoint}}/payments' \\_125--header 'X-VTEX-API-AppKey: {{X-VTEX-API-AppKey}}' \\_125--header 'X-VTEX-API-AppToken: {{X-VTEX-API-AppToken}}' \\_125--header 'Content-Type: application/json' \\_125--header 'Accept: application/json' \\_125--data-raw '{_125  "value": 199.62,_125  "reference": "22590454",_125  "orderId": "v22590454abc",_125  "transactionId": "888EC956B26A4F53B3A8F2D420271195",_125  "paymentId": "FAD57B4061034026A78545DD3FC6A85D",_125  "paymentMethod": "Visa",_125  "merchantName": "mystore",_125  "card": {_125      "holder": "John Doe",_125      "number": "4682185088924788",_125      "csc": "021",_125      "expiration": {_125          "month": "06",_125          "year": "2029"_125      }_125  },_125  "currency": "BRL",_125  "installments": 1,_125  "deviceFingerprint": "12ade389087fe",_125  "miniCart": {_125    "buyer": {_125      "id": "3287c060-2e43-4dc7-b730-4b8d2a9bd114",_125      "firstName": "Mary",_125      "lastName": "Rose",_125      "document": "11112222333",_125      "documentType": "cpf",_125      "email": "mary.rose@example.com",_125      "phone": "+5521978888888"_125    },_125    "shippingAddress": {_125      "country": "BRA",_125      "street": "Praia de Botafogo",_125      "number": "300",_125      "complement": "3rd Floor",_125      "neighborhood": "Botafogo",_125      "postalCode": "22250040",_125      "city": "Rio de Janeiro",_125      "state": "RJ"_125    },_125    "billingAddress": {_125      "country": "BRA",_125      "street": "Brigadeiro Faria Lima Avenue",_125      "number": "4440",_125      "complement": "10tth Floor",_125      "neighborhood": "Itaim Bibi",_125      "postalCode": "04538132",_125      "city": "Rio de Janeiro",_125      "state": "RJ"_125    },_125    "items": [_125      {_125        "id": "25807",_125        "name": "SellerX product",_125        "price": 71.2,_125        "quantity": 1,_125        "discount": 0,_125        "categoryId": "1000097",_125        "sellerId": "sellerX"_125      },_125      {_125        "id": "29052",_125        "name": "marketplace product",_125        "price": 69.9,_125        "quantity": 1,_125        "discount": 0,_125        "categoryId": "1000148",_125        "sellerId": "1"_125      },_125      {_125        "id": "48760",_125        "name": "SellerY product",_125        "price": 19.2,_125        "quantity": 1,_125        "discount": 0,_125        "categoryId": "1000104",_125        "sellerId": "SellerY"_125      }_125    ],_125    "shippingValue": 39.32,_125    "taxValue": 0,_125    "url": "<https://admin.mystore.example.com/admin/checkout/#/orders?q=v22590454abc>"_125  },_125  "callbackUrl": "<https://api.example.com/some-path/to-notify/status-changes?an=mystore>",_125  "returnUrl": "<https://mystore.example.com/checkout/order/v22590454abc>",_125  "recipients": [_125    {_125      "id": "mystore",_125      "name": "Company XPTO",_125      "documentType": "CNPJ",_125      "document": "01239313000160",_125      "role": "marketplace",_125      "chargeProcessingFee": true,_125      "chargebackLiable": true,_125      "amount": 92.36_125    },_125    {_125      "id": "sellerX",_125      "name": "Company X",_125      "documentType": "CNPJ",_125      "document": "88888888000173",_125      "role": "seller",_125      "chargeProcessingFee": false,_125      "chargebackLiable": false,_125      "amount": 73.18,_125      "commissionAmount": 13.94_125    },_125    {_125      "id": "sellerY",_125      "name": "Company Y",_125      "documentType": "CNPJ",_125      "document": "99.999.999/0001-26",_125      "role": "seller",_125      "chargeProcessingFee": false,_125      "chargebackLiable": false,_125      "amount": 34.08,_125      "commissionAmount": 8.52_125    }_125  ]_125}'
For transactions where the split in authorization is disabled, there are no changes in the normal authorization payload.
Capture Stage
The list of recipients involved in the transaction will be sent in this stage when the split is configured for the capture step (according to the payment method).
Split example:
- The registered commission is 16% for the Marketplace when buying the product from sellerA.
- The cart is made up of products from sellerA.
- An order with the value of 45.00 is registered.
- The following containers will be sent with these values: a. 7.2 for the marketplace. b. 37.8 for seller A.
For transactions where the split in authorization is disabled, there are no changes in the normal catch payload.
See below the capture request for this split example:
_36curl --location --request POST 'https://{{providerApiEndpoint}}/payments/{{paymentId}}/settlements' \\_36--header 'X-VTEX-API-AppKey: {{X-VTEX-API-AppKey}}' \\_36--header 'X-VTEX-API-AppToken: {{X-VTEX-API-AppToken}}' \\_36--header 'Content-Type: application/json' \\_36--header 'Accept: application/json' \\_36--data-raw '{_36    "transactionId": "D3AA1FC8372E430E8236649DB5EBD08E",_36    "requestId": "2019-02-04T22:53:42-40000",_36    "paymentId": "F5C1A4E20D3B4E07B7E871F5B5BC9F91",_36    "value": 45.00,_36    "authorizationId": "5784589",_36    "tid": "5784589",_36    "recipients": [_36        {_36            "id": "mystore",_36            "name": "Company XPTO",_36            "documentType": "CNPJ",_36            "document": "05314972000174",_36            "role": "marketplace",_36            "chargeProcessingFee": true,_36            "chargebackLiable": true,_36            "amount": 7.2_36        },_36        {_36            "id": "sellerA",_36            "name": "Company ABC",_36            "documentType": "CNPJ",_36            "document": "24830098000172",_36            "role": "seller",_36            "chargeProcessingFee": true,_36            "chargebackLiable": true,_36            "amount": 37.8,_36            "commissionAmount": 7.2_36        }_36      ]_36}'
Complete Refund Stage
There is no change in the normal complete refund payload.
See below an example of the complete refund request:
_12curl --location --request POST 'https://{{providerApiEndpoint}}/payments/{{paymentId}}/refunds' \\_12--header 'X-VTEX-API-AppKey: {{X-VTEX-API-AppKey}}' \\_12--header 'X-VTEX-API-AppToken: {{X-VTEX-API-AppToken}}' \\_12--header 'Content-Type: application/json' \\_12--header 'Accept: application/json' \\_12--data-raw '{_12    "paymentId": "F5C1A4E20D3B4E07B7E871F5B5BC9F91",_12    "transactionId": "D3AA1FC8372E430E8236649DB5EBD08E",_12    "settleId": "Q5C1A4E20D3B4E07B7E871F5B5BC9F91",_12    "value": 45.0,_12    "requestId": "LA4E20D3B4E07B7E871F5B5BC9F91"_12}'
Partial Refund Stage
To allow the use of the split partial refund functionality, you must request the Partner Support team to enable the
acceptSplitPartialRefundparameter astrueduring the payment provider homologation process.
The “recipient’s” object array will contain the sellers involved in the chargeback. If the chargeback is only applicable for Marketplace items, the “recipient’s” object array will be sent containing only the Marketplace data.
Split example of a partial refund of a seller item:
- Sending the array of recipient objects with the Marketplace value to be refunded and the seller value to be refunded.
- The registered commission is 16% for the Marketplace when buying the product from sellerA.
- The cart was made up of items from sellerA.
- The value to be refunded from sellerA item is 20.00.
- The following containers will be sent with these values: a. 3.2 for the marketplace. b. 16.8 for seller A.
See below the partial refund request for this split example:
_31curl --location --request POST 'https://{{providerApiEndpoint}}/payments/{{paymentId}}/refunds' \\_31--header 'X-VTEX-API-AppKey: {{X-VTEX-API-AppKey}}' \\_31--header 'X-VTEX-API-AppToken: {{X-VTEX-API-AppToken}}' \\_31--header 'Content-Type: application/json' \\_31--header 'Accept: application/json' \\_31--data-raw '{_31  "requestId": "LA4E20D3B4E07B7E871F5B5BC9F91",_31  "settleId": "Q5C1A4E20D3B4E07B7E871F5B5BC9F91",_31  "paymentId": "F5C1A4E20D3B4E07B7E871F5B5BC9F91",_31  "tid": "10022005181543584603",_31  "value": 20.00,_31  "transactionId": "D3AA1FC8372E430E8236649DB5EBD08E",_31  "recipients": [_31    {_31      "id": "mystore",_31      "name": "Company XPTO",_31      "documentType": "CNPJ",_31      "document": "05314972000174",_31      "role": "marketplace",_31      "amount": 3.2_31    },_31    {_31      "id": "sellerA",_31      "name": "Company ABC",_31      "documentType": "CNPJ",_31      "document": "24830098000172",_31      "role": "seller",_31      "amount": 16.8_31    }_31  ]_31}'
Split example of a partial refund of a Marketplace item:
- Sending the array of recipient objects with the Marketplace value to be refunded.
- The value to be refunded from the Marketplace item is 20.00.
- A recipient object with a value of 20.00 will be sent to the Marketplace.
See below the partial refund request for this split example:
_23curl --location --request POST 'https://{{providerApiEndpoint}}/payments/{{paymentId}}/refunds' \\_23--header 'X-VTEX-API-AppKey: {{X-VTEX-API-AppKey}}' \\_23--header 'X-VTEX-API-AppToken: {{X-VTEX-API-AppToken}}' \\_23--header 'Content-Type: application/json' \\_23--header 'Accept: application/json' \\_23--data-raw '{_23  "requestId": "LA4E20D3B4E07B7E871F5B5BC9F91",_23  "settleId": "Q5C1A4E20D3B4E07B7E871F5B5BC9F91",_23  "paymentId": "F5C1A4E20D3B4E07B7E871F5B5BC9F91",_23  "tid": "10022005181543584603",_23  "value": 20.00,_23  "transactionId": "D3AA1FC8372E430E8236649DB5EBD08E",_23  "recipients": [_23    {_23      "id": "mystore",_23      "name": "Company XPTO",_23      "documentType": "CNPJ",_23      "document": "05314972000174",_23      "role": "marketplace",_23      "amount": 20.00_23    }  _23  ]_23}'
Cancellation Stage
In this version of the Split Payouts, there are no changes in the normal cancellation payload.
See below an example of the cancellation request:
_10curl --location --request POST 'https://{{providerBaseUrl}}/payments/{{paymentId}}/cancellations' \\_10--header 'X-VTEX-API-AppKey: {{X-VTEX-API-AppKey}}' \\_10--header 'X-VTEX-API-AppToken: {{X-VTEX-API-AppToken}}' \\_10--header 'Content-Type: application/json' \\_10--header 'Accept: application/json' \\_10--data-raw '{_10    "paymentId": "F5C1A4E20D3B4E07B7E871F5B5BC9F91",_10    "requestId": "D12D9B80972C462980F5067A3A126837"_10}'
For more information about payment splits on VTEX, see Split Payment.