# APM payments

> Learn how to accept APM payments through SDK integration.

 export const AlertDeprecate = () => (  <svg aria-hidden="true" focusable="false" width="32" height="32" style={{
  marginRight: 16,
  marginTop: 2,
  flexShrink: 0
}} fill="#d97706" // Darker yellow icon
viewBox="0 0 24 24">
      <circle cx="12" cy="12" r="12" fill="#ffe066" /> {/* Lighter yellow circle */}
      <path d="M12 8v4" stroke="#d97706" strokeWidth="2" strokeLinecap="round" />
      <circle cx="12" cy="16" r="1" fill="#d97706" />
    </svg>   Deprecation Notice:       ); <AlertDeprecate>
  The content of this guide has been deprecated.<br />
  We recommend you adopt the <Link href="https://docs.antom.com/ac/cashierpay/element?platform=web-wap">Payment Element integration</Link> for improved experience.
</AlertDeprecate> Antom SDK is a pre-built UI component that simplifies the process of integrating Antom payment services. This component offers a simple and quick way to integrate all payment methods supported by Antom, including credit cards, digital wallets, bank transfers, online banking, and more.

 The SDK component can automatically adapt to the buyer's device and location, performing tasks such as information collection, redirection, switching between apps, displaying QR codes, and more based on the characteristics of the payment method. You can easily provide a comprehensive payment solution on your website or application, saving a significant amount of time and resources required to build each payment API and process.

  

<!-- ToggleTab query="platform" -->

**Tab: Web/WAP**

# User experience {#BkD7T1}

 The following figures show the user journey of paying on a shopping website or mobile web app:

<!-- TabGroup -->

**Tab: Web**

![Web user experience showing APM payment flow via Antom SDK on a shopping website](https://idocs-assets.marmot-cloud.com/storage/idocs87c36dc8dac653c1/yuque/idocs/2025/png/dd5215de-99bc-42aa-b375-706d6d5f6131.png)

**Tab: WAP**

![WAP user experience showing APM payment flow via Antom SDK on a mobile web app](https://idocs-assets.marmot-cloud.com/storage/idocs87c36dc8dac653c1/yuque/idocs/2025/png/2e5e855a-fb1d-43de-849b-23fb2ca42a73.png)

<!-- /TabGroup -->

# Payment flow {#d2GKE1}

 For each payment method, the payment flow is composed of the following steps:

 ![APM payment flow showing the interaction across buyer, merchant client, Antom SDK, bank App, merchant server, and Antom server](https://idocs-assets.marmot-cloud.com/storage/idocs87c36dc8dac653c1/yuque/idocs/2025/png/70a0e1a6-0b2b-4926-b6a3-289cf49f3ae1.png)

 1. **The buyer lands on the checkout page.**
2. **Create a payment session request.**  
   After the buyer selects a payment method and submits the order, you can obtain the payment session by calling the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  .
3. **Invoke the client SDK.**  
   On the client side, invoke the SDK through the payment session. The SDK will handle information collection, redirection, app invocation, QR code display, verification, and other processes based on the payment method's characteristics. Then the SDK will callback the payment result via   `onEventCallback`   or redirect to your payment result page.
4. **Confirm the payment result.**  
   Obtain the payment result by using one of the following two methods:

   - Asynchronous notification: Specify     *paymentNotifyUrl*     in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API to set the address for receiving asynchronous notifications. When the payment is successful or expires, Antom will use  [**notifyPayment**](https://docs.antom.com/ac/ams/paymentrn_online.md)  to send asynchronous notifications to you.
  - Synchronous inquiry: Call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API to check the payment status.

# Integration preparations {#uEXVe}

 Before you start integrating, read the   [Integration guide](https://docs.antom.com/integration_guide_en.md)   and   [API overview](https://docs.antom.com/ac/ams/api_fund.md)   to understand the integration steps of the server-side API and the precautions for calling the API. Furthermore, ensure the following prerequisites are met:

 - Obtained your client ID
- Complete the key configuration
- Complete the configuration of     *paymentNotifyUrl*     to receive the asynchronous notification

 - Integrate the server-side SDK package, install the server-side library, and initialize a request instance. For more details, refer to   [Server-side SDKs](https://docs.antom.com/ac/sdks/server_sdks.md)  .
- Integrate the client-side SDK package by following the steps of   [Integrate the SDK package for Web/WAP](https://docs.antom.com/ac/sdks/web.md)  .

# Integration steps {#AkDW7T}

 Start your integration by taking the following steps:

 1. Create a payment session
2. Create and invoke the SDK
3. Obtain payment result

## Step 1: Create a payment session  **[Server-side]** {#zhBSkk1}

 When a buyer selects a payment method provided by Antom, you need to collect key information such as the payment request ID, order amount, payment method, order description, payment redirect URL, and payment result notification URL, call the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API to create a payment session, and return the payment session to the client.

 Creating a payment session includes the following parameters:

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *productCode* | Yes | The value is fixed to  `CASHIER_PAYMENT`  . |
| *paymentRequestId* | Yes | The unique ID generated by the merchant, a new ID is required every time a payment is initiated. |
| *paymentAmount* | Yes | Payment amount, should be set according to the smallest unit of the order currency, e.g. CNY for cents, KRW for dollars. |
| *paymentMethod* | Yes | Payment method enumeration. |
| *paymentRedirectUrl* | Yes | Merchant's payment result page, according to the server-side results, not a fixed success page. |
| *order* | Yes | Order information, including order amount, order ID, and order description on the merchant side. |
| *paymentNotifyUrl* | No | Payment result notification address, which can be passed in through the API or set as a fixed value through Antom Dashboard. |
| *settlementStrategy* | No | The settlement strategy for the payment request. |
| *paymentFactor.isAuthorization* | No | Indicates whether the payment scenario requires authorization. Valid values are: - `false`  : Default value. Indicates that the payment scenario is a regular payment without authorization. - `true`  : Indicates that the payment scenario requires authorization.   Under the authorization scenario, the payment funds are guaranteed and held on the payment method side. You can use the        [**capture (One-time Payments)**](https://docs.antom.com/ac/ams/capture.md)        API to deduct the payment funds.   > **[INFO]** **Note**    : For APM payments, ensure this parameter is set to   `false`   or left empty to avoid the automatic release of funds to the buyer. |

 The above parameters are the basic parameters for creating a payment session, for full parameters and additional requirements for certain payment methods please refer to  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)   **.**

 The following sample code shows how to call the      [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)      API:

   ```java
private static void createPaymentSession() {
    AlipayPaymentSessionRequest alipayPaymentSessionRequest = new AlipayPaymentSessionRequest();
    alipayPaymentSessionRequest.setProductCode(ProductCodeType.CASHIER_PAYMENT);

    // replace with your paymentRequestId
    String paymentRequestId = UUID.randomUUID().toString();
    alipayPaymentSessionRequest.setPaymentRequestId(paymentRequestId);

    // set amount
    Amount amount = Amount.builder().currency("SGD").value("4200").build();
    alipayPaymentSessionRequest.setPaymentAmount(amount);

    // set paymentMethod
    PaymentMethod paymentMethod = PaymentMethod.builder().paymentMethodType("SHOPEEPAY_SG").build();
    alipayPaymentSessionRequest.setPaymentMethod(paymentMethod);

    // replace with your orderId
    String orderId = UUID.randomUUID().toString();

    // set buyer info
    Buyer buyer = Buyer.builder().referenceBuyerId("yourBuyerId").build();

    // set order info
    Order order = Order.builder().referenceOrderId(orderId)
        .orderDescription("antom testing order").orderAmount(amount).buyer(buyer).build();
    alipayPaymentSessionRequest.setOrder(order);

    // replace with your notify url
    alipayPaymentSessionRequest.setPaymentNotifyUrl("https://www.yourNotifyUrl.com");

    // replace with your redirect url
    alipayPaymentSessionRequest.setPaymentRedirectUrl("https://www.yourMerchantWeb.com");

    AlipayPaymentSessionResponse alipayPaymentSessionResponse = null;
    try {
        alipayPaymentSessionResponse = CLIENT.execute(alipayPaymentSessionRequest);
    } catch (AlipayApiException e) {
        String errorMsg = e.getMessage();
        // handle error condition
    }
}

```

   The following code shows a sample of the request message:

   ```json
{
  "order": {
    "buyer": {
      "referenceBuyerId": "yourBuyerId"
    },
    "orderAmount": {
      "currency": "SGD",
      "value": "4200"
    },
    "orderDescription": "antom testing order",
    "referenceOrderId": "referenceOrderId01"
  },
  "paymentAmount": {
    "currency": "SGD",
    "value": "4200"
  },
  "paymentMethod": {
    "paymentMethodType": "SHOPEEPAY_SG"
  },
  "paymentNotifyUrl": "https://www.yourNotifyUrl.com",
  "paymentRedirectUrl": "https://www.yourMerchantWeb.com",
  "paymentRequestId": "paymentRequestId01",
  "productCode": "CASHIER_PAYMENT"
}

```

   The following code shows a sample of the response, which contains the following parameters:

 - *paymentSessionData*    : the payment session data to be returned to the frontend.
- *paymentSessionExpiryTime*    : the expiration time of the payment session.

   ```json
{
  "paymentSessionData": "gM5Y/Go6+IXMPSOZrYtukCdodtwxuSFsn1vRPSB8CzkDqlb1g2W7dDipOkoMRs/x2W7g7qVhjEM7RUGroIT8Cw==&&SG&&188&&eyJhY3Rpb24iOnsiYXBwIjp7fSwibmVlZFZlcmlmeUFuZFJlc3VtZSI6ZmFsc2UsInNraXBTZGtRdWVyeSI6ZmFsc2UsIndhcCI6e30sIndlYiI6e319LCJjbGllbnRJZCI6IlNBTkRCT1hfNVlCWjBMMlpIUjhSMDg5NzgiLCJleHRlbmRJbmZvIjoie1widmVyc2lvbk1hcFwiOntcIndlYlwiOntcIjEuMS4wXCI6e1widGFyZ2V0V2ViVmVyaXNvblwiOlwiMS4xLjBcIn0sXCIxLjIuMFwiOntcInRhcmdldFdlYlZlcmlzb25cIjpcIjEuMi4wXCJ9fSxcImlPU1wiOntcIjEuMS4wXCI6e1widGFyZ2V0V2ViVmVyaXNvblwiOlwiMS4xLjBcIn0sXCIxLjIuMFwiOntcInRhcmdldFdlYlZlcmlzb25cIjpcIjEuMi4wXCJ9fSxcIkFuZHJvaWRcIjp7XCIxLjEuMFwiOntcInRhcmdldFdlYlZlcmlzb25cIjpcIjEuMS4wXCJ9LFwiMS4yLjBcIjp7XCJ0YXJnZXRXZWJWZXJpc29uXCI6XCIxLjIuMFwifX19fSIsInBheW1lbnRTZXNzaW9uQ29uZmlnIjp7InBheW1lbnRNZXRob2RDYXRlZ29yeVR5cGUiOiJBUE0iLCJwcm9kdWN0U2NlbmUiOiJDQVNISUVSX1BBWU1FTlQiLCJwcm9kdWN0U2NlbmVWZXJzaW9uIjoiMS4wIn19",
  "paymentSessionExpiryTime": "2024-01-01T00:00:00+08:00",
  "paymentSessionId": "gM5Y/Go6+IXMPSOZrYtukCdodtwxuSFsn1vRPSB8Czn6HfgqNzo2ogbIk1NUQS35",
  "result": {
    "resultCode": "SUCCESS",
    "resultMessage": "success.",
    "resultStatus": "S"
  }
}

```

   > **[INFO]** **Common questions**
>
>  **Q: Can I use Chinese characters in the value of the request parameters?**
>
>  A: To avoid incompatibility of a certain payment method, do not use Chinese characters for fields in the request.
>
>  ​
>
>  **Q: How to set the URL to receive the payment notification?**
>
>  A: Specify     *paymentNotifyUrl*     in the      [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)      API to receive the asynchronous notification about the payment result (  [**notifyPayment**](https://docs.antom.com/ac/ams/paymentrn_online.md)  ), or configure the receiving URL in Antom Dashboard. If the URL is specified in both the request and Antom Dashboard, the value specified in the request takes precedence.

## Step 2: Create and invoke the SDK  **[Client-side]** {#oDHQJ1}

 The Antom SDK is a component used for handling payment processes. Initiate the SDK by creating a payment session to collect information, switch between apps, and display QR codes based on the payment method specified in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API.

 After the buyer selects a payment method on the page, you need to create the SDK and initiate it with a payment session.

### 1\. (Optional) Preload the SDK {#WrJee8}

 Before creating a payment session, it is highly recommended that you preload the SDK to improve the rendering speed of the checkout page, reducing the waiting time for buyers during the payment. Follow the code example below to perform the preloading:

 ```javascript
// import { AMSCashierPayment } from '@alipay/ams-checkout';

AMSCashierPayment.preload();

```

### 2\. Instantiate the SDK {#KQO5X1}

 Create the SDK instance by using the  `AMSCashierPayment`  and specify basic configurations. Configuration objects includes the following parameters:

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *environment* | Yes | It is used to pass in environmental information. Valid values are: - `sandbox`  : Sandbox environment - `prod`  : Production environment |
| *locale* | No | It is used to pass in language information. Valid values are listed as follows. You can choose the value to pass based on the region of the payment method. If other values are passed, the local language is used by default: - `en_US`  : English - `pt_BR`  : Portuguese - `ko_KR`  : Korean - `es_ES`  : Spanish - `ms_MY`  : Malay - `in_ID`  : Indonesian - `tl_PH`  : Tagalog - `th_TH`  : Thai - `vi_VN`  : Vietnamese - `fr_FR`  : French - `nl_NL`  : Dutch - `it_IT`  : Italian - `de_DE`  : German - `zh_CN`  : Simplified Chinese - `zh_HK`  : Traditional Chines |
| *analytics* | No | It is used to configure and analyze data. It contains a value below: - *enabled*    : Optional Boolean. It defaults to true, which indicates that you allow the SDK to upload and analyze operational data to deliver a better service. If you do not allow the data to be uploaded and analyzed, specify it as  `false`  . |
| *onLog* | No | It is a callback method that is used to generate the error information about logs and API exceptions during the execution of the SDK. |
| *onEventCallback* | No | A callback function returns a specific event code when a payment event happens during SDK runtime, like payment result or a form submission error. For further information, please refer to the  [References](https://docs.antom.com/ac/sdks/ref.md)  . |

   The following sample code shows how to instantiate the SDK:

   ```javascript
// Import { AMSCashierPayment } from '@alipay/ams-checkout'

const checkoutApp = new window.AMSCashierPayment({
  environment: "sandbox",
  locale: "en_US",
  onLog: ({code, message}) => {},
  onEventCallback: ({code, message}) => {},
});

// Request the server to get paymentSessionData
```

   > **[INFO]** **Note**    : Each instance can handle only one payment session. It is recommended to create the instance before calling   `createComponent`   each time.

### 3\. Invoke the SDK {#iVDpG1}

 Use the   `createComponent`   function in the instance object to create a payment component:

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *sessionData* | Yes | Create a configuration object by using the     *sessionData*     parameter: Pass the complete data in the     *paymentSessionData*     parameter obtained in the response through the   **[**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)**   API to the     *sessionData*     parameter. |

 The following sample code shows how to call   `createComponent`  :

 ```javascript
async function create(sessionData) {
  await checkoutApp.createComponent({
    sessionData: sessionData,
  });
}
```

   Call the   `unmount`   method to free SDK component resources in the following situations:

 - When the buyer switches views to exit the checkout page, free the component resources created in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  .
- When the buyer initiates multiple payments, free the component resources created in the previous  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  .
- When the buyer completes the payment and sets     *notRedirectAfterComplete*     to true, free the component resources after obtaining specific payment result codes.

 ```javascript
// Free SDK component resources
checkoutApp.unmount();
```

   > **[INFO]** **Common questions**
>
>  **Q: What can I do when I receive SDK_CREATEPAYMENT_PARAMETER_ERROR?**
>
>  A: When you receive this event code, check if the     *sessionData*     passed in is correct and complete.
>
>  
>  **Q: What can I do when I receive SDK_PAYMENT_ERROR or a rendering view error occurred?**
>
>  A: Check the network request for any exceptions during API initialization, such as network timeouts. Ensure that the environment for creating payment session request is consistent with the environment used for SDK instantiation. Check whether the parameters in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API are passed correctly. If the API exceptions persist, feel free to contact us for further troubleshooting.

### 4\. Display payment results {#Go6NO1}

 You can use the payment component to handle payment methods which require a redirect (asynchronous payments), and those that do not (synchronous payments).

#### Redirect payment {#6ubRS1}

 The payment process of some payment methods needs to leave the original web page and jump to the organization's page to complete the payment, such as the payment method of online banking. In this case, the buyer is redirected to the     *paymentRedirectUrl*     that you provided in the      [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)      API after completing the payment. You can obtain the payment result by active query and and show it to the buyer in that URL.

#### Non-redirect payment {#NEgUb1}

 For payment without redirection, the payment result will be given through the   `onEventCallback`   function. The payment result here is only for front-end display, and the final order status is subject to the server side. You need to customize the processing flow you want for each payment result through the data in the result of   `onEventCallback`  .

   The following are the possible event codes of the payment result returned by   `onEventCallback`  .

   | **Event code** | **Message** | **Solution** |
| --- | --- | --- |
| SDK_PAYMENT_SUCCESSFUL | Payment was successful. | Redirect the buyer to the payment result page, then confirm the payment result with the server. |
| SDK_PAYMENT_PROCESSING | Payment is being processed. | Confirm the payment result with the server. If the payment is successful, redirect the buyer to the payment result page. If the payment fails, guide the buyer to retry payment. |
| SDK_PAYMENT_FAIL | Payment failed. | Check the value of          *result.paymentResultCode*     in the   `onEventCallback`   result data for details. Guide the buyer to retry payment based on the obtained information. > **[INFO]** **Note**    : For the meaning of     *result.paymentResultCode*    , refer to   [Description of payment failure or abnormal payment status](https://docs.antom.com/ac/sdks/ref.md?locale=en-us#nuduf)  . |
| SDK_PAYMENT_CANCEL | Payment was canceled. | The SDK can be re-invoked with a     *paymentSessionData*     within the validity period; if it has expired, the     *paymentSessionData*     needs to be re-requested. |
| SDK_PAYMENT_ERROR | The payment status was abnormal. | Confirm the payment result with the server.    If it is confirmed that the payment has failed, guide the buyer to    retry payment   . |

 The following sample code shows how to process the   `onEventCallback`  :

 ```javascript
function onEventCallback({ code, result }) {
  switch (code) {
    case 'SDK_PAYMENT_SUCCESSFUL':
      // Payment was successful. Redirect the buyer to the payment result page, then confirm the payment result with the server.
      break;
    case 'SDK_PAYMENT_PROCESSING':
      console.log('Check the payment result data', result);
      // Payment is being processed. Confirm the payment result with the server. If the payment is successful, redirect the buyer to the payment result page. If the payment fails, guide the buyer to retry payment.
      break;
    case 'SDK_PAYMENT_FAIL':
      console.log('Check the payment result data', result);
      // Payment failed. Check the value of result.paymentResultCode in the onEventCallback result data for details. Guide the buyer to retry payment based on the obtained information.
      break;
    case 'SDK_PAYMENT_CANCEL':
      // Payment was canceled. The SDK can be re-invoked with a paymentSessionData within the validity period; if it has expired, the paymentSessionData needs to be re-requested.
      break;
    case 'SDK_PAYMENT_ERROR':
      console.log('Check the payment result data', result);
      // The payment status was abnormal. Confirm the payment result with the server. If it is confirmed that the payment has failed, guide the buyer to retry payment.
      break;
    default:
      break;
  }
}
```

## Step 3: Obtain payment result  **[Server-side]** {#SnQk71}

 After the buyer completes the payment or the payment times out, Antom sends the corresponding payment results to you through server interactions, you can obtain the payment result by one of the following methods:

 - Receive the asynchronous notification
- Inquire about the result

### Receive the asynchronous notification {#WDV7j1}

 When the payment reaches a final status of success or failure, Antom sends an asynchronous notification to the     *paymentNotifyUrl*     specified in the      [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)      API through the  [**notifyPayment**](https://docs.antom.com/ac/ams/paymentrn_online.md)  API. When you receive the notification from Antom, you are required to return a response as instructed in  [Return a receipt acknowledgment message](https://docs.antom.com/ac/cashierpay/notifications.md#vQK5A)  .

 Antom allows you to specify the URL in     *paymentNotifyUrl*     parameter within the      [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)      API. If the URL for each payment is the same, you can also configure it in the Antom Dashboard.

   The following code shows a sample of the notification request:

   ```json
{
  "notifyType": "PAYMENT_RESULT",
  "result": {
    "resultCode": "SUCCESS",
    "resultStatus": "S",
    "resultMessage": "success"
  },
  "paymentRequestId": "paymentRequestId01",
  "paymentId": "20240101123456789XXXX",
  "paymentAmount": {
    "value": "4200",
    "currency": "SGD"
  },
  "paymentCreateTime": "2024-01-01T00:00:00+08:00",
  "paymentTime": "2024-01-01T00:01:00+08:00"
}

```

   How to verify the signature of the notification and make a response to the notification, see  [Sign a request and verify the signature](https://docs.antom.com/ac/ams/digital_signature.md)  .

   > **[INFO]** **Common questions**
>
>  **Q: When will the notification be sent?**
>
>  A: It depends on whether the payment is completed:
>
>  - If the payment is successfully completed, Antom will usually send you an asynchronous notification within 3 to 5 seconds. For some payment methods like cash payment, the notification might take a bit longer.
> - If the payment is not completed, Antom needs to close the order first before sending an asynchronous notification. The time it takes for different payment methods to close the order varies, usually defaulting to 14 minutes.
>
>   **Q: Will the asynchronous notification be re-sent?**
>
>  A: Yes, the asynchronous notification will be re-sent automatically within 24 hours for the following cases:
>
>  - If you didn't receive the asynchronous notification due to network reasons.
> - If you receive an asynchronous notification from Antom, but you didn't make a response to the notification in the  [Sample code](https://docs.antom.com/ac/cashierpay/notifications.md#vQK5A)  format.
>
>  The notification can be resent up to 8 times or until a correct response is received to terminate delivery. The sending intervals are as follows: 0 minutes, 2 minutes, 10 minutes, 10 minutes, 1 hour, 2 hours, 6 hours, and 15 hours.
>
>  
>  **Q: When responding to asynchronous notification, do I need to add a digital signature?**
>
>  A: If you receive an asynchronous notification from Antom, you are required to return the response in the  [Sample code](https://docs.antom.com/ac/cashierpay/notifications.md#vQK5A)  format   , but you do not need to countersign the response.
>
>  
>  **Q: What are the key parameters in the notification that I need to use?**
>
>  A: Pay attention to the following key parameters:
>
>  - *result*    : indicates the payment result of the order.
> - *paymentRequestId*    : indicates the payment request number you generated for consult, cancel, and reconciliation.
> - *paymentId*    : indicates the payment order number generated by Antom used for refund and reconciliation.
> - *paymentAmount*    : indicates the payment amount.

### Inquire about the result {#wlAb81}

 You can call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API to initiate a query on the result of an order.

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *paymentRequestId* | No | The payment request ID generated by the merchant. |

 The parameter is not a full set of parameters, please refer to the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)      API for full set of parameters and additional requirements for certain payment methods.

   The following sample code shows how to call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API:

 ```java
public static void inquiryPayment() {
    AlipayPayQueryRequest alipayPayQueryRequest = new AlipayPayQueryRequest();

    // replace with your paymentRequestId
    alipayPayQueryRequest.setPaymentRequestId("yourPaymentRequestId");

    AlipayPayQueryResponse alipayPayQueryResponse = null;
    try {
        alipayPayQueryResponse = CLIENT.execute(alipayPayQueryRequest);
    } catch (AlipayApiException e) {
        String errorMsg = e.getMessage();
        // handle error condition
    }
}

```

   The following code shows a sample of the request message:

   ```json
{
  "paymentRequestId": "paymentRequestId01"
}

```

   The following code shows a sample of the response message:

   ```json
{
  "result": {
    "resultCode": "SUCCESS",
    "resultStatus": "S",
    "resultMessage": "Success"
  },
  "paymentStatus": "SUCCESS",
  "paymentRequestId": "paymentRequestId01",
  "paymentId": "20240101123456789XXXX",
  "paymentAmount": {
    "value": "4200",
    "currency": "SGD"
  },
  "paymentCreateTime": "2024-01-01T00:00:00+08:00",
  "paymentTime": "2024-01-01T00:01:00+08:00",
  "paymentMethodType": "SHOPEEPAY_SG"
}

```

   > **[INFO]** **Common questions**
>
>  **Q: How often should I call the inquiryPayment API?**
>
>  A: Call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API constantly with an interval of 2 seconds until the final payment result is obtained or an asynchronous payment result notification is received.
>
>  
>  **Q: What are the key parameters in the notification that I need to use?**
>
>  A: Pay attention to these key parameters:
>
>  - *result*    : represents the result of this  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API call, the result of the order needs to be judged according to     *paymentStatus*     :
>
>    - `SUCCESS`   and   `FAIL`   represent the final result.
>   - `PROCESSING`   represents the processing.
>
>  - *paymentAmount*    : indicates the payment amount.

# Sample codes {#xjYO01}

 Full sample codes for front-end:

 ```javascript
import { AMSCashierPayment } from '@alipay/ams-checkout'

// (Optional) Preload the SDK. Before creating a payment session, preload the SDK to improve the rendering speed of the checkout page, reducing the waiting time for buyers during the payment.
AMSCashierPayment.preload();

// Listen for the submit event on your web checkout counter.
document
  .querySelector("button#submit")
  .addEventListener("click", handleSubmit);

async function handleSubmit() {

  // Step 1: Get a payment session from your server.
  const paymentSessionData = await getPaymentSessionData();

  // Step 2: Create and invoke the SDK.
  // Create an AMSCashierPayment instance.
  const checkoutApp = new window.AMSCashierPayment({
    environment: "sandbox",
    locale: "en_US",
    onLog: ({code, message}) => {},
    onEventCallback: onEventCallback,
  });

  // Show the checkout ui by invoking createComponent method.
  await checkoutApp.createComponent({
    sessionData: paymentSessionData,
  });
}

// Handle payment results.
const onEventCallback = function({ code, result }) {
  switch (code) {
    case 'SDK_PAYMENT_SUCCESSFUL':
      // Payment was successful. Redirect the buyer to the payment result page, then confirm the payment result with the server.
      break;
    case 'SDK_PAYMENT_PROCESSING':
      console.log('Check the payment result data', result);
      // Payment is being processed. Confirm the payment result with the server. If the payment is successful, redirect the buyer to the payment result page. If the payment fails, guide the buyer to retry payment.
      break;
    case 'SDK_PAYMENT_FAIL':
      console.log('Check the payment result data', result);
      // Payment failed. Check the value of result.paymentResultCode in the onEventCallback result data for details. Guide the buyer to retry payment based on the obtained information.
      break;
    case 'SDK_PAYMENT_CANCEL':
      // Payment was canceled. The SDK can be re-invoked with a paymentSessionData within the validity period; if it has expired, the paymentSessionData needs to be re-requested.
      break;
    case 'SDK_PAYMENT_ERROR':
      console.log('Check the payment result data', result);
      // The payment status was abnormal. Confirm the payment result with the server. If it is confirmed that the payment has failed, guide the buyer to retry payment.
      break;
    case 'SDK_FORM_VERIFICATION_FAILED':
      // Verification failed after form submission. Guide the buyer to confirm the input and retry payment.
      break;
    default:
      break;
  }
}

```

# Event codes {#ZedB51}

 Status codes: Returned by   `onEventCallback`   during the component's runtime lifecycle.

   | **Type** | **Code** | **Description** | **Further action** |
| --- | --- | --- | --- |
| Status codes | SDK_FORM_VERIFICATION_FAILED | Verification failed after form submission. | Guide the buyer to confirm the input and retry payment. |

**Tab: iOS**

# User experience {#BkD7T2}

 The following figures show the user journey of paying on an app:

 ![iOS user experience showing APM payment flow via Antom SDK on a shopping app](https://idocs-assets.marmot-cloud.com/storage/idocs87c36dc8dac653c1/yuque/idocs/2025/png/45ae4b66-7747-4ee7-8d36-3d4233d1bb44.png)

# Payment flow {#d2GKE2}

 For each payment method, the payment flow is composed of the following steps:

 ![APM payment flow showing the interaction across buyer, merchant client, Antom SDK, bank App, merchant server, and Antom server](https://idocs-assets.marmot-cloud.com/storage/idocs87c36dc8dac653c1/yuque/idocs/2025/png/70a0e1a6-0b2b-4926-b6a3-289cf49f3ae1.png)

 1. **The buyer lands on the checkout page.**
2. **Create a payment session request.**  
   After the buyer selects a payment method and submits the order, you can obtain the payment session by calling the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  .
3. **Invoke the client SDK.**  
   On the client side, invoke the SDK through the payment session. The SDK will handle information collection, redirection, app invocation, QR code display, verification, and other processes based on the payment method's characteristics. Then the SDK will callback the payment result via   `onEventCallback`   or redirect to your payment result page.
4. **Confirm the payment result.**  
   Obtain the payment result by using one of the following two methods:

   - Asynchronous notification: Specify     *paymentNotifyUrl*     in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API to set the address for receiving asynchronous notifications. When the payment is successful or expires, Antom will use  [**notifyPayment**](https://docs.antom.com/ac/ams/paymentrn_online.md)  to send asynchronous notifications to you.
  - Synchronous inquiry: Call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API to check the payment status.

# Integration preparations {#g7B00}

 Before you start integrating, read the   [Integration guide](https://docs.antom.com/integration_guide_en.md)   and   [API overview](https://docs.antom.com/ac/ams/api_fund.md)   to understand the integration steps of the server-side API and the precautions for calling the API. Furthermore, ensure the following prerequisites are met:

 - Obtained your client ID
- Complete the key configuration
- Complete the configuration of     *paymentNotifyUrl*     to receive the asynchronous notification
- Integrate the server-side SDK package, install the server-side library, and initialize a request instance. For more details, refer to   [Server-side SDKs](https://docs.antom.com/ac/sdks/server_sdks.md)  .

 - Integrate the client-side SDK package by following the steps of   [Integrate the SDK package for iOS](https://docs.antom.com/ac/sdks/ios.md)  .

 > **[INFO]** **Note**    : Currently, Flutter and React Native (RN) development frameworks are not supported.

# Integration steps {#AkD7T2}

 Start your integration by taking the following steps:

 1. Create a payment session
2. Create and invoke the SDK
3. Obtain payment result

## Step 1: Create a payment session  **[Server-side]** {#zhBSk2}

 When a buyer selects a payment method provided by Antom, you need to collect key information such as the payment request ID, order amount, payment method, order description, payment redirect URL, and payment result notification URL, call the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API to create a payment session, and return the payment session to the client.

 Creating a payment session includes the following parameters:

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *productCode* | Yes | The value is fixed to  `CASHIER_PAYMENT`  . |
| *paymentRequestId* | Yes | The unique ID generated by the merchant, a new ID is required every time a payment is initiated. |
| *paymentAmount* | Yes | Payment amount, should be set according to the smallest unit of the order currency, e.g. CNY for cents, KRW for dollars. |
| *paymentMethod* | Yes | Payment method enumeration. |
| *paymentRedirectUrl* | Yes | Merchant's payment result page, according to the server-side results, not a fixed success page. |
| *order* | Yes | Order information, including order amount, order ID, and order description on the merchant side. |
| *paymentNotifyUrl* | No | Payment result notification address, which can be passed in through the API or set as a fixed value through Antom Dashboard. |
| *settlementStrategy* | No | The settlement strategy for the payment request. |
| *paymentFactor.isAuthorization* | No | Indicates whether the payment scenario requires authorization. Valid values are: - `false`  : Default value. Indicates that the payment scenario is a regular payment without authorization. - `true`  : Indicates that the payment scenario requires authorization.   Under the authorization scenario, the payment funds are guaranteed and held on the payment method side. You can use the        [**capture (One-time Payments)**](https://docs.antom.com/ac/ams/capture.md)        API to deduct the payment funds.   > **[INFO]** **Note**    : For APM payments, ensure this parameter is set to   `false`   or left empty to avoid the automatic release of funds to the buyer. |

 The above parameters are the basic parameters for creating a payment session, for full parameters and additional requirements for certain payment methods please refer to  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)   **.**

 The following sample code shows how to call the      [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)      API:

   ```java
private static void createPaymentSession() {
    AlipayPaymentSessionRequest alipayPaymentSessionRequest = new AlipayPaymentSessionRequest();
    alipayPaymentSessionRequest.setProductCode(ProductCodeType.CASHIER_PAYMENT);

    // replace with your paymentRequestId
    String paymentRequestId = UUID.randomUUID().toString();
    alipayPaymentSessionRequest.setPaymentRequestId(paymentRequestId);

    // set amount
    Amount amount = Amount.builder().currency("SGD").value("4200").build();
    alipayPaymentSessionRequest.setPaymentAmount(amount);

    // set paymentMethod
    PaymentMethod paymentMethod = PaymentMethod.builder().paymentMethodType("SHOPEEPAY_SG").build();
    alipayPaymentSessionRequest.setPaymentMethod(paymentMethod);

    // replace with your orderId
    String orderId = UUID.randomUUID().toString();

    // set buyer info
    Buyer buyer = Buyer.builder().referenceBuyerId("yourBuyerId").build();

    // set order info
    Order order = Order.builder().referenceOrderId(orderId)
        .orderDescription("antom testing order").orderAmount(amount).buyer(buyer).build();
    alipayPaymentSessionRequest.setOrder(order);

    // replace with your notify url
    alipayPaymentSessionRequest.setPaymentNotifyUrl("https://www.yourNotifyUrl.com");

    // replace with your redirect url
    alipayPaymentSessionRequest.setPaymentRedirectUrl("https://www.yourMerchantWeb.com");

    AlipayPaymentSessionResponse alipayPaymentSessionResponse = null;
    try {
        alipayPaymentSessionResponse = CLIENT.execute(alipayPaymentSessionRequest);
    } catch (AlipayApiException e) {
        String errorMsg = e.getMessage();
        // handle error condition
    }
}

```

   The following code shows a sample of the request message:

   ```json
{
  "order": {
    "buyer": {
      "referenceBuyerId": "yourBuyerId"
    },
    "orderAmount": {
      "currency": "SGD",
      "value": "4200"
    },
    "orderDescription": "antom testing order",
    "referenceOrderId": "referenceOrderId01"
  },
  "paymentAmount": {
    "currency": "SGD",
    "value": "4200"
  },
  "paymentMethod": {
    "paymentMethodType": "SHOPEEPAY_SG"
  },
  "paymentNotifyUrl": "https://www.yourNotifyUrl.com",
  "paymentRedirectUrl": "https://www.yourMerchantWeb.com",
  "paymentRequestId": "paymentRequestId01",
  "productCode": "CASHIER_PAYMENT"
}

```

   The following code shows a sample of the response, which contains the following parameters:

 - *paymentSessionData*    : the payment session data to be returned to the frontend.
- *paymentSessionExpiryTime*    : the expiration time of the payment session.

   ```json
{
  "paymentSessionData": "gM5Y/Go6+IXMPSOZrYtukCdodtwxuSFsn1vRPSB8CzkDqlb1g2W7dDipOkoMRs/x2W7g7qVhjEM7RUGroIT8Cw==&&SG&&188&&eyJhY3Rpb24iOnsiYXBwIjp7fSwibmVlZFZlcmlmeUFuZFJlc3VtZSI6ZmFsc2UsInNraXBTZGtRdWVyeSI6ZmFsc2UsIndhcCI6e30sIndlYiI6e319LCJjbGllbnRJZCI6IlNBTkRCT1hfNVlCWjBMMlpIUjhSMDg5NzgiLCJleHRlbmRJbmZvIjoie1widmVyc2lvbk1hcFwiOntcIndlYlwiOntcIjEuMS4wXCI6e1widGFyZ2V0V2ViVmVyaXNvblwiOlwiMS4xLjBcIn0sXCIxLjIuMFwiOntcInRhcmdldFdlYlZlcmlzb25cIjpcIjEuMi4wXCJ9fSxcImlPU1wiOntcIjEuMS4wXCI6e1widGFyZ2V0V2ViVmVyaXNvblwiOlwiMS4xLjBcIn0sXCIxLjIuMFwiOntcInRhcmdldFdlYlZlcmlzb25cIjpcIjEuMi4wXCJ9fSxcIkFuZHJvaWRcIjp7XCIxLjEuMFwiOntcInRhcmdldFdlYlZlcmlzb25cIjpcIjEuMS4wXCJ9LFwiMS4yLjBcIjp7XCJ0YXJnZXRXZWJWZXJpc29uXCI6XCIxLjIuMFwifX19fSIsInBheW1lbnRTZXNzaW9uQ29uZmlnIjp7InBheW1lbnRNZXRob2RDYXRlZ29yeVR5cGUiOiJBUE0iLCJwcm9kdWN0U2NlbmUiOiJDQVNISUVSX1BBWU1FTlQiLCJwcm9kdWN0U2NlbmVWZXJzaW9uIjoiMS4wIn19",
  "paymentSessionExpiryTime": "2024-01-01T00:00:00+08:00",
  "paymentSessionId": "gM5Y/Go6+IXMPSOZrYtukCdodtwxuSFsn1vRPSB8Czn6HfgqNzo2ogbIk1NUQS35",
  "result": {
    "resultCode": "SUCCESS",
    "resultMessage": "success.",
    "resultStatus": "S"
  }
}

```

   > **[INFO]** **Common questions**
>
>  **Q: Can I use Chinese characters in the value of the request parameters?**
>
>  A: To avoid incompatibility of a certain payment method, do not use Chinese characters for fields in the request.
>
>  ​
>
>  **Q: How to set the URL to receive the payment notification?**
>
>  A: Specify     *paymentNotifyUrl*     in the      [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)      API to receive the asynchronous notification about the payment result (  [**notifyPayment**](https://docs.antom.com/ac/ams/paymentrn_online.md)  ), or configure the receiving URL in Antom Dashboard. If the URL is specified in both the request and Antom Dashboard, the value specified in the request takes precedence.

## Step 2: Create and invoke the SDK  **[Client-side]** {#oDHQJ2}

 The Antom SDK is a component used for handling payment processes. Initiate the SDK by creating a payment session to collect information, switch between apps, and display QR codes based on the payment method specified in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API.

 After the buyer selects a payment method on the page, you need to create the SDK and initiate it with a payment session.

### 1\. (Optional) Preload the SDK {#WrrerJee}

 Before creating a payment session, it is highly recommended that you preload the SDK to improve the rendering speed of the checkout page, reducing the waiting time for buyers during the payment. Follow the code example below to perform the preloading:

 ```objectivec
[[AMSCashierPayment shared] preload];

```

### 2\. Instantiate the SDK {#KQO5X2}

 Create the SDK instance by using the   `AMSCashierPayment`   and specify basic configurations. Creating an   `AMSCashierPaymentConfiguration`   object includes the following parameters:

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *locale* | No | It is used to pass in language information. Valid values are listed as follows. You can choose the value to pass based on the region of the payment method. If other values are passed, the local language is used by default: - `en_US`  : English - `pt_BR`  : Portuguese - `ko_KR`  : Korean - `es_ES`  : Spanish - `ms_MY`  : Malay - `in_ID`  : Indonesian - `tl_PH`  : Tagalog - `th_TH`  : Thai - `vi_VN`  : Vietnamese - `fr_FR`  : French - `nl_NL`  : Dutch - `it_IT`  : Italian - `de_DE`  : German - `zh_CN`  : Simplified Chinese - `zh_HK`  : Traditional Chinese |
| *options* | No | It is used to specify whether to use the default loading pattern and the sandbox environment. Valid values are: - `"sandbox", "true"`  : Sandbox environment - `"sandbox", "false"`  : Production environment |

 Implementing the   `AMSPaymentProtocol`   used for handling corresponding events in subsequent processes. It contains the following method:

 | **Method** | **Required** | **Description** |
| --- | --- | --- |
| *onEventCallback* | No | A callback function that monitors payment events on the checkout page, returning     *eventCode*     and     *eventResult*    . |

 > **[INFO]** **Note**    : Each instance can handle only one payment session. It is recommended to create the instance before calling   `createComponent`   each time.

 The following sample code shows how to instantiate the SDK:

   ```objectivec
#import <AMSComponent/AMSComponent-Swift.h>

AMSCashierPaymentConfiguration *componentConfig = [AMSCashierPaymentConfiguration new];
componentConfig.locale = @"en_US";
// Set the sandbox environment. If you leave it empty, the production environment is used by default.
NSDictionary *options = @{@"sandbox": @"true"};
componentConfig.options = options;

[[AMSCashierPayment shared] initConfiguration:componentConfig];

// Set the callback to monitor payment events on the checkout page.
[AMSCashierPayment shared].paymentDelegate = self;

// The server calls the createPaymentSession API to obtain paymentSessionData.

```

### 3\. Invoke the SDK {#iVDpG2}

 Call the   `createComponent`   method:

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *sessionData* | Yes | Create a configuration object by using the     *sessionData*     parameter: Pass the complete data in the     *paymentSessionData*     parameter obtained in the response through the   **[**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)**   API to the     *sessionData*     parameter. |

 Call the   `onDestroy`   method to free SDK component resources in the following situations:

 - When the buyer exits the checkout page, free the component resources created in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  .
- When the buyer initiates multiple payments, free the component resources created in the previous  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  .

 The following sample code shows how to invoke the SDK:

 ```objectivec
[[AMSCashierPayment shared] createComponent:sessionData];

//Free SDK component resources
[[AMSCashierPayment shared] onDestroy];
```

### 4\. Display payment results {#Go6NO2}

 You can use the payment component to handle payment methods which require a redirect (asynchronous payments), and those that do not (synchronous payments).

#### Redirect payment {#6ubRS2}

 The payment process of some payment methods needs to leave the original web page and jump to the organization's page to complete the payment, such as the payment method of online banking. In this case, the buyer is redirected to the     *paymentRedirectUrl*     that you provided in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API after completing the payment. You can obtain the payment result by active query and and show it to the buyer in that URL.

#### Non-redirect payment {#NEgUb2}

 For payment without redirection, the payment result will be given through the   `onEventCallback`   function. The payment result here is only for front-end display, and the final order status is subject to the server side. You need to customize the processing flow you want for each payment result through the data in the result of   `onEventCallback`  .

   The following are the possible event codes of the payment result returned by   `onEventCallback`  .

   | **Event code** | **Message** | **Solution** |
| --- | --- | --- |
| SDK_PAYMENT_SUCCESSFUL | Payment was successful. | Redirect the buyer to the payment result page, then confirm the payment result with the server. |
| SDK_PAYMENT_PROCESSING | Payment is being processed. | Confirm the payment result with the server. If the payment is successful, redirect the buyer to the payment result page. If the payment fails, guide the buyer to retry payment. |
| SDK_PAYMENT_FAIL | Payment failed. | Check the value of          *result.paymentResultCode*     in the   `onEventCallback`   result data for details. Guide the buyer to retry payment based on the obtained information. > **[INFO]** **Note**    : For the meaning of     *result.paymentResultCode*    , refer to   [Description of payment failure or abnormal payment status](https://docs.antom.com/ac/sdks/ref.md?locale=en-us#nuduf)  . |
| SDK_PAYMENT_CANCEL | Payment was canceled. | The SDK can be re-invoked with a     *paymentSessionData*     within the validity period; if it has expired, the     *paymentSessionData*     needs to be re-requested. |
| SDK_PAYMENT_ERROR | The payment status was abnormal. | Confirm the payment result with the server.    If it is confirmed that the payment has failed, guide the buyer to    retry payment   . |

 The following sample code shows how to process the   `onEventCallback`  :

 ```objectivec
#import <AMSComponent/AMSComponent-Swift.h>

#pragma AMSPaymentProtocol
- (void)onEventCallback:(NSString *)eventCode eventResult:(AMSEventResult *)eventResult
{
    if ([eventCode isEqualToString:@"SDK_PAYMENT_SUCCESSFUL"]) {
        // Payment was successful. Redirect the buyer to the payment result page, then confirm the payment result with the server.
    } else if ([eventCode isEqualToString:@"SDK_PAYMENT_PROCESSING"]) {
        // Payment is being processed. Confirm the payment result with the server. If the payment is successful, redirect the buyer to the payment result page. If the payment fails, guide the buyer to retry payment.

    } else if ([eventCode isEqualToString:@"SDK_PAYMENT_FAIL"]) {
        // Payment failed. Check the value of result.paymentResultCode in the onEventCallback result data for details. Guide the buyer to retry payment based on the obtained information.

    } else if ([eventCode isEqualToString:@"SDK_PAYMENT_CANCEL"]) {
        // Payment was canceled. The SDK can be re-invoked with a paymentSessionData within the validity period; if it has expired, the paymentSessionData needs to be re-requested.

    } else if ([eventCode isEqualToString:@"SDK_PAYMENT_ERROR"]) {
        // The payment status was abnormal. Confirm the payment result with the server. If it is confirmed that the payment has failed, guide the buyer to retry payment.
    }
    NSLog(@"eventCode%@ eventResult%@", eventCode, eventResult);
}
```

## Step 3: Obtain payment result  **[Server-side]** {#SnQk72}

 After the buyer completes the payment or the payment times out, Antom sends the corresponding payment results to you through server interactions, you can obtain the payment result by one of the following methods:

 - Receive the asynchronous notification
- Inquire about the result

### Receive the asynchronous notification {#WDV7j2}

 When the payment reaches a final status of success or failure, Antom sends an asynchronous notification to the     *paymentNotifyUrl*     specified in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API through the  [**notifyPayment**](https://docs.antom.com/ac/ams/paymentrn_online.md)  API. When you receive the notification from Antom, you are required to return a response as instructed in  [Return a receipt acknowledgment message](https://docs.antom.com/ac/cashierpay/notifications.md#vQK5A)  .

 Antom allows you to specify the URL in     *paymentNotifyUrl*     parameter within the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API. If the URL for each payment is the same, you can also configure it in the Antom Dashboard.

   The following code shows a sample of the notification request:

   ```json
{
  "notifyType": "PAYMENT_RESULT",
  "result": {
    "resultCode": "SUCCESS",
    "resultStatus": "S",
    "resultMessage": "success"
  },
  "paymentRequestId": "paymentRequestId01",
  "paymentId": "20240101123456789XXXX",
  "paymentAmount": {
    "value": "4200",
    "currency": "SGD"
  },
  "paymentCreateTime": "2024-01-01T00:00:00+08:00",
  "paymentTime": "2024-01-01T00:01:00+08:00"
}

```

   How to verify the signature of the notification and make a response to the notification, see  [Sign a request and verify the signature](https://docs.antom.com/ac/ams/digital_signature.md)  .

   > **[INFO]** **Common questions**
>
>  **Q: When will the notification be sent?**
>
>  A: It depends on whether the payment is completed:
>
>  - If the payment is successfully completed, Antom will usually send you an asynchronous notification within 3 to 5 seconds. For some payment methods like cash payment, the notification might take a bit longer.
> - If the payment is not completed, Antom needs to close the order first before sending an asynchronous notification. The time it takes for different payment methods to close the order varies, usually defaulting to 14 minutes.
>
>   **Q: Will the asynchronous notification be re-sent?**
>
>  A: Yes, the asynchronous notification will be re-sent automatically within 24 hours for the following cases:
>
>  - If you didn't receive the asynchronous notification due to network reasons.
> - If you receive an asynchronous notification from Antom, but you didn't make a response to the notification in the  [Sample code](https://docs.antom.com/ac/cashierpay/notifications.md#vQK5A)  format.
>
>  The notification can be resent up to 8 times or until a correct response is received to terminate delivery. The sending intervals are as follows: 0 minutes, 2 minutes, 10 minutes, 10 minutes, 1 hour, 2 hours, 6 hours, and 15 hours.
>
>  
>  **Q: When responding to asynchronous notification, do I need to add a digital signature?**
>
>  A: If you receive an asynchronous notification from Antom, you are required to return the response in the  [Sample code](https://docs.antom.com/ac/cashierpay/notifications.md#vQK5A)  format   , but you do not need to countersign the response.
>
>  
>  **Q: What are the key parameters in the notification that I need to use?**
>
>  A: Pay attention to the following key parameters:
>
>  - *result*    : indicates the payment result of the order.
> - *paymentRequestId*    : indicates the payment request number you generated for consult, cancel, and reconciliation.
> - *paymentId*    : indicates the payment order number generated by Antom used for refund and reconciliation.
> - *paymentAmount*    : indicates the payment amount.

### Inquire about the result {#wlAb82}

 You can call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API to initiate a query on the result of an order.

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *paymentRequestId* | No | The payment request ID generated by the merchant. |

 The parameter is not a full set of parameters, please refer to the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)      API for full set of parameters and additional requirements for certain payment methods.

   The following sample code shows how to call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API:

 ```java
public static void inquiryPayment() {
    AlipayPayQueryRequest alipayPayQueryRequest = new AlipayPayQueryRequest();

    // replace with your paymentRequestId
    alipayPayQueryRequest.setPaymentRequestId("yourPaymentRequestId");

    AlipayPayQueryResponse alipayPayQueryResponse = null;
    try {
        alipayPayQueryResponse = CLIENT.execute(alipayPayQueryRequest);
    } catch (AlipayApiException e) {
        String errorMsg = e.getMessage();
        // handle error condition
    }
}

```

   The following code shows a sample of the request message:

   ```json
{
  "paymentRequestId": "paymentRequestId01"
}

```

   The following code shows a sample of the response message:

   ```json
{
  "result": {
    "resultCode": "SUCCESS",
    "resultStatus": "S",
    "resultMessage": "Success"
  },
  "paymentStatus": "SUCCESS",
  "paymentRequestId": "paymentRequestId01",
  "paymentId": "20240101123456789XXXX",
  "paymentAmount": {
    "value": "4200",
    "currency": "SGD"
  },
  "paymentCreateTime": "2024-01-01T00:00:00+08:00",
  "paymentTime": "2024-01-01T00:01:00+08:00",
  "paymentMethodType": "SHOPEEPAY_SG"
}

```

   > **[INFO]** **Common questions**
>
>  **Q: How often should I call the inquiryPayment API?**
>
>  A: Call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API constantly with an interval of 2 seconds until the final payment result is obtained or an asynchronous payment result notification is received.
>
>  
>  **Q: What are the key parameters in the notification that I need to use?**
>
>  A: Pay attention to these key parameters:
>
>  - *result*    : represents the result of this  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API call, the result of the order needs to be judged according to     *paymentStatus*     :
>
>    - `SUCCESS`   and   `FAIL`   represent the final result.
>   - `PROCESSING`   represents the processing.
>
>  - *paymentAmount*    : indicates the payment amount.

# Sample codes {#xjYO02}

 Full sample codes for front-end:

 ```objectivec
#import <AMSComponent/AMSComponent-Swift.h>

// (Optional) Preload the SDK. Before creating a payment session, preload the SDK to improve the rendering speed of the checkout page, reducing the waiting time for buyers during the payment.
[[AMSCashierPayment shared] preload];

// Step 1: Get Antom PaymentSessionData from your server.
NSString *paymentSessionData = "YOUR_PAYMENT_SESSIONDATA";

// Step 2: Create an AMSConfiguration object.
AMSCashierPaymentConfiguration *componentConfig = [AMSCashierPaymentConfiguration new];
componentConfig.locale = @"en_US";

// Set the sandbox environment. If you leave it empty, the production environment is used by default.
NSDictionary *options = @{@"sandbox": @"true"};
componentConfig.options = options;
[[AMSCashierPayment shared] initConfiguration:componentConfig];

// Set the callback to monitor payment events on the checkout page.
[AMSCashierPayment shared].paymentDelegate = self;

// Create and render the component.
[[AMSCashierPayment shared] createComponent:paymentSessionData];

#pragma AMSPaymentProtocol
- (void)onEventCallback:(NSString *)eventCode eventResult:(AMSEventResult *)eventResult
{
    if ([eventCode isEqualToString:@"SDK_PAYMENT_SUCCESSFUL"]) {
        // Payment was successful. Redirect the buyer to the payment result page, then confirm the payment result with the server.
    } else if ([eventCode isEqualToString:@"SDK_PAYMENT_PROCESSING"]) {
        // Payment is being processed. Confirm the payment result with the server. If the payment is successful, redirect the buyer to the payment result page. If the payment fails, guide the buyer to retry payment.
    } else if ([eventCode isEqualToString:@"SDK_PAYMENT_FAIL"]) {
        // Payment failed. Check the value of result.paymentResultCode in the onEventCallback result data for details. Guide the buyer to retry payment based on the obtained information.
    } else if ([eventCode isEqualToString:@"SDK_PAYMENT_CANCEL"]) {
        // Payment was canceled. The SDK can be re-invoked with a paymentSessionData within the validity period; if it has expired, the paymentSessionData needs to be re-requested.
    } else if ([eventCode isEqualToString:@"SDK_PAYMENT_ERROR"]) {
        // The payment status was abnormal. Confirm the payment result with the server. If it is confirmed that the payment has failed, guide the buyer to retry payment.
    } else if ([eventCode isEqualToString:@"SDK_FORM_VERIFICATION_FAILED"]) {
       // Verification failed after form submission. Guide the buyer to confirm the input and retry payment.
    }
    NSLog(@"eventCode%@ eventResult%@", eventCode, eventResult);
}

// After the payment is completed or when the buyer exits the checkout page, free SDK component resources.
[[AMSCashierPayment shared] onDestroy];
```

# Event codes {#ZedB52}

 Status codes: Returned by   `onEventCallback`   during the component's runtime lifecycle.

   | **Type** | **Code** | **Description** | **Further action** |
| --- | --- | --- | --- |
| Status codes | SDK_FORM_VERIFICATION_FAILED | Verification failed after form submission. | Guide the buyer to confirm the input and retry payment. |

**Tab: Android**

# User experience {#BkD7T3}

 The following figures show the user journey of paying on an app:

 ![Android user experience showing APM payment flow via Antom SDK on a shopping app](https://idocs-assets.marmot-cloud.com/storage/idocs87c36dc8dac653c1/yuque/idocs/2025/png/45ae4b66-7747-4ee7-8d36-3d4233d1bb44.png)

# Payment flow {#d2GKE3}

 For each payment method, the payment flow is composed of the following steps:

 ![APM payment flow showing the interaction across buyer, merchant client, Antom SDK, bank App, merchant server, and Antom server](https://idocs-assets.marmot-cloud.com/storage/idocs87c36dc8dac653c1/yuque/idocs/2025/png/70a0e1a6-0b2b-4926-b6a3-289cf49f3ae1.png)

 1. **The buyer lands on the checkout page.**
2. **Create a payment session request.**  
   After the buyer selects a payment method and submits the order, you can obtain the payment session by calling the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  .
3. **Invoke the client SDK.**  
   On the client side, invoke the SDK through the payment session. The SDK will handle information collection, redirection, app invocation, QR code display, verification, and other processes based on the payment method's characteristics. Then the SDK will callback the payment result via   `OnCheckoutListener`   or redirect to your payment result page.
4. **Confirm the payment result.**  
   Obtain the payment result by using one of the following two methods:

   - Asynchronous notification: Specify     *paymentNotifyUrl*     in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API to set the address for receiving asynchronous notifications. When the payment is successful or expires, Antom will use  [**notifyPayment**](https://docs.antom.com/ac/ams/paymentrn_online.md)  to send asynchronous notifications to you.
  - Synchronous inquiry: Call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API to check the payment status.

# Integration preparations {#s1IDK}

 Before you start integrating, read the   [Integration guide](https://docs.antom.com/integration_guide_en.md)   and   [API overview](https://docs.antom.com/ac/ams/api_fund.md)   to understand the integration steps of the server-side API and the precautions for calling the API. Furthermore, ensure the following prerequisites are met:

 - Obtained your client ID
- Complete the key configuration
- Complete the configuration of     *paymentNotifyUrl*     to receive the asynchronous notification
- Integrate the server-side SDK package, install the server-side library, and initialize a request instance. For more details, refer to   [Server-side SDKs](https://docs.antom.com/ac/sdks/server_sdks.md)  .

 - Integrate the client-side SDK package by following the steps of   [Integrate the SDK package for Android](https://docs.antom.com/ac/sdks/android.md)  .

 > **[INFO]** **Note**    : Currently, Flutter and React Native (RN) development frameworks are not supported.

# Integration steps {#AkD7T3}

 Start your integration by taking the following steps:

 1. Create a payment session
2. Create and invoke the SDK
3. Obtain payment result

## Step 1: Create a payment session  **[Server-side]** {#zhBSk3}

 When a buyer selects a payment method provided by Antom, you need to collect key information such as the payment request ID, order amount, payment method, order description, payment redirect URL, and payment result notification URL, call the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API to create a payment session, and return the payment session to the client.

 Creating a payment session includes the following parameters:

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *productCode* | Yes | The value is fixed to  `CASHIER_PAYMENT`  . |
| *paymentRequestId* | Yes | The unique ID generated by the merchant, a new ID is required every time a payment is initiated. |
| *paymentAmount* | Yes | Payment amount, should be set according to the smallest unit of the order currency, e.g. CNY for cents, KRW for dollars. |
| *paymentMethod* | Yes | Payment method enumeration. |
| *paymentRedirectUrl* | Yes | Merchant's payment result page, according to the server-side results, not a fixed success page. |
| *order* | Yes | Order information, including order amount, order ID, and order description on the merchant side. |
| *paymentNotifyUrl* | No | Payment result notification address, which can be passed in through the API or set as a fixed value through Antom Dashboard. |
| *settlementStrategy* | No | The settlement strategy for the payment request. |
| *paymentFactor.isAuthorization* | No | Indicates whether the payment scenario requires authorization. Valid values are: - `false`  : Default value. Indicates that the payment scenario is a regular payment without authorization. - `true`  : Indicates that the payment scenario requires authorization.   Under the authorization scenario, the payment funds are guaranteed and held on the payment method side. You can use the        [**capture (One-time Payments)**](https://docs.antom.com/ac/ams/capture.md)        API to deduct the payment funds.   > **[INFO]** **Note**    : For APM payments, ensure this parameter is set to   `false`   or left empty to avoid the automatic release of funds to the buyer. |

 The above parameters are the basic parameters for creating a payment session, for full parameters and additional requirements for certain payment methods please refer to  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)   **.**

 The following sample code shows how to call the      [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)      API:

   ```java
private static void createPaymentSession() {
    AlipayPaymentSessionRequest alipayPaymentSessionRequest = new AlipayPaymentSessionRequest();
    alipayPaymentSessionRequest.setProductCode(ProductCodeType.CASHIER_PAYMENT);

    // replace with your paymentRequestId
    String paymentRequestId = UUID.randomUUID().toString();
    alipayPaymentSessionRequest.setPaymentRequestId(paymentRequestId);

    // set amount
    Amount amount = Amount.builder().currency("SGD").value("4200").build();
    alipayPaymentSessionRequest.setPaymentAmount(amount);

    // set paymentMethod
    PaymentMethod paymentMethod = PaymentMethod.builder().paymentMethodType("SHOPEEPAY_SG").build();
    alipayPaymentSessionRequest.setPaymentMethod(paymentMethod);

    // replace with your orderId
    String orderId = UUID.randomUUID().toString();

    // set buyer info
    Buyer buyer = Buyer.builder().referenceBuyerId("yourBuyerId").build();

    // set order info
    Order order = Order.builder().referenceOrderId(orderId)
        .orderDescription("antom testing order").orderAmount(amount).buyer(buyer).build();
    alipayPaymentSessionRequest.setOrder(order);

    // replace with your notify url
    alipayPaymentSessionRequest.setPaymentNotifyUrl("https://www.yourNotifyUrl.com");

    // replace with your redirect url
    alipayPaymentSessionRequest.setPaymentRedirectUrl("https://www.yourMerchantWeb.com");

    AlipayPaymentSessionResponse alipayPaymentSessionResponse = null;
    try {
        alipayPaymentSessionResponse = CLIENT.execute(alipayPaymentSessionRequest);
    } catch (AlipayApiException e) {
        String errorMsg = e.getMessage();
        // handle error condition
    }
}

```

   The following code shows a sample of the request message:

   ```json
{
  "order": {
    "buyer": {
      "referenceBuyerId": "yourBuyerId"
    },
    "orderAmount": {
      "currency": "SGD",
      "value": "4200"
    },
    "orderDescription": "antom testing order",
    "referenceOrderId": "referenceOrderId01"
  },
  "paymentAmount": {
    "currency": "SGD",
    "value": "4200"
  },
  "paymentMethod": {
    "paymentMethodType": "SHOPEEPAY_SG"
  },
  "paymentNotifyUrl": "https://www.yourNotifyUrl.com",
  "paymentRedirectUrl": "https://www.yourMerchantWeb.com",
  "paymentRequestId": "paymentRequestId01",
  "productCode": "CASHIER_PAYMENT"
}

```

   The following code shows a sample of the response, which contains the following parameters:

 - *paymentSessionData*    : the payment session data to be returned to the frontend.
- *paymentSessionExpiryTime*    : the expiration time of the payment session.

   ```json
{
  "paymentSessionData": "gM5Y/Go6+IXMPSOZrYtukCdodtwxuSFsn1vRPSB8CzkDqlb1g2W7dDipOkoMRs/x2W7g7qVhjEM7RUGroIT8Cw==&&SG&&188&&eyJhY3Rpb24iOnsiYXBwIjp7fSwibmVlZFZlcmlmeUFuZFJlc3VtZSI6ZmFsc2UsInNraXBTZGtRdWVyeSI6ZmFsc2UsIndhcCI6e30sIndlYiI6e319LCJjbGllbnRJZCI6IlNBTkRCT1hfNVlCWjBMMlpIUjhSMDg5NzgiLCJleHRlbmRJbmZvIjoie1widmVyc2lvbk1hcFwiOntcIndlYlwiOntcIjEuMS4wXCI6e1widGFyZ2V0V2ViVmVyaXNvblwiOlwiMS4xLjBcIn0sXCIxLjIuMFwiOntcInRhcmdldFdlYlZlcmlzb25cIjpcIjEuMi4wXCJ9fSxcImlPU1wiOntcIjEuMS4wXCI6e1widGFyZ2V0V2ViVmVyaXNvblwiOlwiMS4xLjBcIn0sXCIxLjIuMFwiOntcInRhcmdldFdlYlZlcmlzb25cIjpcIjEuMi4wXCJ9fSxcIkFuZHJvaWRcIjp7XCIxLjEuMFwiOntcInRhcmdldFdlYlZlcmlzb25cIjpcIjEuMS4wXCJ9LFwiMS4yLjBcIjp7XCJ0YXJnZXRXZWJWZXJpc29uXCI6XCIxLjIuMFwifX19fSIsInBheW1lbnRTZXNzaW9uQ29uZmlnIjp7InBheW1lbnRNZXRob2RDYXRlZ29yeVR5cGUiOiJBUE0iLCJwcm9kdWN0U2NlbmUiOiJDQVNISUVSX1BBWU1FTlQiLCJwcm9kdWN0U2NlbmVWZXJzaW9uIjoiMS4wIn19",
  "paymentSessionExpiryTime": "2024-01-01T00:00:00+08:00",
  "paymentSessionId": "gM5Y/Go6+IXMPSOZrYtukCdodtwxuSFsn1vRPSB8Czn6HfgqNzo2ogbIk1NUQS35",
  "result": {
    "resultCode": "SUCCESS",
    "resultMessage": "success.",
    "resultStatus": "S"
  }
}

```

   > **[INFO]** **Common questions**
>
>  **Q: Can I use Chinese characters in the value of the request parameters?**
>
>  A: To avoid incompatibility of a certain payment method, do not use Chinese characters for fields in the request.
>
>  ​
>
>  **Q: How to set the URL to receive the payment notification?**
>
>  A: Specify     *paymentNotifyUrl*     in the      [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)      API to receive the asynchronous notification about the payment result (  [**notifyPayment**](https://docs.antom.com/ac/ams/paymentrn_online.md)  ), or configure the receiving URL in Antom Dashboard. If the URL is specified in both the request and Antom Dashboard, the value specified in the request takes precedence.

## Step 2: Create and invoke the SDK  **[Client-side]** {#oDHQJ3}

 The Antom SDK is a component used for handling payment processes. Initiate the SDK by creating a payment session to collect information, switch between apps, and display QR codes based on the payment method specified in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API.

 After the buyer selects a payment method on the page, you need to create the SDK and initiate it with a payment session.

### 1\. (Optional) Preload the SDK {#SDUT56}

 Before creating a payment session, it is highly recommended that you preload the SDK to improve the rendering speed of the checkout page, reducing the waiting time for buyers during the payment.

 Call the   `preload`   method:

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *context* | Yes | The Android context object. |

 Follow the code example below to perform the preloading:

 ```java
AMSCashierPayment.preload(context.getApplicationContext());

```

### 2\. Instantiate the SDK {#KQO5X3}

 Create the SDK instance by using the   `AMSCashierPayment`   and specify basic configurations. Creating a configuration object includes the following methods:

 | **Method name** | **Required** | **Description** |
| --- | --- | --- |
| *setLocale* | No | It is used to pass in language information. Valid values are listed as follows. You can choose the value to pass based on the region of the payment method. If other values are passed, the local language is used by default: - `"en", "US"`  : English - `"pt", "BR"`  : Portuguese - `"ko", "KR"`  : Korean - `"es", "ES"`  : Spanish - `"ms", "MY"`  : Malay - `"in", "ID"`  : Indonesian - `"tl", "PH"`  : Tagalog - `"th", "TH"`  : Thai - `"vi", "VN"`  : Vietnamese - `"fr", "FR"`  : French - `"nl", "NL"`  : Dutch - `"it", "IT"`  : Italian - `"de", "DE"`  : German - `"zh", "CN"`  : Simplified Chinese - `"zh", "HK"`  : Traditional Chinese |
| *setOption* | No | It is used to specify whether to use the default loading pattern and the sandbox environment. Valid values are: - `"sandbox", "true"`  : Sandbox environment - `"sandbox", "false"`  : Production environment |
| *setOnCheckoutListener* | No | Create an instance of the     **OnCheckoutListener**     API, which is used for event handling in the subsequent process. The API includes the following method: - `onEventCallback`  : Required. A callback function that monitors payment events on the checkout page, returning     *eventCode*     and     *eventResult*     . |

   > **[INFO]** **Note**    : Each instance can handle only one payment session. It is recommended to create the instance before calling   `createComponent`   each time.

 The following sample code shows how to instantiate the SDK:

   ```java
AMSCashierPaymentConfiguration configuration = new AMSCashierPaymentConfiguration();
configuration.setLocale(new Locale("en", "US"));
// Set the sandbox environment. If you leave it empty, the production environment is used by default.
configuration.setOption("sandbox", "true");
// Set the callback to monitor payment events on the checkout page.
configuration.setOnCheckoutListener(new OnCheckoutListener() {
    @Override
    public void onEventCallback(String eventCode, AMSEventResult eventResult) {
        Log.e(TAG, "onEventCallback eventCode=" + eventCode + " eventResult=" + eventResult.toString());
    }
});
// Instantiate AMSCashierPayment.
AMSCashierPayment checkout = new AMSCashierPayment.Builder(activity, configuration).build();
```

### 3\. Invoke the SDK {#iVDpG3}

 Call the   `createComponent`   method:

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *sessionData* | Yes | Create a configuration object by using the     *sessionData*     parameter: Pass the complete data in the     *paymentSessionData*     parameter obtained in the response through the   **[**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)**   API to the     *sessionData*     parameter. |

 Call the   `onDestroy`   method to free SDK component resources in the following situations:

 - When the buyer exits the checkout page, free the component resources created in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  .
- When the buyer initiates multiple payments, free the component resources created in the previous  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  .

 The following sample code shows how to invoke the SDK:

 ```java
checkout.createComponent(activity, sessionData);

//Free SDK component resources
checkout.onDestroy();
```

### 4\. Display payment results {#Go6NO3}

 You can use the payment component to handle payment methods which require a redirect (asynchronous payments), and those that do not (synchronous payments).

#### Redirect payment {#6ubRS3}

 The payment process of some payment methods needs to leave the original web page and jump to the organization's page to complete the payment, such as the payment method of online banking. In this case, the buyer is redirected to the     *paymentRedirectUrl*     that you provided in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API after completing the payment. You can obtain the payment result by active query and and show it to the buyer in that URL.

#### Non-redirect payment {#NEgUb3}

 For payment without redirection, the payment result will be given through the   `onEventCallback`   function. The payment result here is only for front-end display, and the final order status is subject to the server side. You need to customize the processing flow you want for each payment result through the data in the result of   `onEventCallback`  .

   The following are the possible event codes of the payment result returned by   `onEventCallback`  .

   | **Event code** | **Message** | **Solution** |
| --- | --- | --- |
| SDK_PAYMENT_SUCCESSFUL | Payment was successful. | Redirect the buyer to the payment result page, then confirm the payment result with the server. |
| SDK_PAYMENT_PROCESSING | Payment is being processed. | Confirm the payment result with the server. If the payment is successful, redirect the buyer to the payment result page. If the payment fails, guide the buyer to retry payment. |
| SDK_PAYMENT_FAIL | Payment failed. | Check the value of          *result.paymentResultCode*     in the   `onEventCallback`   result data for details. Guide the buyer to retry payment based on the obtained information. > **[INFO]** **Note**    : For the meaning of     *result.paymentResultCode*    , refer to   [Description of payment failure or abnormal payment status](https://docs.antom.com/ac/sdks/ref.md?locale=en-us#nuduf)  . |
| SDK_PAYMENT_CANCEL | Payment was canceled. | The SDK can be re-invoked with a     *paymentSessionData*     within the validity period; if it has expired, the     *paymentSessionData*     needs to be re-requested. |
| SDK_PAYMENT_ERROR | The payment status was abnormal. | Confirm the payment result with the server.    If it is confirmed that the payment has failed, guide the buyer to    retry payment   . |

 The following sample code shows how to process the   `onEventCallback`  :

 ```java
@Override
public void onEventCallback(String eventCode, AMSEventResult eventResult) {
    AlipayLog.e(TAG, "onEventCallback eventCode=" + eventCode + " eventResult=" + eventResult.toString());

    if (!TextUtils.isEmpty(eventCode)) {
        if ("SDK_PAYMENT_SUCCESSFUL".equals(eventCode)) {
            // Payment was successful. Redirect the buyer to the payment result page, then confirm the payment result with the server.
        } else if ("SDK_PAYMENT_PROCESSING".equals(eventCode)) {
            // Payment is being processed. Confirm the payment result with the server. If the payment is successful, redirect the buyer to the payment result page. If the payment fails, guide the buyer to retry payment.
        } else if ("SDK_PAYMENT_FAIL".equals(eventCode)) {
            // Payment failed. Check the value of result.paymentResultCode in the onEventCallback result data for details. Guide the buyer to retry payment based on the obtained information.
        } else if ("SDK_PAYMENT_CANCEL".equals(eventCode)) {
            // Payment was canceled. The SDK can be re-invoked with a paymentSessionData within the validity period; if it has expired, the paymentSessionData needs to be re-requested.
        } else if ("SDK_PAYMENT_ERROR".equals(eventCode)) {
            // The payment status was abnormal. Confirm the payment result with the server. If it is confirmed that the payment has failed, guide the buyer to retry payment.
        }
    }
}
```

## Step 3: Obtain payment result  **[Server-side]** {#SnQk73}

 After the buyer completes the payment or the payment times out, Antom sends the corresponding payment results to you through server interactions, you can obtain the payment result by one of the following methods:

 - Receive the asynchronous notification
- Inquire about the result

### Receive the asynchronous notification {#WDV7j3}

 When the payment reaches a final status of success or failure, Antom sends an asynchronous notification to the     *paymentNotifyUrl*     specified in the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API through the  [**notifyPayment**](https://docs.antom.com/ac/ams/paymentrn_online.md)  API. When you receive the notification from Antom, you are required to return a response as instructed in  [Return a receipt acknowledgment message](https://docs.antom.com/ac/cashierpay/notifications.md#vQK5A)  .

 Antom allows you to specify the URL in     *paymentNotifyUrl*     parameter within the  [**createPaymentSession (One-time Payments)**](https://docs.antom.com/ac/ams/session_cashier.md)  API. If the URL for each payment is the same, you can also configure it in the Antom Dashboard.

   The following code shows a sample of the notification request:

   ```json
{
  "notifyType": "PAYMENT_RESULT",
  "result": {
    "resultCode": "SUCCESS",
    "resultStatus": "S",
    "resultMessage": "success"
  },
  "paymentRequestId": "paymentRequestId01",
  "paymentId": "20240101123456789XXXX",
  "paymentAmount": {
    "value": "4200",
    "currency": "SGD"
  },
  "paymentCreateTime": "2024-01-01T00:00:00+08:00",
  "paymentTime": "2024-01-01T00:01:00+08:00"
}

```

   How to verify the signature of the notification and make a response to the notification, see  [Sign a request and verify the signature](https://docs.antom.com/ac/ams/digital_signature.md)  .

   > **[INFO]** **Common questions**
>
>  **Q: When will the notification be sent?**
>
>  A: It depends on whether the payment is completed:
>
>  - If the payment is successfully completed, Antom will usually send you an asynchronous notification within 3 to 5 seconds. For some payment methods like cash payment, the notification might take a bit longer.
> - If the payment is not completed, Antom needs to close the order first before sending an asynchronous notification. The time it takes for different payment methods to close the order varies, usually defaulting to 14 minutes.
>
>   **Q: Will the asynchronous notification be re-sent?**
>
>  A: Yes, the asynchronous notification will be re-sent automatically within 24 hours for the following cases:
>
>  - If you didn't receive the asynchronous notification due to network reasons.
> - If you receive an asynchronous notification from Antom, but you didn't make a response to the notification in the  [Sample code](https://docs.antom.com/ac/cashierpay/notifications.md#vQK5A)  format.
>
>  The notification can be resent up to 8 times or until a correct response is received to terminate delivery. The sending intervals are as follows: 0 minutes, 2 minutes, 10 minutes, 10 minutes, 1 hour, 2 hours, 6 hours, and 15 hours.
>
>  
>  **Q: When responding to asynchronous notification, do I need to add a digital signature?**
>
>  A: If you receive an asynchronous notification from Antom, you are required to return the response in the  [Sample code](https://docs.antom.com/ac/cashierpay/notifications.md#vQK5A)  format   , but you do not need to countersign the response.
>
>  
>  **Q: What are the key parameters in the notification that I need to use?**
>
>  A: Pay attention to the following key parameters:
>
>  - *result*    : indicates the payment result of the order.
> - *paymentRequestId*    : indicates the payment request number you generated for consult, cancel, and reconciliation.
> - *paymentId*    : indicates the payment order number generated by Antom used for refund and reconciliation.
> - *paymentAmount*    : indicates the payment amount.

### Inquire about the result {#wlAb83}

 You can call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API to initiate a query on the result of an order.

 | **Parameter name** | **Required** | **Description** |
| --- | --- | --- |
| *paymentRequestId* | No | The payment request ID generated by the merchant. |

 The parameter is not a full set of parameters, please refer to the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)      API for full set of parameters and additional requirements for certain payment methods.

   The following sample code shows how to call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API:

 ```java
public static void inquiryPayment() {
    AlipayPayQueryRequest alipayPayQueryRequest = new AlipayPayQueryRequest();

    // replace with your paymentRequestId
    alipayPayQueryRequest.setPaymentRequestId("yourPaymentRequestId");

    AlipayPayQueryResponse alipayPayQueryResponse = null;
    try {
        alipayPayQueryResponse = CLIENT.execute(alipayPayQueryRequest);
    } catch (AlipayApiException e) {
        String errorMsg = e.getMessage();
        // handle error condition
    }
}

```

   The following code shows a sample of the request message:

   ```json
{
  "paymentRequestId": "paymentRequestId01"
}

```

   The following code shows a sample of the response message:

   ```json
{
  "result": {
    "resultCode": "SUCCESS",
    "resultStatus": "S",
    "resultMessage": "Success"
  },
  "paymentStatus": "SUCCESS",
  "paymentRequestId": "paymentRequestId01",
  "paymentId": "20240101123456789XXXX",
  "paymentAmount": {
    "value": "4200",
    "currency": "SGD"
  },
  "paymentCreateTime": "2024-01-01T00:00:00+08:00",
  "paymentTime": "2024-01-01T00:01:00+08:00",
  "paymentMethodType": "SHOPEEPAY_SG"
}

```

   > **[INFO]** **Common questions**
>
>  **Q: How often should I call the inquiryPayment API?**
>
>  A: Call the  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API constantly with an interval of 2 seconds until the final payment result is obtained or an asynchronous payment result notification is received.
>
>  
>  **Q: What are the key parameters in the notification that I need to use?**
>
>  A: Pay attention to these key parameters:
>
>  - *result*    : represents the result of this  [**inquiryPayment**](https://docs.antom.com/ac/ams/paymentri_online.md)  API call, the result of the order needs to be judged according to     *paymentStatus*     :
>
>    - `SUCCESS`   and   `FAIL`   represent the final result.
>   - `PROCESSING`   represents the processing.
>
>  - *paymentAmount*    : indicates the payment amount.

# Sample codes {#xjYO03}

 Full sample codes for front-end:

 ```java
// (Optional) Preload the SDK. Before creating a payment session, preload the SDK to improve the rendering speed of the checkout page, reducing the waiting time for buyers during the payment.
AMSCashierPayment.preload(context);

// Step 1: Get Antom PaymentSessionData from your server.
String paymentSessionData = "YOUR_PAYMENT_SESSIONDATA";

AMSCashierPaymentConfiguration configuration = new AMSCashierPaymentConfiguration();
configuration.setLocale(new Locale("en", "US"));
// Set the sandbox environment. If you leave it empty, the production environment is used by default.
configuration.setOption("sandbox", "true");
// Set the callback to monitor payment events on the checkout page.
configuration.setOnCheckoutListener(new OnCheckoutListener() {
    @Override
    public void onEventCallback(String eventCode, AMSEventResult eventResult) {
        AlipayLog.e(TAG, "onEventCallback eventCode=" + eventCode + " eventResult=" + eventResult.toString());

        if (!TextUtils.isEmpty(eventCode)) {
            if ("SDK_PAYMENT_SUCCESSFUL".equals(eventCode)) {
                // Payment was successful. Redirect the buyer to the payment result page, then confirm the payment result with the server.
            } else if ("SDK_PAYMENT_PROCESSING".equals(eventCode)) {
                // Payment is being processed. Confirm the payment result with the server. If the payment is successful, redirect the buyer to the payment result page. If the payment fails, guide the buyer to retry payment.
            } else if ("SDK_PAYMENT_FAIL".equals(eventCode)) {
                // Payment failed. Check the value of result.paymentResultCode in the onEventCallback result data for details. Guide the buyer to retry payment based on the obtained information.
            }else if ("SDK_PAYMENT_CANCEL".equals(eventCode)) {
                // Payment was canceled. The SDK can be re-invoked with a paymentSessionData within the validity period; if it has expired, the paymentSessionData needs to be re-requested.
            } else if ("SDK_PAYMENT_ERROR".equals(eventCode)) {
                // The payment status was abnormal. Confirm the payment result with the server. If it is confirmed that the payment has failed, guide the buyer to retry payment.
            } else if ("SDK_FORM_VERIFICATION_FAILED".equals(eventCode)) {
                // Verification failed after form submission. Guide the buyer to confirm the input and retry payment.
            }
        }
    }
});
// Instantiate AMSCashierPayment.
AMSCashierPayment checkout = new AMSCashierPayment.Builder(activity, configuration).build();

// Create and render the component.
checkout.createComponent(activity, sessionData);

// After the payment is completed or when the buyer exits the checkout page, free SDK component resources.
checkout.onDestroy();
```

# Event codes {#ZedB53}

 Status codes: Returned by   `onEventCallback`   during the component's runtime lifecycle.

   | **Type** | **Code** | **Description** | **Further action** |
| --- | --- | --- | --- |
| Status codes | SDK_FORM_VERIFICATION_FAILED | Verification failed after form submission. | Guide the buyer to confirm the input and retry payment. |

<!-- /ToggleTab -->