Aggregator API & SDK
Build apps on TON Blockchais like a pro.
SDK for swap.coffee DEX aggregator
Installation
For JavaScript/TypeScript please use the SDK as stated below:
For any other languages, you can integrate with our Rest API directly.
Concepts
Here are some basic concepts you need to know to work with our SDK:
- Asset: an asset is a token on a blockchain. It can be a native token (like TON on TON blockchain) or a token on a smart contract, named
jetton
(like USDT on TON blockchain). Each asset has a unique address on the blockchain. - Route: a route is a set of paths.
- Path: a path is a list of tokens (or, more specifically, of liquidity pools) that can be swapped one to another. For example, if you want to swap TON to USDT, you can do it directly
(TON -> USDT)
or you can do it in two steps(TON -> CES -> USDT)
. The first path has one swap, the second path has two swaps. - Price impact: when you perform a swap, the price of the token can change. The price impact depends on the amount you want to swap and the liquidity of the token. If you swap a small amount, the price impact will be low. If you swap a large amount, the price impact will be high.
- Price slippage: the price of a token can change over time. When you swap tokens, you can set a slippage tolerance. If the price of the token changes more than the slippage tolerance, the transaction will fail.
- Transaction: a transaction is a set of messages that are sent to the blockchain. Each message can be a token transfer, a contract call, etc. When you swap tokens, you need to build a transaction that will swap the tokens for you.
How the DEX aggregator works
- You provide the input token, the output token and the amount you want to swap.
- We build the optimal route for you. This route can be direct
(input token -> output token)
or indirect(input token -> token1 -> token2 -> ... -> output token)
. By splitting the amount of the input token between several liquidity pools on several decentralized exchanges (DeDust, STONfi), we can reduce the price impact and get a better price for you. - We build the transactions that will swap the tokens.
- You send the transactions to the blockchain.
- If slippage tolerance is exceeded, the transaction is reverted.
- You receive the output token in your wallet.
Basic usage
Swapping assets using our SDK and TonConnect SDK:
Advanced usage
There are extra settings that you’re able to modify in case you want more control over how routes are built.
The following parameters can be configured:
max_length
- maximum path length in tokens, default is 3. If specified, it must be an integer from [2; 5] range. Value of 2 indicates that route can contain only direct-swaps paths(input token -> output token)
. Value of 3, however, states that route may also contain paths with one intermediate token(input token -> token1 -> output token)
.
max_splits
- maximum number of splits, default is 4. If specified, it must be an integer from [1; 20] range. Each split is a distinct path, therefore a distinct blockchain-message that will need to be sent. Currently, wallet contract implementations up to version 4 support not more than 4 messages in a single transaction. Starting from version 5, the number of message in a single transaction has been raise to 255(but ours upper limit is 20)
.
-
pool_selector
is an optional property that allows you to control which liquidity pools can be used whilst building the route.blockchains
- allows you to limit liquidity pools to given blockchains only. Currently swap.coffee does not support cross-chain swaps, but who knows what will happen tomorrow 🙂 If you’re certain that you won’t like performing cross-chain swaps in the future, set this property to [“ton”] value. By default, all supported blockchains are used.dexes
- allows you to limit liquidity pools to given dexes only. Supported values: [“stonfi”] and [“dedust”]. By default, all supported DEXes are used.max_volatility
- allows you to filter out those liquidity pools, whose price volatility over the last 15 minutes exceeded the given value. The higher the value, the more volatile liquidity pools may be used.
Difference between direct and indirect routes
Direct routes without intermediate tokens are usually more safe, but less profitable. Indirect routes with intermediate tokens are usually more profitable, but less safe, because the price of intermediate tokens can change during the transaction and chance that your may receive intermediate tokens instead of the desired output token is higher.
It’s called “slippage” - the difference between the expected price and the actual price. For example, when you set slippage
to 0.1, it means that transaction within blockchain must be aborted if any of the swaps lead to obtaining less than 1 - 0.1 = 0.9 = 90%
of amount predicted by swap.coffee backend in returned route. Due to async nature of TON blockchain, such slippage
abortion can also lead to a situation where the sender is left with neither the input nor the output tokens, but an intermediate one.
There are also tokens that can’t be swapped directly, because they don’t share a liquidity pool. Therefore, if you setup parameters so that only direct swaps allowed (i.e. if max_length
is equal to 2), an empty route will be built for you.
ExactOut swaps
If you want to receive an exact amount of output tokens, you can specify the output_amount
parameter. Those swaps, in comparison to direct swaps (when you specify input_amount
instead), may be significantly less profitable. They also do not support splitting, which means that resulting route (if exists) will always contain a single path.
Checking transaction results
Along with the transaction built, you receive the route_id
field. It can be used to check the result of the route’s transactions on the blockchain.
Basic usage
SDK contains inbuilt method for waiting execution results of route’s transactions. Promise will be resolved when all transactions are in terminal status (one of succeeded
, failed
or timed_out
).
Advanced usage
By passing route_id
into getTransactionsResult()
you can obtain current state of the route’s transactions execution status. Response is as follows:
Each element of the returned array corresponds to a path at the same index that you provided buildTransactionsV2
method with.
Each element of the steps
corresponds to a single swap within transaction. For example, if transaction is
A -> B -> C
, then first entry in steps describes A -> B
swap, and the second one describes B -> C
.
TX_STATUS
is a status of a single transaction (related to a single path):pending
indicates that none of transaction’s swaps have occurred in the blockchain yet.partially_complete
- some (but not all) of transaction’s swaps have occurred in the blockchain.succeeded
- all of transaction’s swaps have successfully executed in blockchain.failed
- one of transaction’s swaps has been aborted due to slippage tolerance.timed_out
- our service could not wait enough for at least one of transaction’s swaps to be present in the blockchain. This happens if more than 5 minutes have passed since the transaction was generated by our service until the user sent it, or if more than 3 minutes have passed since the previous swap for this transaction appeared in the blockchain.
SWAP_STATUS
is a status of a single swap within a transaction:pending
indicates that this swap didn’t happen yet.cancelled
- swap has been cancelled due to one of the previous swaps within same transaction failure or timeout.succeeded
- swap successfully executed.failed
- swap has been aborted due to slippage tolerance.timed_out
- our service could not wait enough for this swap to be present in the blockchain.
- Transaction’s
input
andoutput
fields are presented only if at least one swap has already moved to statussucceeded
orfailed
. - Swap’s
input
andoutput
fields are presented only if swap’s status issucceeded
orfailed
. On failure (abortion due to slippage tolerance)output
is equal toinput
.
The route status can be requested within 2 hours from the moment route_id
(and transactions) is generated via
buildTransactionsV2
.
If all transactions are in terminal status (one of succeeded
, failed
or timed_out
), further receipt of this
route’s status will be possible only for 1 minute, after which it will be invalidated from our service’s cache.
Custom fees (for partners only)
Starting January 2025 various DEXes stopped sharing their fees with aggregators like swap.coffee. We understand the desire of our partners to continue earning on user transactions, and therefore we provide functionality for charging additional fees.
To do so, it is enough to pass the custom_fee
property to the transaction construction method:
As you can see, there are multiple opportunities to set up custom fees collection. Let’s dive into them!
First of all, custom_fee
specification looks as follows:
You can charge fixed, volume-percentage based or even both fees every time user performs a swap throughout your service. Each method adds additional transaction to the list given to the user for signing. Regardless of the method chosen and tokens of the swap, fees are always charged in TONs.
Please, keep in mind that whatever value you set, it is not the amount you will receive: all custom fees are divided equally between our partners and swap.coffee itself, which means that you will get only half of what you request from users there.
Fixed fee
This method is for those cases when you want to charge the same level of commission on all trades through your service,
regardless of the trade volume. Please keep in mind that fixed_fee
is specified in nano-TONs and must not be less than
10_000_000
, which equals 0.01 TON.
Volume-percentage based fee
If you want to charge more commission the larger the trade the user makes, this method is for you.
percentage_fee
is specified in a parts on 1/1_000_000
: 1
stands for 0.0001%
and 1_000_000
stands for 100%
.
Whilst using this method, you must also specify min_percentage_fee_fixed
property, which, as much as fixed_fee
,
must not be less than 10_000_000
, which equals 0.01 TON. It is also up to you whether you want to limit fee value
from the above by setting max_percentage_fee_fixed
property.
Overall, the final formula for volume-percentage based fee is as follows:
Thinking of max splits
Because of custom fees being withdrawn as a separate transactions, it is very important to keep in mind that wallet contracts have a limitation of how much transactions they may send at once: no more than 4 for walletV4 and no more than 255 for walletV5.
That’s why, when adding custom fees, you must guarantee that the route for which you are building transactions
contains no more than max_wallet_txs - 1
splits: it is configurable during buildRoute
invocation (as shown above).
Default value of max_splits
for buildRoute
is 4. So, if you haven’t specified that property before, it’s time to
start doing so. If you have tools to determine user’s wallet version, you may set it to 3
for walletV4 and to 20
for walletV5. Otherwise, you can just leave it at 3
, because in most circumstances this is more than enough for a
good route to be built.
API Limitations
Our API is public, which means it allows unauthorized access. However, this access has its limitations:
input_token
andoutput_token
must be different, so you are not allowed to find cycles. Moreover, none of the returned routes will contain cycles within them (so no paths liketoken1 -> token2 -> token1 -> token3
will be returned).- We believe that the average load of 1 request per second per IP is more than acceptable. If you generate more, the API
will return a
429 Too Many Requests
error. - The entire commission received through the execution of transactions built by swap.coffee goes to swap.coffee.
If you are considering integration with our product, building an arbitrage service or generating a higher level of load, please contact us via Telegram @swap.coffee DEV Chat.