This package contains the Synapse Protocol Cross-Chain Swap and Bridging SDK
The Synapse SDK allows you to interact with Synapse Protocol router contracts deployed on 19 chains. It handles:
- Bridging operations (cross-chain swaps):
- Getting bridge quotes
- Initiating bridge transactions
- Tracking the status of bridge transactions
- On-chain swap operations:
- Getting swap quotes
- Initiating swap transactions
- Utilities for getting miscellaneous data related to protocol, fees, and chains
npm install @synapsecns/sdk-router
To use the SDK, first instantiate it with chain IDs and Ethereum providers:
import { SynapseSDK } from '@synapsecns/sdk-router'
import { JsonRpcProvider } from '@ethersproject/providers'
// Create providers for Ethereum, Arbitrum and Optimism
// Any Ethers.js provider can be used
const arbitrumProvider = new JsonRpcProvider('<arbitrum-provider-url>')
const ethereumProvider = new JsonRpcProvider('<ethereum-provider-url>')
const optimismProvider = new JsonRpcProvider('<optimism-provider-url>')
// Create lists of chain IDs and their matching providers
const chainIds = [1, 10, 42161]
const providers = [ethereumProvider, optimismProvider, arbitrumProvider]
const synapseSDK = new SynapseSDK(chainIds, providers)
Following concepts are abstracted away from the SDK consumer:
- Optional swaps on origin and destination chains.
- Bridge modules and their supported tokens.
The SDK consumer simply operates by getting quotes for sending tokenIn
on the origin chain and receiving tokenOut
on the destination chain. The SDK then finds the bridge modules that can fulfill this bridging intent.
Below is the example of how to get the list of quotes for sending 1000 USDC from Ethereum and receiving USDT on Arbitrum:
const bridgeQuotes: BridgeQuote[] = await synapseSDK.allBridgeQuotes(
// 1
originChainId,
// 42161
destChainId,
// Address of the token to start from on origin chain: 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48
tokenIn,
// Address of the token to end with on destination chain: 0xFd086bC7CD5C481DCC9C85ebE478A1C0b69FCbb9
tokenOut,
// Amount of tokens to bridge, in origin token decimals: 1_000_000_000
amountIn,
// Deadline for the transaction to be initiated on the origin chain, in seconds (optional)
deadline
)
The returned list is sorted by the maxAmountOut
field, so the first quote is the one yielding the highest amount of tokens on the destination chain.
If either of the input/output tokens is a native gas token (e.g. ETH on Ethereum/Arbitrum, AVAX on Avalanche, etc.), the specialized address 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE
should be used instead of the token address.
Note: The
bridgeQuote
method is a wrapper around theallBridgeQuotes
method.bridgeQuote
returns only the first quote from the list, whileallBridgeQuotes
returns the entire list.
BridgeQuote
objects are returned by the bridgeQuote
and allBridgeQuotes
methods. They contain the following fields:
export type BridgeQuote = {
// Address of the Router contract that user will be interacting with
routerAddress: string
// Expected final amount of tokens to be received on the destination chain by the user,
// if the bridge transaction is completed right after the quote is generated
maxAmountOut: BigNumber
// Query object for the origin chain
originQuery: Query
// Query object for the destination chain
destQuery: Query
// Estimated median time for the bridge transaction to be completed
estimatedTime: number
// Name of the "bridge module" that will be used to bridge the tokens.
// Supported values are "SynapseBridge", "SynapseCCTP" and "SynapseRFQ"
bridgeModuleName: string
// Amount of native gas tokens that user will receive on the destination chain
// on top of the token amount
gasDropAmount: BigNumber
}
Note:
Query
objects contain information about the optional swaps to be performed on behalf of the user on origin and destination chains. The exact composition of theQuery
object, as well as the concept of the optional swaps, is abstracted away from the SDK consumer. A collection of methods to modify theQuery
object is provided in theSynapseSDK
class, allowing the consumer to be unaware of the underlying object structure.
Some of the returned quotes may contain information about the optional swaps on origin and destination chains. As the liquidity composition may change over time, it is recommended to apply slippage to the quotes to account for the possible price changes. If no slippage is applied, the user transaction might be reverted due to insufficient funds. The default value for the slippage is 10 basis points (0.1%).
const { originQuery, destQuery } = await synapseSDK.applyBridgeSlippage(
// fields from the BridgeQuote object returned by the allBridgeQuotes method
bridgeQuote.bridgeModuleName,
bridgeQuote.originQuery,
bridgeQuote.destQuery,
// Numerator of the slippage percentage, optional (defaults to 10)
slipNumerator,
// Denominator of the slippage percentage, optional (defaults to 10000)
slipDenominator
)
Note: this method will not modify the original
Query
objects, but will return new ones. This allows to change the applied slippage without having to re-fetch the quotes.
Bridge quotes returned by the allBridgeQuotes
method come with the deadlines set in Query
objects. It is possible to further modify the deadlines before initiating the bridge transaction.
const { originQuery, destQuery } = synapseSDK.applyBridgeDeadline(
// fields from the BridgeQuote object returned by the allBridgeQuotes method
bridgeQuote.bridgeModuleName,
bridgeQuote.originQuery,
bridgeQuote.destQuery,
// New deadline for the origin chain (optional, default depends on the bridge module)
originDeadline,
// New deadline for the destination chain (optional, default depends on the bridge module)
destDeadline
)
We recommend to not modify the destination deadline, as the behavior of the bridge module with expired destination deadline varies depending on the module.
Note: this method will not modify the original
Query
objects, but will return new ones. This allows to change the deadlines without having to re-fetch the quotes.
At this stage, we assume that following steps have been performed:
- Bridge quotes have been fetched using the
allBridgeQuotes
orbridgeQuote
methods. - Slippage has been applied to the quotes using the
applyBridgeSlippage
method. - User has approved
routerAddress
to spendamountIn
oftokenIn
on the origin chain.
Use the bridge
method to form the payload for the bridge transaction:
// Populate the transaction data using the SDK
const { data, to, value } = await synapseSDK.bridge(
// User address on the destination chain that will receive the tokens
addressTo,
bridgeQuote.routerAddress,
// Parameters used to obtain the BridgeQuote object
originChainId,
destinationChainId,
tokenIn,
amountIn,
// Query objects with applied slippage
originQuery,
destQuery
)
data
and to
fields should be then used to initiate the bridge transaction on the origin chain, with value
being the amount of native gas tokens to be sent along with the transaction.
// Initiate the populated transaction, e.g. using ethers.js Signer
const tx = await signer.sendTransaction({
to,
data,
value,
})
Note: the
bridge
method will not initiate the transaction, but will only return the payload for the transaction. The transaction should be initiated by the user using their preferred method (e.g. MetaMask, WalletConnect, etc.).
In order to track the status of the bridge transaction, the consumer first needs to fetch its unique "Synapse Tx ID" using the getSynapseTxId
method:
const synapseTxId = await synapseSDK.getSynapseTxId(
// Chain ID of the ORIGIN chain
originChainId,
bridgeQuote.bridgeModuleName,
// Transaction hash of the bridge transaction on the origin chain
txHash
)
synapseTxId
value stays the same for a given bridge transaction, so there is no need to re-fetch it every time the status is checked.
The status of the bridge transaction can then be checked using the getBridgeTxStatus
method:
const status: boolean = await synapseSDK.getBridgeTxStatus(
// Chain ID of the DESTINATION chain
destChainId,
bridgeQuote.bridgeModuleName,
synapseTxId
)