Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat/obridge docs #322

Open
wants to merge 14 commits into
base: main
Choose a base branch
from
Empty file added specs/README.md
Empty file.
Empty file added specs/obridge/SUMMARY.md
Empty file.
49 changes: 49 additions & 0 deletions specs/obridge/background.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
## Introduction

**Obridge** is a cross-chain solution developed by **Oraichain**. It is designed to create a trustless bridge that enables the seamless transfer of assets between Oraichain and various blockchain ecosystems. Obridge supports EVM-compatible chains such as **Tron, BNB Chain, Ethereum**, as well as **XRPL, Bitcoin, Ton**, and the **Cosmos Ecosystem**. This innovative bridge ensures secure and efficient cross-chain communication, providing a decentralized way to move assets across different networks without relying on trusted intermediaries.

![Obridge Connecting Oraichain to Various Blockchains](./image/obridge_universal.png)

### How Obridge Works

1. [Gravity Bridge](https://github.com/oraichain/Gravity-Bridge): OBridge is built on top of the Gravity Bridge, utilizing its robust framework to enable secure and efficient asset transfers between EVM-compatible networks like Ethereum, BNB Chain, and TRON, and the Cosmos ecosystem.

2. [Inter-Blockchain Communication (IBC)](https://github.com/oraichain/ibc-bridge-wasm): OBridge integrates the IBC protocol to ensure seamless asset transfers within the Cosmos ecosystem. This integration maintains security and efficiency without the need for trusted intermediaries, allowing for a more decentralized and trustless environment.

3. [Ton IBC](https://github.com/oraichain/tonbridge-cw-contracts): OBridge establishes an IBC bridge between Ton and Oraichain, leveraging IBC technology to facilitate secure and trustless cross-chain communication and asset transfers.

4. [Cw-Bitcoin](https://github.com/oraichain/cw-bitcoin): The Bitcoin trustless bridge within OBridge is implemented using CosmWasm contracts. This bridge facilitates the use of Taproot Bitcoin scripts, enabling secure and decentralized Bitcoin transfers within the Oraichain ecosystem.

5. [XRPL Bridge](https://github.com/oraichain/cw-xrpl-bridge): OBridge employs a multi-signing account on the XRPL for its bridge. This account holds tokens issued on the XRPL and manages them according to the specific workflow. Depending on the scenario, it either uses the received token balance to send tokens to XRPL accounts (if the account is not an issuer) or mints and sends tokens to XRPL accounts. Multi-signing and public keys associated with each relayer from the contract are used for transaction signing, ensuring security and trustlessness.

### Supported Chains

We currently support the following blockchain platforms:

- **Cosmos-Based Chains:**

- Cosmoshub
- Oraichain
- Osmosis
- Injective
- Noble

- **EVM-Based Chains:**

- Ethereum
- BNB Chain
- Tron

- **XRP Ledger**

- **TON - Telegram Open Network**

- **Bitcoin**

### Integration

- [Orai - EVM](./orai-evm.md)
- [Orai - Cosmos](./orai-cosmos.md)
- [Orai - Bitcoin]()
- [Orai - TON](./orai-ton.md)
- [Orai - XRPL]()
Binary file added specs/obridge/image/evm_to_orai.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added specs/obridge/image/obridge_universal.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added specs/obridge/image/orai_to_evm.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added specs/obridge/image/orai_to_ton.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added specs/obridge/image/ton_to_orai.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Empty file added specs/obridge/orai-cosmos.md
Empty file.
175 changes: 175 additions & 0 deletions specs/obridge/orai-evm.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,175 @@
# Oraichain <=> EVM

## Overview

The Oraichain-EVM bridge leverages GravityBridge and IBC Cw20-Ics 20 to enable seamless asset transfers between Oraichain and EVM chains.

| Name | Chain | Address |
| --------------------- | --------- | --------------------------------------------------------------- |
| Ibc Bridge Wasm | Oraichain | orai195269awwnt5m6c843q6w7hp8rt0k7syfu9de4h0wz384slshuzps8y7ccm |
| OBridge Contract ETH | Ethereum | 0x09Beeedf51AA45718F46837C94712d89B157a9D3 |
| OBridge Contract BSC | BNB Chain | 0xb40C364e70bbD98E8aaab707A41a52A2eAF5733f |
| OBridge contract Tron | Tron | 0x73Ddc880916021EFC4754Cb42B53db6EAB1f9D64 |

## Workflows

### Send from EVM to Oraichain

![EVM to Oraichain](./image/evm_to_orai.png)

### Send from Oraichain to EVM

![Oraichain to EVM](./image/orai_to_evm.png)

## Integration

### EVM to Oraichain

```solidity
interface IGravity {
event SendToCosmosEvent(address indexed _tokenContract, address indexe _sender, string _destination, uint256 _amount, uint256 _eventNonce);

function sendToCosmos(address _tokenContract, string calldata _destination, uint256 _amount) external;
}

```
Comment on lines +28 to +35
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fix typo in event declaration.

There is a typo in the event declaration. Replace indexe with indexed.

- event SendToCosmosEvent(address indexed _tokenContract, address indexe _sender, string _destination, uint256 _amount, uint256 _eventNonce);
+ event SendToCosmosEvent(address indexed _tokenContract, address indexed _sender, string _destination, uint256 _amount, uint256 _eventNonce);
Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
```solidity
interface IGravity {
event SendToCosmosEvent(address indexed _tokenContract, address indexe _sender, string _destination, uint256 _amount, uint256 _eventNonce);
function sendToCosmos(address _tokenContract, string calldata _destination, uint256 _amount) external;
}
```
```solidity
interface IGravity {
event SendToCosmosEvent(address indexed _tokenContract, address indexed _sender, string _destination, uint256 _amount, uint256 _eventNonce);
function sendToCosmos(address _tokenContract, string calldata _destination, uint256 _amount) external;
}
```


To bridge tokens from EVM to Oraichain using OBridge, follow these steps:

- Increase allowance for OBridge contract address: Ensure the Gravity contract has the necessary allowance to transfer the tokens
- Call `sendToCosmos` function in OBridge Contract ETH(BSC, TRON):

**Parameters**:

- `_tokenContract` : The address of the token want to bridge.
- `_destination`: A string combining `${sourceChannel}/${destReceiver}:${memo}`, where **sourceChannel** is the bridge channel between OraiBridge and Oraichain, **destReceiver** is the recipient's address on Oraichain, and [memo](./universal-swap-memo.md) used for UniversalSwap
- \_amount: The amount of tokens to bridge.
Comment on lines +44 to +46
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fix punctuation and grammar issues.

There are some punctuation and grammar issues in the parameters description.

- `_tokenContract` : The address of the token want to bridge.
- `_destination`: A string combining `${sourceChannel}/${destReceiver}:${memo}`, where **sourceChannel** is the bridge channel between OraiBridge and Oraichain, **destReceiver** is the recipient's address on Oraichain, and [memo](./universal-swap-memo.md) used for UniversalSwap
- \_amount: The amount of tokens to bridge.
+ `_tokenContract`: The address of the token to bridge.
+ `_destination`: A string combining `${sourceChannel}/${destReceiver}:${memo}`, where **sourceChannel** is the bridge channel between OraiBridge and Oraichain, **destReceiver** is the recipient's address on Oraichain, and [memo](./universal-swap-memo.md) is used for UniversalSwap.
+ `_amount`: The number of tokens to bridge.
Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
- `_tokenContract` : The address of the token want to bridge.
- `_destination`: A string combining `${sourceChannel}/${destReceiver}:${memo}`, where **sourceChannel** is the bridge channel between OraiBridge and Oraichain, **destReceiver** is the recipient's address on Oraichain, and [memo](./universal-swap-memo.md) used for UniversalSwap
- \_amount: The amount of tokens to bridge.
`_tokenContract`: The address of the token to bridge.
`_destination`: A string combining `${sourceChannel}/${destReceiver}:${memo}`, where **sourceChannel** is the bridge channel between OraiBridge and Oraichain, **destReceiver** is the recipient's address on Oraichain, and [memo](./universal-swap-memo.md) is used for UniversalSwap.
`_amount`: The number of tokens to bridge.
Tools
LanguageTool

[uncategorized] ~44-~44: Loose punctuation mark.
Context: ...): Parameters: - _tokenContract : The address of the token want to bridge...

(UNLIKELY_OPENING_PUNCTUATION)


[uncategorized] ~45-~45: Loose punctuation mark.
Context: ...e token want to bridge. - _destination: A string combining `${sourceChannel}/${...

(UNLIKELY_OPENING_PUNCTUATION)


[uncategorized] ~46-~46: ‘Amount of’ should usually only be used with uncountable or mass nouns. Consider using “number” if this is not the case.
Context: ... used for UniversalSwap - _amount: The amount of tokens to bridge. Example: Brid...

(AMOUNTOF_TO_NUMBEROF)


**Example**: Bridge Orai from the BNB Chain to Oraichain, swaps it for USDC, and then bridges the USDC to Ethereum.

- Init EVM wallet

```ts
import { EvmWallet } from "@oraichain/oraidex-common";
import { Web3Provider } from "@ethersproject/providers";
import { ethers } from "ethers";

window.tronWebDapp = walletType === "owallet" ? window.tronWeb_owallet : window.tronWeb;
window.tronLinkDapp = walletType === "owallet" ? window.tronLink_owallet : window.tronLink;
window.ethereumDapp = walletType === "owallet" ? window.eth_owallet : window.ethereum;

export default class Evm extends EvmWallet {
private provider: Web3Provider;

public checkEthereum() {
if (window.ethereumDapp) return true;
return false;
}

public getSigner() {
if (!this.provider) this.provider = new ethers.providers.Web3Provider(window.ethereumDapp, "any");
return this.provider.getSigner();
}

public isWindowEthereum() {
return !!window.ethereumDapp;
}

public isTron(chainId: string | number) {
return Number(chainId) == Networks.tron;
}

public checkTron() {
if (window.tronWebDapp && window.tronLinkDapp) return true;
return false;
}

public async switchNetwork(chainId: string | number) {
if (this.checkEthereum()) {
await window.ethereumDapp.request!({
method: "wallet_switchEthereumChain",
params: [{ chainId: "0x" + Number(chainId).toString(16) }]
});
}
}

public async getEthAddress() {
if (this.checkEthereum()) {
const [address] = await window.ethereumDapp.request({
method: "eth_requestAccounts",
params: []
});
return address;
}
}
}
```

- Bridge:

```ts
import { Bridge__factory } from "@oraichain/oraidex-common";

const gravityContractAddr = "0xb40C364e70bbD98E8aaab707A41a52A2eAF5733f";
const oraiBscToken = "0xa325ad6d9c92b55a3fc5ad7e412b1518f96441c0";
const destination =
"channel-1/orai1hvr9d72r5um9lvt0rpkd4r75vrsqtw6yujhqs2:CqMBCgdvcmFpZGV4GpcBCpQBCgczMDAwMDAwEogBCj9vcmFpMTl0dGcwajd3NWtyODNqczMydG13bnd4eGRxOXJrbXc0bTNkN21uMmoyaGtwdWd3d2E0dHN6d3Nua2cSBG9yYWkaP29yYWkxNXVuOG1zeDNuNXpmOWFobHhtZmVxZDJrd2E1d20wbnJweGVyMzA0bTluZDVxNnFxMGc2c2t1NXBkZBIIMTA2NTg4ODUYgN7v8N7wgvUXIokBEoYBCgpjaGFubmVsLTI5EjVldGgtbWFpbm5ldDB4OGM3RTBBODQxMjY5YTAxYzBBYjM4OUNlOEZiM0NmMTUwQTk0RTc5Nxo1ZXRoLW1haW5uZXQweEEwYjg2OTkxYzYyMThiMzZjMWQxOUQ0YTJlOUViMGNFMzYwNmVCNDgggISi7d3wgvUXKgAqK29yYWkxaHZyOWQ3MnI1dW05bHZ0MHJwa2Q0cjc1dnJzcXR3Nnl1amhxczI=";
const amount = "2024699112000000000000";
const from = "0x655169CAc525eC903cd11F8690211C134aeD1C0F";

const gravityContract = Bridge__factory.connect(gravityContractAddr, evmWallet.getSigner());
const result = await gravityContract.sendToCosmos(oraiBscToken, destination, amountVal, { from });
```

### Oraichain to EVM

`Bridging Tokens Using IbcBridgeWasm Contract`

**1. Transfer Native Token**
Comment on lines +126 to +128
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fix emphasis used instead of a heading.

The emphasis should be replaced with a heading.

- **1. Transfer Native Token**
+ ### 1. Transfer Native Token
Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
`Bridging Tokens Using IbcBridgeWasm Contract`
**1. Transfer Native Token**
`Bridging Tokens Using IbcBridgeWasm Contract`
### 1. Transfer Native Token
Tools
Markdownlint

128-128: null
Emphasis used instead of a heading

(MD036, no-emphasis-as-heading)


To bridge native tokens from Oraichain using the IbcBridgeWasm contract, you need to call the `TransferToRemote` message. Here’s how to do it:

- Prepare the `TransferToRemote` message:

- **local_channel_id**: The IBC channel on Oraichain for sending tokens.
- **remote_address**: The bech32 address of the recipient on the remote chain (Obridge).
- **remote_denom**: The denomination of the token on the remote chain, identified from the mapping pair (e.g., `oraib0x...` or `eth-mainnet0x...`).
- **timeout**: (Optional) The duration in seconds for how long the packet should live. If not specified, it defaults to `default_timeout`.
- **memo**: The destination address, which must contain the prefix of the destination chain (e.g., `eth-mainnet0x...` or `oraib0x...`).

- **Call the IbcBridgeWasm Contract**:
- Ensure the corresponding funds match the `remote_denom` in the mapping pair.

**2. Transfer Cw20**
To bridge CW20 tokens from Oraichain to a chain use the CW20 Send function along with the TransferToRemote message
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fix missing comma.

There is a missing comma in the sentence.

- To bridge CW20 tokens from Oraichain to a chain use the CW20 Send function along with the TransferToRemote message
+ To bridge CW20 tokens from Oraichain to a chain, use the CW20 Send function along with the TransferToRemote message
Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
To bridge CW20 tokens from Oraichain to a chain use the CW20 Send function along with the TransferToRemote message
To bridge CW20 tokens from Oraichain to a chain, use the CW20 Send function along with the TransferToRemote message
Tools
LanguageTool

[uncategorized] ~144-~144: A comma might be missing here.
Context: ... bridge CW20 tokens from Oraichain to a chain use the CW20 Send function along with t...

(AI_EN_LECTOR_MISSING_PUNCTUATION_COMMA)


**Example**
Transfer Orai from Oraichain to BNB chain

```ts
import { GasPrice, SigningStargateClient } from "@cosmjs/stargate";
import { DirectSecp256k1HdWallet } from "@cosmjs/proto-signing";
import dotenv from "dotenv";
import { ORAI } from "@oraichain/common";
import { Cw20Ics20Client } from "@oraichain/common-contracts-sdk";
import { TransferBackMsg } from "@oraichain/common-contracts-sdk/build/CwIcs20Latest.types";
import { CwIcs20LatestClient } from "@oraichain/common-contracts-sdk";

const signer = await DirectSecp256k1HdWallet.fromMnemonic(process.env.EXAMPLES_MNEMONIC, { prefix: ORAI }); // replace your mnemonic here
const accounts = await signer.getAccounts();
const address = accounts[0].address;
const client = await SigningStargateClient.connectWithSigner("https://rpc.orai.io", signer, {
gasPrice: GasPrice.fromString("0.001orai")
});

const ics20Contract = "orai195269awwnt5m6c843q6w7hp8rt0k7syfu9de4h0wz384slshuzps8y7ccm";
const cwIcs20Client = new CwIcs20LatestClient(client, sender, ics20Contract);

const result = await cwIcs20Client.transferToRemote({
local_channel_id: "channel-29",
remote_address: "oraib1hvr9d72r5um9lvt0rpkd4r75vrsqtw6ytnnvpf",
remote_denom: "oraib0xA325Ad6D9c92B55A3Fc5aD7e412B1518F96441C0",
timeout: "1724320979000000000",
memo: "oraib0x8ae1874f2f9f26eeecf1855adec19f530e3cdeaa"
});
```
Loading
Loading