The interBTC TypeScript library connects the Polkadot and Kusama ecosystems with Bitcoin. It allows the creation of interBTC on Polkadot and kBTC on Kusama, fungible "wrapped" tokens that represent Bitcoin. Wrapped tokens are backed by Bitcoin 1:1 and allow redeeming of the equivalent amount of Bitcoins by relying on a collateralized third-party (Vaults). In comparison to other bridge constructions (like tBTC, wBTC, or RenVM) anyone can become an intermediary by depositing collateral making interBTC the only truly open system.
The bridge itself follows the detailed specification: Explore the specification »
It is implemented as a collection of open-source Substrate modules using Rust: Explore the implementation »
You can visit bridge.interlay.io to see the library in action.
The library assumes you have a version of the Interlay or Kintsugi networks running locally or remotely.
To use the library, you will first need to create a PolkadotJS APIPromise
instance,
and then instantiate a InterBtcApi
instance.
import { createInterBtcApi } from "@interlay/interbtc";
// If you are using a local development environment
// const PARACHAIN_ENDPOINT = "ws://127.0.0.1:9944";
// if you want to use the Interlay-hosted beta network
const PARACHAIN_ENDPOINT = "wss://api.interlay.io/parachain";
const bitcoinNetwork = "mainnet";
const interBTC = await createInterBtcApi(PARACHAIN_ENDPOINT, bitcoinNetwork);
To emit transactions, an account
has to be set.
The account should conform to the AddressOrPair
interface.
If the account is not of the KeyringPair
type, then a signer must also
be provided (such as an injected extension signer, from the Polkadot wallet).
See more details here: https://polkadot.js.org/docs/extension/usage/
import { createTestKeyring } from "@polkadot/keyring/testing";
const keyring = createTestKeyring();
const keypair = keyring.getPairs()[0];
interBTC.setAccount(keypair);
The different functionalities are then exposed through the InterBtcApi
instance.
From the account you set, you can then start requesting to issue interBTC.
import { BitcoinAmount } from "@interlay/monetary-js";
// amount of BTC to convert to interBTC
// NOTE: the bridge fees will be deducted from this. For example, if you request 1 BTC, you will receive about 0.995 interBTC
const amount = BitcoinAmount.from.BTC(0.001);
// request to issue interBTC
const requestResults = await interBTC.issue.request(amount);
// the request results includes the BTC address(es) and the BTC that should be sent to the Vault(s)
// NOTE: the library will automatically distribute issue requests across multiple Vaults if no single Vault can fulfill the request.
// Most of the time, a single Vault will be able to fulfill the request.
At this point, you will need to send BTC using your favorite BTC wallet.
import { BitcoinAmount } from "@interlay/monetary-js";
// the amount wrapped tokens to redeem
// NOTE: the bridge fees will be deducted from this
const amount = BitcoinAmount.from.BTC(0.001);
// your BTC address
const btcAddress = "tb123....";
// the request results includes the BTC address(es) and the BTC that should be sent to the Vault(s)
// NOTE: the library will automatically distribute redeem requests across multiple Vaults if no single Vault can fulfill the request.
// Most of the time, a single Vault will be able to fulfill the request.
const requestResults = await interBTC.redeem.request(amount, btcAddress);
At this point, one more more Vaults will send BTC to the address specified within 24 hours.
Certain API calls require a parameter of type AccountId
, which represents the Polkadot/Kusama account and can be instantiated as follows
import { AccountId } from "@polkadot/types/interfaces/runtime";
const activeStakedRelayerId = await interBTC.api.createType(
"AccountId",
"5Ck5SLSHYac6WFt5UZRSsdJjwmpSZq85fd5TRNAdZQVzEAPT"
);
const feesEarnedByActiveStakedRelayer = await interBTC.stakedRelayer.getFeesEarned(
activeStakedRelayerId
);
There are many examples in the integration tests, showing how to use this library. Take a look at them here: https://github.com/interlay/interbtc-api/tree/master/test/integration
Follow Github's instructions on how to fork a repository to create your own fork.
[email protected]:<your_github_profile>/interbtc-api.git
cd interbtc-api
Start by setting your Node version via nvm
nvm use
(If necessary, nvm
will guide you on how to install the version.)
Next, install the dependencies
yarn install
Finally, build the library
yarn build
To run unit tests only, use
yarn test:unit
Note that the parachain needs to be running for all tests to run.
yarn docker-parachain-start
The default parachain started is an image of Kintsugi (KINT). To choose a specific chain's image, use the optional --chain
parameter.
e.g. to start Kintsugi explicitly
yarn docker-parachain-start --chain=KINT
Or, to start Interlay locally use
yarn docker-parachain-start --chain=INTR
yarn docker-parachain-start
saves docker compose files locally in local-setup
directory to avoid rebuilding all data just to restart the development environment. In most cases, you don't need to rebuild those unless there have been changes to the related docker images in our fork of the parachain-launch project.
In order to force a fresh rebuild of all configuration files used for the local test environment, delete the existing local-setup
folder and run yarn docker-parachain-start
. This will automatically regenerate all configuration files and start the parachain.
rm -r local-setup && yarn docker-parachain-start
Then, to run all tests:
yarn test
yarn test:integration
Note: While the parachain is starting up, there will be warnings from the integration tests until it can locate the locally running test vaults. Expect the startup to take around 2-3 minutes (after yarn docker-parachain-start
), and only start the integration tests after that time frame.
Another option is to switch to the local-setup
directory and there check for the vaults to start - for example, to see the logs for vault 1, use this:
# in <project_folder>/local-setup
docker-compose logs -f vault_1
(Note: The optional -f
flag attaches the terminal to the log output, you will need to Ctrl + C to exit. Alternatively, omit the flag to just get the current latest log entries.)
To stop the locally running parachain, run:
yarn docker-parachain-stop
Note: This will remove the volumes attached to the images. So your chain will start next time in a clean/wiped state.
We only need to update types when we have changed to newer docker images for the parachain / clients.
Run the parachain (via yarn docker-parachain-start
as shown above, or indeed any Substrate node) and download the metadata:
curl -H "Content-Type: application/json" -d '{"id":"1", "jsonrpc":"2.0", "method": "state_getMetadata", "params":[]}' http://localhost:9933 > src/json/parachain.json
(Remember to remove local-setup
if you want to generate types for a new version of the parachain.)
Then, update the metadata by building the library:
yarn build
This library can be used as a script for initializing a local interBTC setup (the services ran using docker-compose), to allow for manual testing of the UI.
yarn install
yarn initialize
By default, every flag is enabled. To get more information about the flags and disable some of them, run
yarn initialize --help
Regtest is a local Bitcoin instance that allows you to practically anything including sending transactions, mining blocks, and generating new addresses. For a full overview, head over to the Bitcoin developer documentation.
Sending Transactions
For the issue process, you need to send a transaction. On regtest this can be achieved with:
bitcoin-cli -regtest -rpcwallet=Alice sendtoaddress VAULT_ADDRESS AMOUNT
Mining Blocks
In regtest, blocks are not automatically produced. After you sent a transaction, you need to mine e.g., 1 block:
bitcoin-cli -regtest generatetoaddress 1 $(bitcoin-cli -regtest getnewaddress)
Getting Balances
You can query the balance of your wallet like so:
bitcoin-cli -regtest -rpcwallet=Alice getbalance
You can hard-reset the docker dependency setup with the following commands:
docker kill $(docker ps -q)
docker rm $(docker ps -a -q)
docker rmi $(docker images -q)
docker volume rm $(docker volume ls -q)
Contributions are what make the open source community such an amazing place to be learn, inspire, and create. Any contributions you make are greatly appreciated.
- Set up git so you can sign your commits (Alternative link: GitHub: Signing commits)
Unsigned PRs cannot be merged, so do not skip this step. - Fork the Project
- Create your Feature Branch (git checkout -b yourname/AmazingFeature)
- Commit your Changes (git commit -m 'Add some AmazingFeature')
- Push to the Branch (git push origin yourname/AmazingFeature)
- Open a Pull Request with a description of feature you are adding
If you are searching for a place to start or would like to discuss features, reach out to us:
(C) Copyright 2022 Interlay Ltd
interbtc-js is licensed under the terms of the Apache License (Version 2.0). See LICENSE.
Website: Interlay.io
Twitter: @interlayHQ
Email: [email protected]
We would like to thank the following teams for their continuous support: