1 簡(jiǎn)介
bsc-relayer 是一個(gè)獨(dú)立進(jìn)程隙券,可單獨(dú)運(yùn)行男应,運(yùn)行時(shí)首先會(huì)向 BSC 查詢是否注冊(cè)過(guò),如果沒(méi)有是尔,則會(huì)自動(dòng)在 BSC 上注冊(cè),需要 deposit 100 BNB(目前合計(jì)約 22 大不溜多軟妹幣......)开仰,注冊(cè)成功后拟枚,該 bsc-relayer 的信息被記錄在 BSC 上名叫 RelayerHub.sol
的 solidity 系統(tǒng)合約中薪铜,之后就可以執(zhí)行中繼的服務(wù)了。
bsc-relayer 主要有兩個(gè)功能:
- 拉取 BC 的塊頭恩溅,并同步給 BSC
- 拉取 BC 的跨鏈數(shù)據(jù)包隔箍,并同步跨鏈數(shù)據(jù)包給 BSC
BSC 上存在 solidity 兩個(gè)系統(tǒng)合約(除了這兩個(gè)還有其它的系統(tǒng)合約),分別為 TendermintLightClient.sol
和 CrossChain.sol
脚乡,bsc-relayer “同步塊頭” 的操作會(huì)調(diào)用 BSC 的 TendermintLightClient.sol
蜒滩,“同步跨鏈數(shù)據(jù)包” 的操作會(huì)調(diào)用 BSC 的 CrossChain.sol
。
除此之外奶稠,bsc-relayer 還有部分其它的功能俯艰,例如本地查詢狀態(tài),tx 追蹤等锌订,這邊暫時(shí)不進(jìn)行詳細(xì)介紹了竹握。
2 bsc-relayer 與 BC、BSC 的連接通道
bsc-relayer 與 BC辆飘、BSC 均通過(guò) RPC 進(jìn)行通信啦辐,RPC 的信息記錄在 config/config.json 文件下。
-
bsc-relayer <-----> BC
bsc-relayer 通過(guò)發(fā)送不同的請(qǐng)求信息獲取 BC 上的數(shù)據(jù)蜈项,例如abci_info
芹关、block
等。 -
bsc-relayer <-----> BSC
bsc-relayer 直接使用了 Etheruem 提供的 RPC 模塊紧卒,可直接調(diào)用發(fā)送數(shù)據(jù)或者請(qǐng)求侥衬,其中最常用的是eth_sendRawTransaction
進(jìn)行交易的發(fā)送(用來(lái)調(diào)用前文提到的兩個(gè)合約TendermintLightClient.sol
和CrossChain.sol
)。
3 拉取常侦、同步跨鏈數(shù)據(jù)包
3.1 拉取跨鏈?zhǔn)录?info
bsc-relayer 會(huì)一直輪詢每個(gè)高度的 BC 塊的所有 跨鏈?zhǔn)录?/strong>浇冰,事件的格式如下:
{
"type": "IBCPackage",
"attributes":
[
{
"key": "IBCPackageInfo",
"value": "96::8::19"
}
]
}
其中:
- type 為 "IBCPackage",表示跨鏈包聋亡;
- key 為 "IBCPackageInfo"肘习,表示跨鏈包的信息;
-
value 通過(guò)“::”分隔為3個(gè)字段坡倔,分別為
CrossChainID of destination chain
漂佩、channel id
、sequence
罪塔;
-
CrossChainID
對(duì)于 bsc-relayer 來(lái)說(shuō)投蝉,srcCrossChainID 為 BC 的 ChainID,destCrossChainID 為 BSC 的 ChainID征堪。 -
channel id
channel id 為跨鏈調(diào)用 BSC 系統(tǒng)合約(solidity)的 id瘩缆,bsc-relayer 中包含有以下 4 個(gè) id:
channel | id | contract | address |
---|---|---|---|
BIND_CHANNELID | 0x01 | TokenManager.sol | 0x0000000000000000000000000000000000001008 |
TRANSFER_IN_CHANNELID | 0x02 | TokenHub.sol | 0x0000000000000000000000000000000000001004 |
TRANSFER_OUT_CHANNELID | 0x02 | TokenHub.sol | 0x0000000000000000000000000000000000001004 |
STAKING_CHANNELID | 0x08 | BSCValidatorSet.sol | 0x0000000000000000000000000000000000001000 |
-
sequence
每個(gè) channel id 都對(duì)應(yīng)有一個(gè) sequence,用來(lái)計(jì)數(shù)佃蚜。
(BSC 相關(guān)系統(tǒng)合約邏輯將在后續(xù)章節(jié)中進(jìn)行介紹)
3.2 拉取跨鏈?zhǔn)录?payload
bsc-relayer 根據(jù) channel id
和 sequence
組合生成唯一標(biāo)識(shí)庸娱,通過(guò) RPC 向 BC 請(qǐng)求對(duì)應(yīng)的 payload
着绊,這些 payload
以 bytes 的形式傳到 bsc-relayer。
3.3 同步
bsc-relayer 將上述的 payload 生成調(diào)用 CrossChain.sol
的 tx 熟尉,通過(guò) RPC 發(fā)送到 BSC归露。
4 拉取、同步 BC 塊頭
bsc-relayer 拉取的是 BC 的塊頭斤儿,BC 的塊頭本質(zhì)上是 Tendermint 塊頭剧包,這里就不得不先介紹一下 Tendermint 的相關(guān)特性了:
In Tendermint, validators agree on a block before processing it. This means that the signatures and state root for that block aren't included until the next block. Thus, each block contains a field called LastCommit, which contains the votes responsible for committing the previous block, and a field in the block header called AppHash, which refers to the Merkle root hash of the application after processing the transactions from the previous block. So, if we want to verify the AppHash from height H, we need the signatures from LastCommit at height H+1. (And remember that this AppHash only contains the results from all transactions up to and including block H-1)
上文的大概意思是說(shuō),一個(gè)塊的 狀態(tài) 和 簽名 等數(shù)據(jù)需要至少等到下一個(gè)塊才能得到往果,所以如果需要驗(yàn)證高度為 H 的塊疆液,需要等到高度為 H+1 的塊的 LastCommit
信息(聲明一下,本人不是很了解 Tendermint 的算法棚放,如果某些地方說(shuō)的不正確請(qǐng)及時(shí)指明)枚粘。
bsc-relayer 拉取 BC 塊頭的行為完全是 自身驅(qū)動(dòng) 的,當(dāng)出現(xiàn)以下兩種情況時(shí)進(jìn)行觸發(fā):
-
BC validator 集合更新
BC 本地會(huì)定期更新 bc-validator 集合飘蚯,BSC 需要獲取每一輪的 bc-validator 集合信息馍迄,用于驗(yàn)證跨鏈數(shù)據(jù),bc-validator 集合信息包括所有 bc-validator 的賬號(hào)局骤、公鑰攀圈、投票(VotingPower)等。
bsc-relayer 會(huì)一直輪詢每個(gè)高度的 BC 塊的 bc-validator 集合是否發(fā)生變化(包括當(dāng)前正在工作的 bc-validator 集合和下一輪即將更新的 bc-validator 集合)峦甩,如果發(fā)生變化赘来,則會(huì)打包 當(dāng)前工作的 bc-validator 集合信息、下一輪更新的 bc-validator 集合信息凯傲、當(dāng)前查詢的 BC 高度的塊頭犬辰,將其生成調(diào)用TendermintLightClient.sol
的 tx,并發(fā)送到 BSC冰单。 -
查到跨鏈數(shù)據(jù)包
bsc-relayer 會(huì)一直輪詢每個(gè)高度的 BC 塊是否有跨鏈數(shù)據(jù)包幌缝,如果在高度 H 查到了跨鏈數(shù)據(jù)包,如前文所述诫欠,則會(huì)先打包高度為 H+1 的塊頭 和 validator 集合信息涵卵,將其生成調(diào)用TendermintLightClient.sol
的 tx,并發(fā)送到 BSC荒叼。
5 gas 問(wèn)題
上述可知 bsc-relayer 會(huì)不停向 BSC 發(fā)送交易轿偎,相當(dāng)于中繼自己出 gas 進(jìn)行工作。為了彌補(bǔ)這一損失被廓,BSC 會(huì)在 RelayerIncentivize.sol
系統(tǒng)合約中向 bsc-relayer 發(fā)放系統(tǒng) reward坏晦,reward 應(yīng)該會(huì)遠(yuǎn)高于扣掉的 gas,否則就虧本 gg 了。
6 總結(jié)
bsc-relayer 作為中繼昆婿,不參與處理具體的數(shù)據(jù)結(jié)構(gòu)间护,只是作為數(shù)據(jù)的監(jiān)控者和搬運(yùn)工。