- You still can use v1 trade sdk . Go to v1.0
- How to migration from v1 to v2. Go to migration guide
- Permissionless DEX Engine Product Introduction
- AWS Onboarding Process
- Download the Latest SDK
- Configs
- Events
- Proxy Setup
- FAQ
We take AWS service as an example to demostrate the whole process that you can get onboarded. You can use other service on other cloud platform of course.
Before we start, make sure you set up AWS CLI https://docs.aws.amazon.com/cli/latest/userguide/getting-started-install.html
Download the latest version of SDK zip file in Releases.
Unzip the file and upload those 2 folders(static, sdk) to AWS S3 space.
wget <zip-url> # download zip file from github
unzip <fileanme.zip> # unzip the zip file
aws s3 mb s3://apx-s3-test # create s3 bucket: apx-s3-test
# upload sdk folder to s3, and grant to all user
aws s3 sync ./sdk s3://apx-s3-test/sdk --grants read=uri=http://acs.amazonaws.com/groups/global/AllUsers
# upload static folder to s3, and grant to all user
aws s3 sync ./static s3://apx-s3-test/static --grants read=uri=http://acs.amazonaws.com/groups/global/AllUsers
# initialize npm and packages
npm init
npm install local-web-servercreate lws.config.js and paste below
module.exports = {
rewrite: [
{
from: "/bapi/(.*)",
to: "https://www.apollox.finance/bapi/$1",
},
{
from: "/fapi/(.*)",
to: "https://www.apollox.finance/fapi/$1",
},
{
from: "/cloud-futures/(.*)",
to: "https://static.apollox.com/cloud-futures/$1",
},
{
from: "/api/(.*)",
to: "https://static.apollox.com/api/$1",
},
],
directory: "./",
logFormat: "stats",
};Create a index.html(can be skipped if you already have) Import the static files that you just uploaded and init the trading page.
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width" />
<title>My Futures Exchange</title>
<link rel="icon" type="image/x-icon" href="favicon.ico" />
<style>
html,
body {
margin: 0;
padding: 0;
width: 100vw;
height: 100vh;
}
#app {
width: 100%;
height: 100%;
overflow: hidden;
}
</style>
<script src="/sdk/sdk.4.0.0.js"></script>
</head>
<body>
<div id="app"></div>
<script>
const { origin, protocol } = window.location;
const isDev = protocol === "http:";
FuturesSDK.createTradeUI({
container: document.getElementById("app"),
config: {
staticBaseUrl: `/static/`,
futuresWsHost: "wss://fstream.apollox.finance/plain",
// It's important!!! DO NOT use origin in production because request will be proxied by client's server to APX domain, then `/common-check-ip` api will detect ip through client's server IP, that'll be wrong. It should detect user's ip.
apiBaseUrl: isDev ? origin : undefined, // in production, you don't need to configurate this, it's default as APX's url
headerConfig: {},
defaultTheme: "light",
customCssUrl: "",
},
state: {
symbol: "BTCUSD",
lng: "en",
},
});
</script>
</body>
</html>Then, run the local server and open http://127.0.0.1:3333, you will see your exchange website is ready!
./node_modules/.bin/ws --port 3333
Deploy the index.html to AWS sever under your domain.
Now, you can visit trading page.
aws s3 cp ./index.html s3://apx-s3-test/index.html --grants read=uri=http://acs.amazonaws.com/groups/global/AllUsers
You can find SDK in the release list Releases list: https://github.com/apollox-finance/broker-web-sdk/releases, and download the latest zip file
Unzip the file and upload those 2 folders to your CDN.
Import sdk.js and initialize SDK.
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width" />
<title>My Futures Exchange</title>
<link rel="icon" type="image/x-icon" href="favicon.ico" />
<style>
html,
body {
margin: 0;
padding: 0;
width: 100vw;
height: 100vh;
}
#app {
width: 100%;
height: 100%;
overflow: hidden;
}
</style>
<script src="/sdk/sdk.2.0.0.js"></script>
<!-- the url of sdk in your CDN -->
</head>
<body>
<div id="app"></div>
<script>
const { origin, protocol } = window.location;
const isDev = protocol === "http:";
FuturesSDK.createTradeUI({
container: document.getElementById("app"),
config: {
staticBaseUrl: `/static/`, // the url of static folder in your CDN
// It's important!!! DO NOT use origin in production because request will be proxied by client's server to APX domain, then `/common-check-ip` api will detect ip through client's server IP, that'll be wrong. It should detect user's ip.
apiBaseUrl: isDev ? origin : undefined, // in production, you don't need to configurate this, it's default as APX's url
graphqlTemplateUrl: 'https://api.thegraph.com/subgraphs/name/apollx-apx/apollox-perp-{{network}}', // graphql template url
headerConfig: {},
defaultTheme: "light",
customCssUrl: ""
},
state: {
symbol: "BTCUSD",
lng: "en",
},
});
</script>
</body>
</html>type Config = {
staticBaseUrl: string;
apiBaseUrl: string;
graphqlTemplateUrl?: string,
brandName?: string;
brokerId: number;
i18nBaseUrl: string;
configBaseUrl: string;
enableThemeToggle?: boolean;
isTestnet?: boolean;
supportNetworks?: SupportedNetwork[];
supportLanguages?: SupportedLanguage[];
headerConfig: HeaderConfig;
shareImgFolder?: string;
loadingImage?: LoadingImage;
notificationPosition?: NotificationPosition;
defaultTheme?: "dark" | "light";
customCssUrl?: string;
};the base url of your static files, for example: the complete url of um_futures.{version}.html would be: ${staticBaseUrl}um_futures.{version}.html
the base url of the api calls, for example: ${apiBaseUrl}/fapi/v1/exchangeInfo
the brand name of customer
will be used when user open position, recorded on the chain
the i18n base url
the config base url
- values ['bnb', 'arb', 'opbnb', 'base']
- default is ['bnb', 'arb']
export type SupportedNetwork = "bnb";default is all languages
export type SupportedLanguage =
| "de"
| "uk-UA"
| "ru"
| "fr"
| "zh-CN"
| "ko"
| "zh-TW"
| "pt-BR"
| "en"
| "es"
| "es-LA"
| "th"
| "ja"
| "pl"
| "tr";the config to set up header
// all navTo support `%lng%` placeholder, can be relpaced with language
type MenuItem = {
text: string | Record<SupportedLanguage, string>;
navTo: string;
navTarget?: "_blank" | "_parent"; // default '_blank', the target attribute if <a> element, see https://www.w3schools.com/tags/att_a_target.asp
};
type HeaderConfig = {
disable?: boolean; // hide header
logo?: {
darkImgUrl: string;
darkRWDImgUrl: string;
lightImgUrl: string;
lightRWDImgUrl: string;
navTo?: string;
navTarget?: "_blank" | "_parent"; // default '_parent'
};
menu?: {
data: MenuItem[];
placement?: "left" | "center"; // default 'center'
};
token?: {
imgUrl: string; // the image url of the token
navTo?: string; // the href url on the token
navTarget?: "_blank" | "_parent"; // default '_blank'
pricePrecision: number; // the precision u want to display
network: SupportedNetwork; // token network
lpPairAddress: string; // the contract address of the pair in liquidity pool, for example: https://bscscan.com/address/0xa0ee789a8f581cb92dd9742ed0b5d54a0916976c is the address of APX/BUSD in pancake pool
lpBaseAddress: string; // the base asset address in the lp pair
lpQuoteAddress: string; // the quote asset address in the lp pair
quote?: {
// Only config it when you must compute twice. For APX token which has configured APX/BUSD lp, add BUSD/USDT lp here, then you can get USDT price of APX
lpPairAddress: string;
lpBaseAddress: string;
lpQuoteAddress: string;
};
};
customerService?: {
navTo: string;
navTarget?: "_blank" | "_parent"; // default '_blank'
};
};Details
Examples// for bnb chain, BABY/BUSD pair
lpPairAddress: '0xe730c7b7470447ad4886c763247012dfd233baff', // BABY/USDT lp address
lpBaseAddress: '0x53e562b9b7e5e94b81f10e96ee70ad06df3d2657', // BABY contract address
lpQuoteAddress: '0x55d398326f99059ff775485246999027b3197955', // BUSD contract address
// for eth chain, UNI/DAI pair
lpPairAddress: '0xc34fecf98a7c48ee48288fc163ca5804d3235be5', // UNI/DAI lp address
lpBaseAddress: '0x1f9840a85d5af5bf1d1762f925bdaddc4201f984', // UNI address
lpQuoteAddress: '0x6b175474e89094c44da98b954eedeac495271d0f', // DAI address
tokenNetwork: 'eth',
// for arb1 chain, WETH/USDC pair
lpPairAddress: '0x905dfcd5649217c42684f23958568e533c711aa3', // WETH/USDC lp address
lpBaseAddress: '0x82af49447d8a07e3bd95bd0d56f35241523fbab1', // WETH address
lpQuoteAddress: '0xff970a61a04b1ca14834a43f5de4533ebddb5cc8', // USDC address
tokenNetwork: 'arb1',default value: static/images/share-poster
set up the images for share poster the image list:
${shareImgFolder}/light_logo.png${shareImgFolder}/dark_logo.png${shareImgFolder}/pc_light_win.png${shareImgFolder}/pc_dark_win.png${shareImgFolder}/pc_light_loss.png${shareImgFolder}/pc_dark_loss.png${shareImgFolder}/mobile_light_win.png${shareImgFolder}/mobile_dark_win.png${shareImgFolder}/mobile_light_loss.png${shareImgFolder}/mobile_dark_loss.png
the loading image config, it will show before the trade page render
type LoadingImage = {
url: string;
width?: number;
backgroundColor?: string;
};the position of all notification toast
type NotificationPosition = {
horizontal: "left" | "right";
vertical: "top" | "bottom";
};to configure the initial theme value, theme value will then be stored in cookies with key 'theme'
Removed in 3.0.
use customCssUrl instead, you can set css variable as below:
:root {
--font-site-main: Roboto;
}Removed in 3.0.
use customCssUrl instead, you can set as below:
@font-face {
font-family: 'Roboto';
font-style: normal;
font-weight: 300;
font-display: swap;
src: url('https://static.apollox.com/cloud-futures/static/fonts/Roboto/Roboto-Light.ttf') format('truetype');
}We support theme styles, you can customize the colors for light mode or dark mode, see color palette.
Removed in 3.0. Use customCssUrl instead
experimental feature, for customized style.
Remove in 3.0. Use customCssUrl instead
custom css url (support since v2.0.3). since we deprecated our darkPalette/lightPalette/variants in v3.0, you have to customize style through this new property. default template we used
We also support several event to let u can customize the exchange more flexible
FuturesSDK.eventListener.on("symbolChange", (data) => {
console.log("symbol change", data.symbol);
window.history.replaceState(
{},
null,
`/${getLngFromUrl()}/futures/${data.symbol}`
); // assume the url is formaated as /:lng/futures/:symbol
});
FuturesSDK.eventListener.on("invalidSymbol", () =>
console.log("invalid symbol")
);
FuturesSDK.eventListener.on("startLoad", () =>
console.log("startLoad event received")
);
FuturesSDK.eventListener.on("lngChange", (data) => {
console.log("lng change", data.lng);
window.history.pushState(
{},
null,
`/${data.lng}/futures/${getSymbolFromUrl()}`
); // assume the url is formaated as /:lng/futures/:symbol
});https://api.your.domain/bapi/* --- proxy to --> https://www.apollox.finance/bapi/*
https://api.your.domain/fapi/* --- proxy to --> https://www.apollox.finance/fapi/*
wss://fstream.your.domain/compress/stream --- proxyt to --> wss://fstream.apollox.finance/compress/stream
https://static.your.domain --- proxy to --> https://static.apollox.com
Set config.enableThemeToggle to false.
Set config.customCssUrl
There are few configs related to the price display:
config.headerConfig.token.pricePrecision: the precision u want to displayconfig.headerConfig.token.lpPairAddress: the contract address of the pair in liquidity pool, for example: 0xa0ee789a8f581cb92dd9742ed0b5d54a0916976c is the address of APX/BUSD in pancake poolconfig.headerConfig.token.lpBaseAddress: the base asset address in the lp pairconfig.headerConfig.token.lpQuoteAddress: the quote asset address in the lp pair
Before is only need if you want to compute twice. For APX token which has configured APX/BUSD lp, add BUSD/USDT lp here, then you can get USDT price of APX
config.headerConfig.token.quote.lpPairAddress: the contract address of the pair in liquidity pool, for example: 0xa0ee789a8f581cb92dd9742ed0b5d54a0916976c is the address of APX/BUSD in pancake poolconfig.headerConfig.token.quote.lpBaseAddress: the base asset address in the lp pairconfig.headerConfig.token.quote.lpQuoteAddress: the quote asset address in the lp pair
Our SDK just generates an iframe into your specified DOM, so you can change the title or favicon in your HTML as normal. See w3school
Since we have two themes, dark and light, and also care about the RWD, we have 5 configs related to the logo.
config.headerCnfig.logo.darkImgUrl: the image url for dark themeconfig.headerCnfig.logo.lightImgUrl: the image url for light theme
There are some classes that you can override them with customCssUrl.
.button {
/* your styles */
}
.sheet-content {
/* your styles */
}
.input-text-field,
.input-base-input {
/* your styles */
}
.dialog-content {
/* your styles */
}
.order-form-card {
/* your styles */
}
.radio-option {
/* your styles */
}
.currency-input {
/* your styles */
}
.direction-switch-long {
/* your styles */
}
.direction-switch-short {
/* your styles */
}- Go to releases and download the latest SDK zip.
- Unzip the file, and upload to your CDN.
- Update the sdk version in your HTML file
<script src="/sdk/sdk.2.0.0.js"></script>
Becasue SDK V2 is permissionless (that means no more whitelist as SDK V1), Please use our production api
wss://fstream.apollox.finance/plain replace of wss://perp-qa.pancakeswap.finance/plain/stream (which been used in SDK v1 QA).
if using wss://fstream.apollox.finance/plain still return ws error, try to clean site data under Application tab in chrome devtools maybe work.
br => brotil
gz => gzip
They are the two most common compression algorithms, widely used in modern web. They help user can download data faster. Check out the article to know more: https://www.siteground.com/blog/brotli-vs-gzip-compression/
Same as before, just ensure you download the br.zip or gz.zip version.
- Go to releases and download the latest SDK zip file.
- Unzip the file, and upload to your CDN.
- Update the sdk version in your HTML file
<script src="/sdk/sdk.2.0.0.js"></script>
- Make sure
/:lng/futures/:symbolcan redirect to your html. Normally, this is configured by your infra team. - Parse the url to get
lng,symbol
function getLngFromUrl() {
return location.pathname.split("/")[1] || "en";
}
function getSymbolFromUrl() {
return location.pathname.split("/")[3] || "BTCUSD";
}- Give the SDK lng and symbol that you get from url
FuturesSDK.createTradeUI({
// ... other configs
state: {
symbol: getSymbolFromUrl(),
lng: getLngFromUrl(),
},
});- Listen
lngChangeandsymbolChangeevents, to change the pathname when lng or symbol changes.
FuturesSDK.eventListener.on("symbolChange", (data) => {
window.history.replaceState(
{},
null,
`/${getLngFromUrl()}/futures/${data.symbol}`
);
});
FuturesSDK.eventListener.on("lngChange", (data) => {
window.history.pushState(
{},
null,
`/${data.lng}/futures/${getSymbolFromUrl()}`
);
});You can check out the full code here: https://github.com/apollox-finance/broker-web-sdk/blob/main/examples/basic/index.html
