Link Custom Tokens Deployed Across Multiple Chains into Interchain Tokens
Custom ERC-20 tokens deployed on multiple chains with specific minting policies, ownership structures, rate limits, and other bespoke functionalities can be turned into Interchain Tokens through the Interchain Token Service.
In this tutorial, you will:
- Link custom tokens deployed across multiple chains into Interchain Tokens
- Deploy a simple ERC-20 token on the Fantom chain
- Deploy a simple ERC-20 token on the Polygon chain
- Deploy a token manager on both Fantom and Polygon
- Transfer mint access to the Token Manage Address on both Fantom and Polygon
- Transfer your token between Fantom and Polygon
Prerequisites
You will need:
- A basic understanding of Solidity and JavaScript
- A wallet with FTM and MATIC funds for testing. If you don’t have these funds, you can get FTM from the Fantom faucet and MATIC from the Polygon faucets (1, 2).
Deploy an ERC-20 token on the Fantom and Polygon testnets
Deploy the following SimpleCustomToken
on the Fantom and Polygon testnets.
This code utilizes OpenZeppelin’s libraries to create a custom ERC20 token with functionalities for minting, burning, and access control. The token includes a minter role, which enables designated addresses to mint or burn tokens. Additionally, it incorporates ERC20Permit for gasless transactions. The contract starts with a predefined supply of tokens minted to the deployer’s address and establishes roles for a default administrator and minter:
Set up your development environment
Create and initialize a project
Open up your terminal and navigate to any directory of your choice. Run the following commands to create and initiate a project:
Install Hardhat and the AxelarJS SDK
Install Hardhat and the AxelarJS SDK with the following commands:
Set up project ABIs
Next, set up the ABIs for the Interchain Token Service and the contract from the token you deployed.
Create a folder named utils
. Inside the folder, create the following new files and add the respective ABIs:
- Add the Interchain Token Service ABI to
interchainTokenServiceABI.json
. - Add your custom token ABI to
customTokenABI**.**json
. You can get this from FTMScan or PolygonScan with the address of your deployed token if your contract is verified. Otherwise, you can get it from the same service you deployed theSimpleCustomToken
on.
Set up an RPC for the local chain
Back in the root directory, set up an RPC for the Fantom and Polygon testnet.
Create an .env
file
To make sure you’re not accidentally publishing your private key, create an [.env
file](https://blog.bitsrc.io/a-gentle-introduction-to-env-files-9ad424cc5ff4) to store it in:
Add your private key to .env
and hardhat.config.js
Export your private key and add it to the .env
file you just created:
If you will push this project on GitHub, create a .gitignore
file and
include .env
.
Then, create a file with the name hardhat.config.js
and add the following code snippet:
Deploy token manager on the Fantom testnet
Now that you have set up an RPC for the Fantom and Polygon testnet, you can deploy a token manager on the Fantom testnet.
Create a customInterchainToken.js
script
Create a new file named customInterchainToken.js
and import the required dependencies:
Ethers.js
- The AxelarJS SDK
- The custom token contract ABI
- The address of the
InterchainTokenService
contract - The address of your token deployed on Fantom and Polygon testnet
Get the signer
Next, create a getSigner()
function in customInterchainToken.js
. This will obtain a signer for a secure transaction:
Get the contract instance
Then, create a getContractInstance()
function in customInterchainToken.js
. This will get the contract instance based on the contract’s address, ABI, and signer:
Deploy a token manager on Fantom
Create a deployTokenManager()
function for the Fantom testnet. This will deploy a token manager with your custom token address:
Add a main()
function
Add a main()
function to execute the customInterchainToken.js
script and handle any errors that may arise:
Run the customInterchainToken.js
script to deploy to Fantom
Run the script in your terminal to register and deploy the token, specifying the fantom
testnet:
If you see something similar to the following on your console, you have successfully registered your token as a canonical Interchain Token.
Store the token ID, Expected Token Manager, and salt value
Copy the token ID, Expected Token Manager, and salt value and store them somewhere safe. You will need them to initiate a remote token manager deployment and token transfer in a later step.
Check the transaction on the Fantom testnet scanner
Check the Fantom testnet scanner to see if you have successfully deployed a token manager.
Remotely deploy a token manager on the Polygon testnet
You’ve just successfully deployed a token manager to Fantom, which you are using as your local chain. Now, deploy a token manager remotely to Polygon, which will act as the remote chain in this tutorial. Remember that you can specify any two chains as your local and remote chains.
Estimate gas fees
In customInterchainToken.js
, call estimateGasFee()
from the AxelarJS SDK to estimate the actual cost of deploying your remote Canonical Interchain Token on a remote chain:
Perform remote token manager deployment
Create a deployRemoteTokenManager()
function. This will deploy the remote Token Manager on the Polygon testnet. Make sure to change the salts to the value you saved from a previous step.
Update main()
to deploy to remote chains
Update main()
to execute deployRemoteTokenManager()
:
Run the customInterchainToken.js
script to deploy to Polygon
Run the script in your terminal to deploy a token manager remotely, once again specifying the fantom
testnet (the source chain where all transactions are taking place):
You should see something similar to the following on your console:
Take a look at the token ID and the expected token manager address. These must match the ones obtained in the previous step, as they are linked with the same salt value when deploying a token manager remotely on the Polygon testnet.
When deploying the token manager to a preferred chain other than the local chain (in our example, the Fantom testnet) while the remote chain is Polygon, make sure to deploy the remote token manager from the local chain. If you deploy to the Polygon testnet, you may encounter a TokenManagerDoesNotExist
error. This error occurs because there is no token manager on Polygon with the tokenId
you are using. You don’t need to deploy a remote token manager for Polygon because there is already one available on Fantom. Therefore, we deploy that token manager from the Fantom testnet.
Check the transaction on the Axelar testnet scanner
Check the Axelarscan testnet scanner to see if you have successfully deployed the remote token manager on the Polygon testnet. It should look something like this.
Add gas if needed. Ensure that Axelar shows a successful transaction before continuing to the next step.
Transfer mint access to the Token Manager address on the Fantom and Polygon testnets
You must transfer mint access to the Token Manager address on both chains before you can mint and burn tokens while moving assets between chains.
Transfer mint access to the Token Manager address on the Fantom testnet
Create a transferMintAccessToTokenManagerOnFantom()
function that will perform the mint access transfer on the Fantom testnet.
Update main()
to transfer mint access on Fantom testnet
Update main()
to execute transferMintAccessToTokenManagerOnFantom()
:
Run the customInterchainToken.js
script to deploy to Fantom testnet
Run the script in your terminal to transfer mint access to the Token Manager specifying the fantom
testnet:
You should see something similar to the following on your console:
Check the transaction on the Fantom testnet scanner
Check the Fantom testnet scanner to see if you have successfully transferred mint access to the Token Manager address.
Transfer mint access to the Token Manager address on the Polygon testnet
Create a transferMintAccessToTokenManagerOnPolygon()
function that will perform the mint access transfer on the Fantom testnet.
Update main()
to transfer mint access on Fantom testnet
Update main()
to execute transferMintAccessToTokenManagerOnPolygon()
:
Run the customInterchainToken.js
script to deploy to the Polygon testnet
Run the script in your terminal to transfer mint access to Token Manager, specifying the polygon
testnet:
You should see something similar to the following on your console:
Check the transaction on the Polygon testnet scanner
Check the Polygon testnet scanner to see if you have successfully transferred Mint access to the Token Manager address.
Transfer your token between chains
Now that you have deployed a TokenManager
on both Fantom and Polygon testnet, you can transfer your token between those two chains using the [interchainTransfer()](https://github.com/axelarnetwork/interchain-token-service/blob/9edc4318ac1c17231e65886eea72c0f55469d7e5/contracts/interfaces/IInterchainTokenStandard.sol#L19)
method.
Initiate a remote token transfer
In customInterchainToken.js
, create a transferTokens()
function that will facilitate remote token transfers between chains. Change the token ID to the tokenId
that you [saved from an earlier step](### Store the token ID, Expected Token Manager, and salt value), and change the address in transfer
to your own wallet address:
Update main()
to execute token transfer
Update the main()
to execute transferTokens()
:
Run the customInterchainToken.js
script to transfer tokens
Run the script in your terminal, specifying the fantom
testnet:
You should see something similar to the following on your console:
If you see this, it means that your interchain transfer has been successful! 🎉
Check the transaction on the Axelar testnet scanner
Check the Axelarscan testnet scanner to see if you have successfully transferred SCT from the Fantom testnet to the Polygon testnet. It should look something like this.
Congratulations!
You have now programmatically linked custom tokens deployed on multiple chains as Interchain Token using Axelar’s Interchain Token Service and transfer it between two chains.
Great job making it this far! To show your support to the author of this tutorial, please post about your experience and tag @axelarnetwork on Twitter (X).
What’s next
For further examples utilizing the Interchain Token Service, check out the following in the axelar-examples
repo on GitHub:
its-custom-token
— Demonstrates how to use the ITS with a custom token implementation.its-interchain-token
— Demonstrates how to deploy Interchain Tokens that are connected across EVM chains and how to send some tokens across.its-canonical-token
- Demonstrates how to deploy canonical Interchain Token and send cross-chain transfer for these tokens.its-lock-unlock-fee
Demonstrates how to deploy deploy interchain token with lock/unlock_fee token manager type.its-executable
Demonstrates how to deploy interchain token and send a cross-chain transfer along with a message.its-mint-burn-from
Demonstrates how to deploy interchain token with usesburnFrom()
on token being bridged rather thanburn()
.