Skip to main content

Gateway Orders

In this guide, we will create an off-ramp order to convert USDT to Nigerian naira(NGN).

Setup

To create an off-ramp order, we have to first import certain values from ethers and define both the stablecoin's token contract and Paycrest's Gateway contract.

In this example, we're performing the transaction on the arbitrum network and we're using the appropriate contract addresses and ABI per contract. We then define variables for the transaction's exchange rate and the user's bank account.

Import Values

import { ethers, Contract, formatUnits, parseUnits, BigNumber, ZeroAddress } from "ethers";

// define token contract and gateway contract
const usdtContract = {
address: "0xdac17f958d2ee523a2206206994597c13d831ec7",
abi: [
"function transfer(address _to, uint256 _value) public returns (bool)",
];
}

const gatewayContract = {
address: '0xE8bc3B607CfE68F47000E3d200310D49041148Fc'
abi: '[{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Paused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Unpaused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"previousAdmin","type":"address"},{"indexed":false,"internalType":"address","name":"newAdmin","type":"address"}],"name":"AdminChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bytes32","name":"splitOrderId","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"orderId","type":"bytes32"},{"indexed":true,"internalType":"address","name":"liquidityProvider","type":"address"},{"indexed":false,"internalType":"uint96","name":"settlePercent","type":"uint96"}],"name":"OrderSettled","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"fee","type":"uint256"},{"indexed":true,"internalType":"bytes32","name":"orderId","type":"bytes32"}],"name":"OrderRefunded","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint64","name":"protocolFee","type":"uint64"}],"name":"ProtocolFeeUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint8","name":"version","type":"uint8"}],"name":"Initialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"beacon","type":"address"}],"name":"BeaconUpgraded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"implementation","type":"address"}],"name":"Upgraded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferStarted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":true,"internalType":"address","name":"token","type":"address"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"protocolFee","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"orderId","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"rate","type":"uint256"},{"indexed":false,"internalType":"string","name":"messageHash","type":"string"}],"name":"OrderCreated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"sender","type":"address"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"SenderFeeTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"treasuryAddress","type":"address"}],"name":"SetFeeRecipient","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"what","type":"bytes32"},{"indexed":true,"internalType":"address","name":"treasuryAddress","type":"address"}],"name":"ProtocolAddressUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"what","type":"bytes32"},{"indexed":true,"internalType":"address","name":"value","type":"address"},{"indexed":false,"internalType":"uint256","name":"status","type":"uint256"}],"name":"SettingManagerBool","type":"event"},{"inputs":[],"name":"acceptOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getFeeDetails","outputs":[{"internalType":"uint64","name":"","type":"uint64"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pendingOwner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"uint96","name":"_rate","type":"uint96"},{"internalType":"address","name":"_senderFeeRecipient","type":"address"},{"internalType":"uint256","name":"_senderFee","type":"uint256"},{"internalType":"address","name":"_refundAddress","type":"address"},{"internalType":"string","name":"messageHash","type":"string"}],"name":"createOrder","outputs":[{"internalType":"bytes32","name":"orderId","type":"bytes32"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"isTokenSupported","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"_orderId","type":"bytes32"}],"name":"getOrderInfo","outputs":[{"components":[{"internalType":"address","name":"sender","type":"address"},{"internalType":"address","name":"token","type":"address"},{"internalType":"address","name":"senderFeeRecipient","type":"address"},{"internalType":"uint256","name":"senderFee","type":"uint256"},{"internalType":"uint256","name":"protocolFee","type":"uint256"},{"internalType":"bool","name":"isFulfilled","type":"bool"},{"internalType":"bool","name":"isRefunded","type":"bool"},{"internalType":"address","name":"refundAddress","type":"address"},{"internalType":"uint96","name":"currentBPS","type":"uint96"},{"internalType":"uint256","name":"amount","type":"uint256"}],"internalType":"struct IGateway.Order","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"_splitOrderId","type":"bytes32"},{"internalType":"bytes32","name":"_orderId","type":"bytes32"},{"internalType":"address","name":"_liquidityProvider","type":"address"},{"internalType":"uint64","name":"_settlePercent","type":"uint64"}],"name":"settle","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"what","type":"bytes32"},{"internalType":"address","name":"value","type":"address"},{"internalType":"uint256","name":"status","type":"uint256"}],"name":"settingManagerBool","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"what","type":"bytes32"},{"internalType":"address","name":"value","type":"address"}],"name":"updateProtocolAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_fee","type":"uint256"},{"internalType":"bytes32","name":"_orderId","type":"bytes32"}],"name":"refund","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint64","name":"_protocolFeePercent","type":"uint64"}],"name":"updateProtocolFee","outputs":[],"stateMutability":"nonpayable","type":"function"},{"stateMutability":"payable","type":"fallback"},{"stateMutability":"payable","type":"receive"}]'
}

const { address: usdtAddress, abi: usdtAbi } = usdtContract;
const { address: gatewayAddress, abi: gatewayAbi } = gatewayContract;
let getRate, getAccount;

Define Variables

Next, let's define our user's signer that'll be used to initiate the contracts. One thing to note is that you require the provider will to get the signer.

// Defining provider, signer, and the contracts

const provider = new ethers.BrowserProvider(window.ethereum);

// Initiating signers
const signer = await provider.getSigner();

// Initiating contracts
const gateway = new Contract(gatewayAddress, gatewayAbi, signer)
const usdtAsset = new Contract(
usdtAddress,
usdtAbi,
signer,
);

Let's verify the user's account by taking their account number and bank to generate the account name. Secondly, let's get the current exchange rate for amount of naira(NGN) per USDT that would be exchanged for this transaction.

Both requests are performed in parallel so it can be completed faster.

P.S: Keep in mind that for the bankData you can know the correct institution code from here

Fetch naira rate and bank verification

// get the  nairaRate and verify account number
const nairaRate = "https://api.paycrest.io/v1/rates/usdt/1/ngn";
const accountName = "https://api.paycrest.io/v1/verify-account";

const bankData = {
institution: "KUDANGPC",
accountIdentifier: "12345678953"
};

try {
const [nairaRate, accountName] = await Promise.all([
fetch(nairaRate), // GET request
fetch(accountName, {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify(bankData)
}) // POST request
]);

getRate = await nairaRate.json();
getAccount = await accountName.json();

console.log("naira rate response:", getRate.data);
console.log("get account response:", getAccount.data);
} catch (error) {
console.error("Error fetching data:", error);
}

Bank Data Encryption

Next, encrypt the user's bank data in a message hash. Here, we fetch the aggregator's public key using fetchAggregatorPublicKey(). Then, we generate the hash by passing the bank data and key into the publicKeyEncrypt() function.

// Encrypt arbitrary data with a public key
async function publicKeyEncrypt(data, publicKeyPEM) {
// First, we need to convert PEM to a format Web Crypto API can use
function pemToArrayBuffer(pem) {
const b64 = pem.replace(/-----BEGIN PUBLIC KEY-----/, '')
.replace(/-----END PUBLIC KEY-----/, '')
.replace(/\s/g, '');
const binary = atob(b64);
const arr = new Uint8Array(binary.length);
for (let i = 0; i < binary.length; i++) {
arr[i] = binary.charCodeAt(i);
}
return arr.buffer;
}

// Convert PEM to ArrayBuffer
const publicKeyBuffer = pemToArrayBuffer(publicKeyPEM);

// Import the public key
const publicKey = await self.crypto.subtle.importKey(
'spki',
publicKeyBuffer,
{
name: 'RSA-OAEP',
hash: 'SHA-256',
},
true,
['encrypt']
);

// Encrypt the data
const encoder = new TextEncoder();
const dataBuffer = encoder.encode(JSON.stringify(data));
const encrypted = await self.crypto.subtle.encrypt(
{
name: 'RSA-OAEP'
},
publicKey,
dataBuffer
);

// Convert the encrypted data to base64 for easy transmission
return btoa(String.fromCharCode.apply(null, new Uint8Array(encrypted)));
}

// Fetch public key of aggregator
const fetchAggregatorPublicKey = async () => {
try {
const response = await fetch("https://api.paycrest.io/v1/pubkey");
if (!response.ok) {
throw new Error(`HTTP error! status: ${response.status}`);
}
return await response.json();
} catch (error) {
console.error("Error fetching aggregator public key:", error);
throw error;
}
};

// Encrypt recipient details
const recipient = {
accountIdentifier: "12345678953",
accountName: getAccount.data,
institution: "KUDANGPC",
providerId: "PROVIDER_ID", // You'll need to contact us to receive an ID
memo: "N/A",
};

const publicKey = await fetchAggregatorPublicKey();
const messageHash = await publicKeyEncrypt(recipient, publicKey.data);

console.log(publicKey, messageHash);

Approve USDT token

Before we can debit USDT from the user's wallet, we need them to confirm the approve method on the usdtAsset contract by passing in the amount(usdtAmount) and the gatewayAddress.

We define the amount by using parseUnits from ethers. It takes in the amount in string and the token's decimal. In this case, USDT's decimal is 6.

// approve on USDT
const usdtAmount = parseUnits('100', 6);

const approveTx =
await usdtAsset.approve(gatewayAddress, usdtAmount);

const approveRct = await approveTx.wait();

console.log(approveRct.logs)

Create Order

Finally, we can now make the order. We make use of the createOrder method on the gateway.

// create order through the gateway contract

const refundAddress = "0x344de...." // this is the user's address to pay into for refunds

try {
const createOrderTx = await gateway.createOrder(
usdtAddress,
parseUnits('10', 6),
Number(getRate.data),
ZeroAddress,
0,
refundAddress,
messageHash
)

const createOrderRct = await createOrderTx.wait();

const orderLog = createOrderRct.logs.find((l) => l.fragment?.name === "OrderCreated")

console.log("emittedCreatedOrder", {
refundAddress: orderLog.args[0],
token: orderLog.args[1],
amount: orderLog.args[2],
fee: orderLog.args[3],
orderId: orderLog.args[4],
rate: orderLog.args[5],
messageHash: orderLog.args[6]
})
} catch (error) {
parseEthersError(error, sablier)
console.error(error)
}

We pass in the token, amount, rate, senderFeeRecipient, senderFee, refundAddress, messageHash respectively. Then we get back the emittedCreatedOrder event that's logged.