Skip to content

🌈🌈🌈 NEAR <> Ethereum Decentralized Bridge

License

Notifications You must be signed in to change notification settings

dndll/rainbow-bridge

Folders and files

NameName
Last commit message
Last commit date

Latest commit

Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 
Β 

Repository files navigation

Rainbow Bridge

Ethereum to Near trustless, fully decentralized, bidirectional bridge

Buildkite Build npm

Pre-requisites

The current version of CLI is all-in-one package -- it is used both for production and testing. As a result, even if you need CLI only for the token transfer you need to install all testing dependencies. This will be changed in the future.

  • Install golang, see.
  • Make sure you are using Node with version <=13. We recommend using nvm for installing node and npm, if you already don't have one. This constraint will be removed soon;
  • You would also need to install resources needed to compile nearcore (in the future this will only be required for the testing CLI):
curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh
source $HOME/.cargo/env
rustup default stable
rustup target add wasm32-unknown-unknown
  • Then install dependencies needed for the compilation of nearcore, see.

Usage

You can either install rainbow-bridge-cli from npm or clone it and use locally.

npm i -g rainbow-bridge-cli

To learn the commands that you can use with the rainbow bridge run

node index.js --help

Parameters of each command can be specified through environment variables, command line arguments, entries in the ~/.rainbow/config.json config file, or the default value will be used -- in that priority. If argument is not provided and there is no default value the program will not execute.

If script successfully executes a command then each parameter provided through the command line argument will be written into the config file. Additionally, if scripts generates new parameters (e.g. it deploys a contract to Ethereum and obtains its address) will also be written into the config file. Arguments should not be specified multiple times.

Note, you can use environment variables to pass sensitive data which will not lead to it being written into the config file.

Security

Bridge is as long as majority (1/2) of Etherem mining power is honest and supermajority (2/3) of NEAR stake is honest. There are no additional security requirements, except that Ethereum should be able to accept 1 transaction within 4 hour period even in the worst congestion scenario.

Gas costs

NEAR fees are negligible, both for bridge maintenance and for token transfer. Ethereum fees are the following:

  • To transfer ERC20 token from ETH to NEAR: Approx 43,989 gas to set allowance and approx 37,407 gas to lock it;
  • To transfer ERC20 token back from NEAR to ETH: Approx 240,531 gas to unlock the token;
  • To submit a NEAR block header: approx 697,140 gas;
  • To challenge a NEAR block header: approx 700k gas.

As of 2020-07-14 (gas price is 40 gwei) the cost of running bridge on NEAR mainnnet and Ethereum mainnet is approx 42 USD/day. The cost of ETH->NEAR transfer of ERC20 token is 1 USD. The cost of NEAR->ETH transfer of ERC20 token is 2 USD.

Using existing Testnet bridge

We are currently running bridge between NEAR Testnet and Ethereum Ropsten. The latency on it sped up to allow rapid experimentation. Specifically, ETH->NEAR direction is 10 blocks which corresponds to 2-3 minutes latency, and NEAR->ETH direction is ~80 seconds. The version of the bridge that will connect NEAR Mainnet to Ethereum Mainnet will use 25 blocks for ETH->NEAR direction and 4 hours for NEAR->ETH direction.

To try cross-blockchain transfer of an existing ERC20 token you would need:

  • Account on Ropsten that owns certain amount of some ERC20 token, in this example we will be using 0x88f975D5A1153Ea92AF66e7c4292576a329c04B6. Ask [email protected] to transfer you some (if you want to transfer your own token you would have to setup your own locker and minting contracts);
  • Account on https://wallet.testnet.near.org/;

Let's set them up:

  1. Install metamask and switch to Ropsten. Then create an account;
  2. Go to https://faucet.ropsten.be/ and send some tokens to this account;
  3. Add 0x88f975D5A1153Ea92AF66e7c4292576a329c04B6 as a custom token in Metamask (use 0 as precision);
  4. Ask [email protected] to deposit you some ERC20 tokens. After the deposit you should be able to observe non-zero balance in Metamask;
  5. Export private key from this account;
  6. Go to https://wallet.testnet.near.org/ and create an account;
  7. Install https://github.com/near/near-shell and run near login. In the opened browser tab/window allow new access key;
  8. Record secret key that you have just added from ~/.near-credentials/default/<account name>.json, include ed25519: prefix;
  9. If you don't have access to an Ethereum node, go to http://infura.io/ and create a free account. Locate and record websocket URL to ropsten: wss://ropsten.infura.io/ws/v3/<project_id>;

Prepare CLI:

git clone https://github.com/near/rainbow-bridge/
cd rainbow-bridge/environment
yarn
node index.js clean
node index.js prepare

Then edit ~/.rainbow/config.json to be:

{
        "nearNetworkId": "testnet",
        "nearNodeUrl": "https://rpc.testnet.internal.near.org/",
        "ethNodeUrl": "wss://ropsten.infura.io/ws/v3/<project_id>",
        "nearMasterAccount": "<your_near_account_id>",
        "nearMasterSk": "<your_near_sk>",
        "nearClientAccount": "eth_on_near_client_10",
        "nearProverAccount": "eth_on_near_prover_10",
        "ethMasterSk": "<your_eth_private_key>",
        "ethEd25519Address": "0x40b2d1334B7Fbbe2D4E1eb0Df689Af3D2a3903D2",
        "ethClientAddress": "0x276D4d74Dc14251c8D75Ff4aE9175142E1C2254d",
        "ethProverAddress": "0x69e75769a1D228f1c660869FB69455190fe9a80b",
        "ethErc20Address": "0x88f975D5A1153Ea92AF66e7c4292576a329c04B6",
        "nearFunTokenAccount": "funtoken10",
        "ethLockerAddress": "0xAfa909a33241d0271B7fd73b57C34439e9fBC84a",
}

Now let's run the transfer from ETH to NEAR:

node index.js transfer-eth-erc20-to-near --amount 100 --eth-sender-sk <you_eth_private_key> --near-receiver-account <your_near_account_id> --near-master-account <your_near_account_id> --near-master-sk <your_near_sk>

Wait until the command finishes. You should see something like:

P3oQ5ohqrayePpKy26RXdXPgUAD8DByeC5jhYA6kLJxTmXPd6gVHuP
Approving token transfer.
Approved token transfer.
Transferring tokens from the ERC20 account to the token locker account.
Success.
Transferring 100 tokens from 0x88f975D5A1153Ea92AF66e7c4292576a329c04B6 ERC20. From 0xe3628e6AB18A6B0F60Ed8540690d18b6d9C88a46 sender to flow10.testnet recipient
Eth2NearClient is currently at block 8292239. Waiting for block 8292241 to be confirmed. Sleeping for 10 sec.
Eth2NearClient is currently at block 8292241. Waiting for block 8292241 to be confirmed. Sleeping for 10 sec.
Eth2NearClient is currently at block 8292242. Waiting for block 8292241 to be confirmed. Sleeping for 10 sec.
Eth2NearClient is currently at block 8292243. Waiting for block 8292241 to be confirmed. Sleeping for 10 sec.
Eth2NearClient is currently at block 8292244. Waiting for block 8292241 to be confirmed. Sleeping for 10 sec.
Eth2NearClient is currently at block 8292246. Waiting for block 8292241 to be confirmed. Sleeping for 10 sec.
Eth2NearClient is currently at block 8292247. Waiting for block 8292241 to be confirmed. Sleeping for 10 sec.
Eth2NearClient is currently at block 8292248. Waiting for block 8292241 to be confirmed. Sleeping for 10 sec.
Eth2NearClient is currently at block 8292249. Waiting for block 8292241 to be confirmed. Sleeping for 10 sec.
Balance of flow10.testnet before the transfer is 0
TxHash Hm6eRyMCoLKRAkg7DMQuN4e25kLU4GUPLVFjD6PaWXPi
[ 'token: 88f975d5a1153ea92af66e7c4292576a329c04b6; sender: e3628e6ab18a6b0f60ed8540690d18b6d9c88a46; amount: 100; recipient: flow10.testnet',
  'Refunding 6300000000000000000000 tokens for storage' ]
Transferred
Balance of flow10.testnet after the transfer is 100

Currently, NEAR wallet does not display balances of the fungible token, but fortunately the CLI tool queries the balance before and after the transfer.

Now let's run the transfer from NEAR to ETH:

node index.js transfer-eth-erc20-from-near --amount 1 --near-sender-account <your_near_account_id> --near-sender-sk <your_near_sk> --eth-receiver-address <your_eth_address> --eth-master-sk <your_eth_private_key>

Wait until the command finishes. You should observe something like:

Balance of flow10.testnet before burning: 100
Burning 1 tokens on NEAR blockchain in favor of e3628e6AB18A6B0F60Ed8540690d18b6d9C88a46.
Current light client head is: hash=F2fkvdda1bwZiM5oMW1e5RLa5nLEGbrNQbFKJyCr63V6, height=9593826
Block 9593969 is not available on the light client yet. Current height of light client is 9593826. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=GEqJfWRnNyvC6T7yJtzLrNxKqohVWX34La4xWZMeuHXy, height=9593959
Block 9593969 is not available on the light client yet. Current height of light client is 9593959. Sleeping 10 seconds.
Current light client head is: hash=72TJ5LrvpNyagTWPvEEiGHG84fyAWZCGwUjSXEe8mgLV, height=9594083
Near2EthClient block is at 9594083 which is further than the needed block 9593969
Burnt "1"
Balance of flow10.testnet after burning: 99
ERC20 balance of 0xe3628e6AB18A6B0F60Ed8540690d18b6d9C88a46 before the transfer: 99800
ERC20 balance of 0xe3628e6AB18A6B0F60Ed8540690d18b6d9C88a46 after the transfer: 99801

Congrats, you have completed a roundtrip of ERC20 tokens from ETH to NEAR and back to ETH!

Local test run

To locally test the bridge run:

node index.js clean
node index.js prepare
node index.js start near-node
node index.js start ganache

Initializing the contracts

First let's initialize the contracts that bridge needs to function:

node index.js init-near-contracts
node index.js init-eth-ed25519
node index.js init-eth-client --eth-client-lock-eth-amount 1000 --eth-client-lock-duration 10
node index.js init-eth-prover

Now, let's set up token on Ethereum blockchain that we can transfer to NEAR blockchain (this can be your own token).

node index.js init-eth-erc20
node index.js init-eth-locker

Now, let's initialize token on NEAR blockchain that will mirror the token on Ethereum side.

node index.js init-near-fun-token

Starting the services

Now start the services that will relay the information between the chains:

node index.js start eth-relay
node index.js start near-relay --eth-master-sk 0x2bdd21761a483f71054e14f5b827213567971c676928d9a1808cbfa4b7501201
node index.js start near-watchdog --eth-master-sk 0x2bdd21761a483f71054e14f5b827213567971c676928d9a1808cbfa4b7501202

Note, you can observe the logs of the relays by running:

pm2 logs

Transferring tokens

Finally, let's transfer some tokens

node index.js transfer-eth-erc20-to-near --amount 1000 --eth-sender-sk 0x2bdd21761a483f71054e14f5b827213567971c676928d9a1808cbfa4b7501200 --near-receiver-account rainbow_bridge_eth_on_near_prover --near-master-account nearfuntoken

Note, when we deployed ERC20 to the Ethereum blockchain we have minted a large number of tokens to the default master key of Ganache, so we have transferred ERC20 tokens from it to alice.test.near. Notice that we are using nearfuntoken account here to pay for the NEAR gas fees, any account for which we know a secret key would've worked too. You must observe blocks being submitted.

Now let's try to transfer one token back to Ethereum

node index.js transfer-eth-erc20-from-near --amount 1 --near-sender-account rainbow_bridge_eth_on_near_prover --near-sender-sk ed25519:3D4YudUQRE39Lc4JHghuB5WM8kbgDDa34mnrEP5DdTApVH81af7e2dWgNPEaiQfdJnZq1CNPp5im4Rg5b733oiMP --eth-receiver-address 0xEC8bE1A5630364292E56D01129E8ee8A9578d7D8

You should observe the change of the ERC20 balance as reported by the CLI.

Docker:

Currently we have the following docker options:

  1. Rainbow Docker image containing rainbow ready for running
    • run the rainbow docker image with a custom command
  2. A development docker compose setup (docker-compose-dev.yml)
    • ganache
    • local near node
    • eth-relay
  3. A production docker compose setup (docker-compose-prod.yml)
    • eth-relay

Running the docker setup:

  1. One options is to adapt the current config.json specified in the root folder of the project and build a new image.
  2. Specifying the configuration flags through environment variables.

We recommend a usage of both, encouraging using the config.json for common configurations, while passing the secrets through environment variables.

Examples:

# Creating a docker image
docker build .

# Running the development env with config setup
docker-compose up

# Running the development env with ENV overrides 
docker-compose -f docker-compose-dev.yml up -e MASTER_SK=<key> -e ...

# Running the production env just use:
docker-compose -f docker-compose-prod.yml instead

About

🌈🌈🌈 NEAR <> Ethereum Decentralized Bridge

Resources

License

Code of conduct

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Rust 42.6%
  • JavaScript 22.8%
  • Go 20.6%
  • Solidity 11.6%
  • Shell 2.1%
  • Nix 0.1%
  • Other 0.2%