Public
Documentation Settings

XPay

XPay Fusion

XPay Fusion

XPay Fusion is a PCI DSS-compliant payment solution to collect payments from your customers on your webapp without them having to redirect to a third-party payment page.

Integrating XPay Fusion

XPay%20e2e%20Integration%20-%20Sequence%20Diagram

XPay e2e Integration - Sequence Diagram

To integrate XPay Fusion with your eCommerce store, you need to:

  • Integrate Fusion APIs on the server side and
  • Add the Fusion embedded element on the client side by using XPay JS SDK.

Step by step flow

  • When your customer comes to the checkout page, they enter their card details into the card details form rendered in a secure iframe, enabled by the Fusion Embedded Element. They then click the Place Order button.
  • The Fusion Embedded Element can be added to your frontend project as follows:
javascript
npm i @xstak/xpay-element

3. Import the XPay provider and add the required keys

Plain Text
import { XPay } from '@xstak/xpay-element';
function XPayPayment() {
return  (
<XPay xpay={{ publishableKey: "", accountId: "", hmacSecret: ""}}>
<Payment />
</XPay>
)}

4. Render the Fusion Embedded Element

javascript
import React, { useState } from 'react';
import { PaymentElement, useXpay } from '@xstak/xpay-element';
export const Payment = () => {
    const options = {
        override: true,
        fields: {
          creditCard: {
            placeholder: "4234 1234 1234 1234",
            label: "Enter your credit card",
          },
          exp: {
            placeholder: "Exp. Date",
          },
        },
        style: {
          ".input": {
            "border": "1px solid black",
            "border-radius": "15px"
           },
          ".invalid": {},
          ".label": {
            "font-size": "18px"
           },
          ":focus": {},
          ":hover": {},
          "::placeholder": {},
          "::selection": {},
        },
      };
    const xpay = useXpay();
    const pay = async() => { 
const { message, error } = await xpay.confirmPayment("card", payment_intent_client_secret, customer = {name: ‘Dane Joe}, encryptionKey);
//if payment is successful, request backend to create order.
//else if payment is unsuccessful, show error message
    }
    return(
        <>
        <PaymentElement options={options} onReady={(event) => {
            console.log("ready event", event);
          }} />
        <Button type="primary" onClick={pay}>
          Pay Now
        </Button>
    )
}

Important: XPay SDK supports custom styling and changing labels and placeholder texts of input fields as per your design to provide a seamless experience to your customers.

5. When the customer clicks on the Place Order button, the frontend requests the backend to start the payment processing by passing relevant details about the cart (cart id, cart total etc) and the customer (customer name and contact details etc). Since the order would not have been created as yet, it is recommended that the cart id be used at this step.

6. The backend receives this request from the frontend and creates a payment intent by calling the Create Payment Intent API and returns the response to the frontend:

a. The cart id should be added in the metadata object, with the key of order_reference, of Create Payment Intent call payload to reference it later. However, any other key/value can also be added in the metadata object.

b. Important: Payment Intent id is the single source of truth in XPay Fusion and it will be used as a reference for the payment and any possible future actions such as refund, etc.

bash
curl --location 'https://xstak-pay-stg.xstak.com/public/v1/payment/intent' \
--header 'x-api-key: {{x-api-key}}' \
--header 'x-account-id: {{x-account-id}}' \
--header 'x-signature: {{x-signature}}' \
--header 'Content-Type: application/json' \
--data '{
   "amount": 10,
   "currency": "PKR",
   "payment_method_types": "card",
   "customer": {
       "name": "",
       "email": "",
       "phone": ""
   },
   "metadata":{
       "order_reference":""
   },
   "gateway_instance_id": "" // get from gateways in the settings
}'

c. Payment intent client secret and encryptionKey are included in the response of the Create Payment Intent call. This client secret and encryptionKey will be needed to capture the payment and pass in confirmPayment SDK method as shown below.

7. Once the frontend receives the payment intent from the backend, it will call the confirmPayment method of the XPay JS SDK and pass it the payment intent client secret.

Plain Text
const { message, error } = await xpay.confirmPayment("card", payment_intent_client_secret, customer = {name: ‘Dane Joe}, encryptionKey);

8. This confirmPayment SDK method will return a promise which will be resolved with a response object citing either success or failure message with the reason.

a. If successful, the frontend can request the backend to create the order. It is recommended that the backend retrieve the payment intent from XPay by calling the Retrieve Payment Intent API, verify the payment information and then create the order and inform the frontend.

b. If unsuccessful, the frontend should show an error message to the customer.

c. You can also set up webhooks to get notified of payment status asynchronously.

Form Validation Error Handling

We have an event named ready which will return true if all fields have been filled by user and form is ready to be submitted.

It will ensure that the website only enables the payment button if all required data has been added to the iframe form.

Plain Text
<code class=&#x27;preserveHtml&#x27; class=&#x27;preserveHtml&#x27; class=&#x27;preserveHtml&#x27; class=&#x27;preserveHtml&#x27; class=&#x27;preserveHtml&#x27; class=&#x27;preserveHtml&#x27; >xpay.on(&#x27;ready&#x27;, callbackFunction);</code>

Payment Intent

POSTCreate Payment Intent

https://xstak-pay-stg.xstak.com/public/v1/payment/intent

Supporting multiple payment gateways and currency MIDs with one XPay account and website: You can add multiple PSPs in XPay. Add a gateway instance in the store gateway settings and add the ID of the instance in the gateway_instance_id field. By adding this you can support multiple payment gateways and MIDs for currencies on the same XPay account.

Separate Authorization and Capture

To authorize an amount and capture it later, add the following in the PI API payload.

Plain Text
"capture_method": "manual"
HEADERS
x-api-key

{{x-api-key}}

x-account-id

{{x-account-id}}

x-signature

{{x-signature}}

Content-Type

application/json

Bodyraw
{
    "amount": 10,
    "currency": "PKR",
    "payment_method_types": "card",
    "customer": {
        "name": "",
        "email": "",
        "phone": ""
    },
    "shipping": {
        "address1": "",
        "city": "",
        "country": "", // complete country name
        "province": "",
        "zip": ""
    },
    "metadata": {
        "order_reference": "",
        "rule_attribute": "", //rule attribute
        "continue_payment": true // pass this value to perform payment if a customer's token limit of 5 is reached
    },
    "gateway_instance_id": "", // get from gateways in the settings
    "capture_method": "manual" // to authorize amount and capture later. If not required, then this can be removed.
}

POSTCapture an authorized amount

https://xstak-pay-stg.xstak.com/public/v1/payment/capture?pi_client_secret={{pi_client_secret}}

Supporting multiple payment gateways and currency MIDs with one XPay account and website: You can add multiple PSPs in XPay. Add a gateway instance in the store gateway settings and add the ID of the instance in the gateway_instance_id field. By adding this you can support multiple payment gateways and MIDs for currencies on the same XPay account.

Separate Authorization and Capture

To authorize an amount and capture it later, add the following in the PI API payload.

Plain Text
"capture_method": "manual"
HEADERS
x-api-key

{{x-api-key}}

x-account-id

{{x-account-id}}

x-signature

{{x-signature}}

Content-Type

application/json

PARAMS
pi_client_secret

{{pi_client_secret}}

Bodyraw
{
    "amount": 10 //only add an amount if you need to capture partial amount.
}
Loading