Shipping module
Last updated: Jul 7th, 4:35am
Overview
The shipping module presents shipping details to a buyer during the Pay with PayPal and Pay with Venmo flow. The merchant has several options available for controlling how shipping addresses and shipping options are handled. The server-side shipping callbacks allow you to update the shipping and order amount information as buyers make changes on the review page.
Merchant and buyer interaction
Buyers can use the shipping module to specify the shipping address and shipping options on the review page. The system sends a callback to the merchant's URL with the updated shipping information.
When the buyer provides their shipping address, a callback is sent to the merchant server with the buyer's address using the server-side shipping callbacks. In response, the merchant can send the shipping options and updated order cost amounts.
You can use server-side callbacks to:
- Verify that you support the shipping method.
- Update shipping costs.
- Change line items in the cart.
- Inform the buyer that you do not support their shipping method.
While both server-side and client-side callbacks are possible, it is generally recommended to use server-side callbacks, as client-side callbacks may not be available in all situations. For example, client-side callbacks using JS-SDK are designed primarily for web-based integrations and are incompatible with Venmo. They may not be suitable for native mobile applications that often require different APIs or SDKs tailored for mobile platforms.
The following sample PayPal and Venmo review pages show a buyer's shipping address and options:
  
 
How it works
The shipping address and options callback process involves the following steps:
- A buyer clicks on the PayPal or Venmo button displayed on the merchant site.
- The merchant makes a Create Order API request to set up the flow, including the parameters needed to enable the server-side callbacks.
- The buyer is taken to the flow where they are authenticated and shown the review page.
- A server-side callback is created to the merchant's server with the default shipping address from the buyer's wallet.
- The merchant processes the callback and responds with shipping options for the address and updated order amount.
- The review page updates to reflect the merchant's revised order amount and shipping options.
- If the buyer changes the shipping address or shipping options, a callback is made to the merchant. The merchant's response updates the review page.
- The buyer clicks the pay button on the review page to complete the transaction, which is returned to the merchant.
- The merchant may make a Show Order Details API call to get the final order information and verify it still matches the information in their system.
- The merchant captures the order to complete the transaction.
Create order
Use the Orders v2 API Create Order request to set up the flow used by the buyer to approve a transaction. Both Pay with PayPal and Pay with Venmo support the shipping module. The request can utilize either payment_source:paypal or payment_source:venmo. There are several
    ways to determine how the shipping address and options are displayed on the review page and which events create callbacks when the buyer interacts with them.
Enabling server-side callbacks
The payment_source.*.experience_context.order_update_callback_config object is used to enable and configure server-side callbacks during the flow. The callback_url field specifies the endpoint on your
    servers where PayPal sends the callback request. The callback_events array specifies which callback events you support. The flow supports the following events:
- SHIPPING_ADDRESS: This event happens when the review page loads for the first time and when the buyer changes their shipping address. We recommend merchants to subscribe only to- SHIPPING_ADDRESS. This requires a merchant to respond with all the shipping options and amounts calculated on the initial callback. Since all options have been returned, the merchant callback does not have to be called when the selection option changes. This method reduces delays and the number of requests for PayPal.
- SHIPPING_OPTIONS: This event happens when the buyer selects new shipping options and when they change shipping options. We recommend merchants subscribe to this if they want to be notified or need to recalculate amounts with their callback URL when a selected shipping option changes for the current address. If- SHIPPING_OPTIONSis not needed by the merchant, they should subscribe to- SHIPPING_ADDRESS.
Configuring shipping preferences
The shipping preference is set using the payment_source.*.experience_context.shipping_preference field. There are 3 possible values:
- GET_FROM_FILE: This default preference retrieves the customer-provided shipping address their wallet.
- NO_SHIPPING: This option removes the shipping address information from the review page and the API response. However, the- shipping.phone_numberand- shipping.email_addressfields are returned for digital goods delivery.
- SET_PROVIDED_ADDRESS: The review page shows the shipping address provided by the merchant in the create order request. The buyer cannot change this address on the review page. If the merchant does not pass an address, the buyer can choose the address on PayPal.
If you do not pass a shipping preference, the default behavior is GET_FROM_FILE. The review page shows the default shipping address from the buyer's wallet. The buyer can select a different shipping address from
    their wallet or add a new address. Callbacks happen if the shipping preference is GET_FROM_FILE.
Passing a shipping address
If you have a shipping address from the buyer, pass it to Pay with PayPal or Pay with Venmo on the create order request. If you pass a shipping address with a shipping preference of GET_FROM_FILE, it will display on the review page, where the buyer can change the shipping address. You can use the shipping preference SET_PROVIDED_ADDRESS to prevent the buyer from changing the shipping address on the review page. 
If your transaction does not involve shipping, you can pass the shipping preference NO_SHIPPING and no shipping information will be displayed on the review page.
Here is a sample Create Order API request to a merchant with a callback config for Pay with PayPal (payment_source:paypal) and Pay with Venmo (payment_source:venmo).
- Pay with PayPal
- Pay with Venmo
1curl -v -X POST https://api-m.sandbox.paypal.com/v2/checkout/orders \2-H 'Content-Type: application/json' \3-H 'PayPal-Request-Id: 7b92603e-77ed-4896-8e78-5dea2050476a' \4-H 'Authorization: Bearer 6V7rbVwmlM1gFZKW_8QtzWXqpcwQ6T5vhEGYNJDAAdn3paCgRpdeMdVYmWzgbKSsECednupJ3Zx5Xd-g' \5-d '{6 "intent": "CAPTURE", "payment_source": {7 "paypal": {8 "experience_context": {9 "user_action": "PAY_NOW",10 "shipping_preference": "GET_FROM_FILE",11 "return_url": "https://example.com/returnUrl",12 "cancel_url": "https://example.com/cancelUrl",13 "order_update_callback_config": {14 "callback_events": ["SHIPPING_ADDRESS", "SHIPPING_OPTIONS"],15 "callback_url": "https://example.com/orders?cart_id=h98h98h&session_id=89h788fg8"16 }17 }18 }19 }, "purchase_units": [{20 "reference_id": "d9f80740-38f0-11e8-b467-0ed5f89f718b",21 "items": [{22 "name": "T-Shirt",23 "description": "Super Fresh Shirt",24 "unit_amount": {25 "currency_code": "USD",26 "value": "50.00"27 },28 "quantity": "1",29 "category": "PHYSICAL_GOODS",30 "sku": "sku01",31 "image_url": "https://example.com/static/images/items/1/tshirt_green.jpg",32 "url": "https://example.com/url-to-the-item-being-purchased-1",33 "upc": {34 "type": "UPC-A",35 "code": "123456789012"36 }37 }, {38 "name": "Shoes",39 "description": "Running, Size 10.5",40 "sku": "sku02",41 "unit_amount": {42 "currency_code": "USD",43 "value": "25.00"44 },45 "quantity": "2",46 "category": "PHYSICAL_GOODS",47 "image_url": "https://example.com/static/images/items/1/shoes_running.jpg",48 "url": "https://example.com/url-to-the-item-being-purchased-2",49 "upc": {50 "type": "UPC-A",51 "code": "987654321012"52 }53 }],54 "amount": {55 "currency_code": "USD",56 "value": "100.00",57 "breakdown": {58 "item_total": {59 "currency_code": "USD",60 "value": "100.00"61 }62 }63 }64 }]65 }
Server-side shipping callbacks
The Server-Side Callback API request to a merchant is consistent with GET orders and includes selected addresses and options outside of the order object.
- Sending order data is structurally similar to making a GET request.
- Updated shipping addresses and options separately sent from the order to avoid confusion.
- If the address or option is incomplete, it will not be set in order to prevent misunderstandings.
- The cart identifier will be embedded in the callback URL if merchants cannot associate their shopping cart with the order.
- Merchants with client-side shopping carts should include item details when they create the order for server-side shipping callbacks.
Callback request to merchant for shipping address event
The following sample is a server-side callback request to the merchant for a shipping event. The callback identifies the order, shipping_address, shipping_option, and purchase_units.
The initial callback to the merchant does not include the "shipping_option" section.
1{2 "id": "5O190127TN364715T",3 "shipping_address": {4 "country_code": "US",5 "admin_area_1": "TX",6 "admin_area_2": "Dallas",7 "postal_code": "75001"8 },9 "shipping_option": {10 "id": "2",11 "amount": {12 "currency_code": "USD",13 "value": "20.00"14 },15 "type": "SHIPPING",16 "label": "Free Shipping"17 },18 "purchase_units": [{19 "reference_id": "d9f80740-38f0-11e8-b467-0ed5f89f718b",20 "amount": {21 "currency_code": "USD",22 "value": "100.00"23 }24 }]25}
Callback request fields
| Field | Description | Data type | 
|---|---|---|
| 
 | Unique identifier for the shipping order. For example,  | string | 
| 
 | Shipping address information. Includes  | object | 
| 
 | Identifies the shipping address country. For example,  | string | 
| 
 | Identifies the shipping address state or region. For example,  | string | 
| 
 | Identifies the shipping address, city, or municipality. For example,  | string | 
| 
 | Identifies the postal code. For example,  | number | 
| 
 | Shipping option information. Includes  | string | 
| 
 | Determines which option is selected by default. Possible values are trueorfalse | boolean | 
| 
 | Indicates the type of shipping option. For example,  | number | 
| 
 | Shipping option amount. Includes  | object | 
| 
 | Indicates the type of currency. For example,  | string | 
| 
 | String denoting the purchase price and shipping option price, formatted as  | number | 
| 
 | An enum that indicates the type of shipping. For example,  | string | 
| 
 | Dropdown menu option selected by the buyer. Indicates whether the shipping is free or not. For example,  | string | 
| 
 | Defines the purchase unit. Includes  | object | 
| 
 | Identifier associated with the purchase units. For example,  | string | 
Merchant response
The merchant response to a server-side callback request is similar to an order. A change to the customer’s shipping address or options can affect the following fields within the data structure:
- purchase_units[].shipping.optionsrepresent the shipping modules associated with the purchase units within the order. The purchase order field includes contact and delivery information, including phone number, account email address, order ID, and reference ID.
- purchase_units[].amountrepresents the purchase unit amount in the order.
Buyer changes can be managed through one of the following approaches:
- Asking the merchant to return an updated order limited to the fields that can be changed.
- Asking the merchant to return a list of patch operations.
Merchant success response
When the order request is acceptable, the merchant responds with a success message that contains amount and shipping option changes.
Here is a sample HTTP 200 OK callback response from a merchant with shipping options and amount updates.
1{2 "id": "8HFTASDATTV",3 "purchase_units": [{4 "reference_id": "d9f80740-38f0-11e8-b467-0ed5f89f718b",5 "amount": {6 "currency_code": "USD",7 "value": "105.00",8 "breakdown": {9 "item_total": {10 "currency_code": "USD",11 "value": "100.00"12 },13 "tax_total": {14 "currency_code": "USD",15 "value": "5.00"16 },17 "shipping": {18 "currency_code": "USD",19 "value": "0.00"20 }21 }22 },23 "shipping_options": [{24 "id": "1",25 "amount": {26 "currency_code": "USD",27 "value": "0.00"28 },29 "type": "SHIPPING",30 "label": "Free Shipping",31 "selected": true32 }, {33 "id": "2",34 "amount": {35 "currency_code": "USD",36 "value": "7.00"37 },38 "type": "SHIPPING",39 "label": "USPS Priority Shipping",40 "selected": false41 }, {42 "id": "3",43 "amount": {44 "currency_code": "USD",45 "value": "10.00"46 },47 "type": "SHIPPING",48 "label": "1-Day Shipping",49 "selected": false50 }]51 }]52}
Merchant success response fields
A merchant response can include the fields identified in the following table.
| Field | Description | Date type | 
|---|---|---|
| 
 | Identifies the merchant. For example,  | string | 
| 
 | Purchase unit information. Includes  | object | 
| 
 | Uniquely identifies the purchase transaction. For example,  | string | 
| 
 | Purchase unit amount in the order. Includes  | object | 
| 
 | Code indicating the amount currency type. For example,  | string | 
| 
 | String denoting the purchase unit amount in the order, formatted as  | amount | 
| 
 | Amount breakdown. Includes  | object | 
| 
 | Amount item breakdown. Includes  | object | 
| 
 | Amount item value. For example,  | number | 
| 
 | Amount tax breakdown. Includes  | object | 
| 
 | String denoting the | number | 
| 
 | Amount shipping breakdown. Includes  | object | 
| 
 | String denoting the value. amount, formatted as  In this example, there are no shipping costs associated with the purchase unit. | number | 
| 
 | Shipping option information. Includes  | object | 
| 
 | Determines which options are selected by default. Possible values are trueorfalse. | boolean | 
| 
 | Shipping option identifier. For example,  | number | 
| 
 | Amount for the selected shipping option. For example,  | object | 
| 
 | String denoting the value. amount for the selected shipping option, formatted as  | number | 
| 
 | An enum that indicates the type of delivery for the selected option. For example,  | string | 
| 
 | Dropdown menu option selected by the buyer. Indicates the type of  | string | 
The order amount automatically updates based on the options selected by buyers unless you are subscribed to the shipping_options callback. If you are subscribed to shipping_options, ensure the callback response includes the updated total cost so that the checkout page can be updated. You can use the onShippingOptionsChange() callback to update your own database with the new cost amount.
Merchant decline response
A merchant can issue a 422 error to decline a callback event. In the response, the merchant should provide a reason why the callback is being declined. The following reasons are supported:
| Callback event type | Error | Description | 
|---|---|---|
| Shipping address | 
 | Your order can't be shipped to this address. | 
| Shipping address | 
 | Your order can't be shipped to this country. | 
| Shipping address | 
 | Your order can't be shipped to this state. | 
| Shipping address | 
 | Your order can't be shipped to this zip. | 
| Shipping option | 
 | The shipping method you selected is unavailable. To continue, choose another way to get your order. | 
| Shipping option | 
 | Part of your order isn't available at this store. | 
Here is a sample decline callback from the merchant. In this case, there is an HTTP 422 Unprocessable Entity response, which is a country error.
1{2 "name": "UNPROCESSABLE_ENTITY",3 "details": [{4 "issue": "COUNTRY_ERROR"5 }]6}