单公证人模式实现测试链间跨链【入门教程(下)】

2024-02-04 08:10

本文主要是介绍单公证人模式实现测试链间跨链【入门教程(下)】,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

3.5监听区块链部分代码

监听器用的是web3.js 文件,使用VS Code编码

下面代码中存在的abi文件需要自己去复制合约的abi字节码,复制后在vs中建立一个xxx.abi文件,将内容粘贴进去即可,如果是在同个文件夹内使用我这种路径方式即可,实在不行就写绝对路径吧。
在这里插入图片描述
使用subscribe监听事件,并发送交易

var Web3 = require("web3");
const {Transaction: Tx, Transaction} = require("ethereumjs-tx");
var fs = require('fs');// fs模块读取.sol合约文件var  privateKey = Buffer.from("2bc4b12344dcd517c48f7eba2e5b5d0c206c6a124d7fd938f392a881785a4021",'hex');// 私钥
var  fromAddress = "0x6123493654b3E23D9007e56EE7131b6dd305dfDd"; //账户地址//---------------------------------------
//-------Ropsten测试网配置----------------
var ropsten_wssurl = "wss://ropsten.infura.io/ws/v3/99b7804fba9a484f8afd1fac5bc75cdf"; 
// 如果报错Error: Transaction has been reverted by the EVM 可能是vpn出问题了,检查网络
// chainid=3  如果报错Error: connection not open on send()  换个infura节点即可var ropsten_web3 = new Web3(new Web3.providers.WebsocketProvider(ropsten_wssurl));
var ropsten_ABI = JSON.parse(fs.readFileSync("ropsten_bridge.abi").toString());
var ropsten_contract_address = '0x173EB118FA344f9D1E2f9b3cB94F4E650D6637B5';//ropsten的跨链桥地址***
//               NATIVE_RETH = 0x4a03c24614a251a1b75c4ccCC5b3bE7b542f716f
//                ERC20_BETH = 0x0392Ae4b926ABf4abDDE26e9E02103d9ac4E6275
var ropsten_contract = new ropsten_web3.eth.Contract(ropsten_ABI,ropsten_contract_address);//---------------------------------------
//-------Rinkeby测试网配置----------------
var rinkeby_wssurl = "wss://rinkeby.infura.io/ws/v3/99b7804fba9a484f8afd1fac5bc75cdf"; 
var rinkeby_web3 = new Web3(new Web3.providers.WebsocketProvider(rinkeby_wssurl));
var rinkeby_ABI = JSON.parse(fs.readFileSync("rinkeby_bridge.abi").toString());
var rinkeby_contract_address = '0x729eC6Df5470099AADfa1ebD770722b45393cC7A';//rinkeby的跨链桥地址***
//               NATIVE_BETH = 0x988b22b8502eC4443dC2a0D5DbbbdB85aBcE3b47
//                ERC20_RETH = 0x94F8fF14EE6D4fbB90750fF4E6630080Cbc29b5d
var rinkeby_contract = new rinkeby_web3.eth.Contract(rinkeby_ABI,rinkeby_contract_address); console.log('ropsten_bridge subscription start');
ropsten_web3.eth.subscribe('logs', {address: ropsten_contract_address,topics:['0x4db14528ca520835e36decc6d6140e6cefbd849bedda8745497406a62317c80d'] //********这里记得换成监听事件的十六进制}, function(error, result){if (!error)console.log('this is ropsten_subcription');info=ropsten_web3.eth.abi.decodeLog([{ // 解析日志内容type: 'address',name: 'token'},{type:'string',name:'symbol'},{type:'uint',name:'fromChainID'},{type:'address',name:'from'},{type:'uint',name:'toChainID'},{type:'address',name:'toAddress'},{type:'uint',name:'amount'}],result.data,result.topics);{ // 获取log的信息SYMBOL = info[1];TO_CHAINID = info[4];TO = info[5];VALUE = info[6];}if(TO_CHAINID == 4){//跨到rinkeby,使用rinkeby_web3往rinkeby发交易var DATA=rinkeby_contract.methods.receipt(SYMBOL,TO,VALUE).encodeABI(); rinkeby_web3.eth.getTransactionCount(fromAddress, rinkeby_web3.eth.defaultBlock.pending).then(function(nonce){var rawTx={nonce: rinkeby_web3.utils.toHex(nonce++),gasLimit: rinkeby_web3.utils.toHex(99000),gasPrice: rinkeby_web3.utils.toHex(10e9), // 10 Gweito: rinkeby_contract_address,//目标合约地址from: fromAddress, //privateChain_data: DATA // thanks @abel30567}var tx=new Tx(rawTx,{chain:'rinkeby'});// 如果说invalid account ,在tx后面加连的名称 tx.sign(privateKey);var serializedTx = tx.serialize();console.log('will send transaction');rinkeby_web3.eth.sendSignedTransaction('0x' + serializedTx.toString('hex')).on('receipt', console.log);console.log('had been sent transaction');})   }}
)console.log('rinkeby_bridge subscription start');
rinkeby_web3.eth.subscribe('logs', {address: rinkeby_contract_address,//监听rinkeby链上的合约地址的事件topics:['0x4db14528ca520835e36decc6d6140e6cefbd849bedda8745497406a62317c80d'] //********这里记得换}, function(error, result){if (!error)console.log('this is rinkeby_subcription');info=rinkeby_web3.eth.abi.decodeLog([{ // 解析日志内容type: 'address',name: 'token'},{type:'string',name:'symbol'},{type:'uint',name:'fromChainID'},{type:'address',name:'from'},{type:'uint',name:'toChainID'},{type:'address',name:'toAddress'},{type:'uint',name:'amount'}],result.data,result.topics);{ // 获取log的信息SYMBOL = info[1];TO_CHAINID = info[4];TO = info[5];VALUE = info[6];}if(TO_CHAINID == 3){//跨到ropsten,使用ropsten_web3往ropsten发交易var DATA=ropsten_contract.methods.receipt(SYMBOL,TO,VALUE).encodeABI(); ropsten_web3.eth.getTransactionCount(fromAddress, ropsten_web3.eth.defaultBlock.pending).then(function(nonce){var rawTx={nonce: ropsten_web3.utils.toHex(nonce++),gasLimit: ropsten_web3.utils.toHex(99000),gasPrice: ropsten_web3.utils.toHex(10e9), // 10 Gweito: ropsten_contract_address,//目标合约地址from: fromAddress, //privateChain_data: DATA // thanks @abel30567}var tx=new Tx(rawTx,{chain:'ropsten'});// 如果说invalid account ,在tx后面加连的名称 tx.sign(privateKey);var serializedTx = tx.serialize();console.log('will send transaction');ropsten_web3.eth.sendSignedTransaction('0x' + serializedTx.toString('hex')).on('receipt', console.log);console.log('had been sent transaction');})   }}
)

3.6 智能合约、监听 初始化 设置

3.6.1 钱包与编译器连接

使用小狐狸钱包,点开左上角连接按钮,将钱包与Remix编译器进行连接。并选择实验所需连接的区块链。接下来会在两个区块链间反复切换,我会在图片中表明当前环境所处区块链。
在这里插入图片描述
在这里插入图片描述

接着点击Remix编译器左侧,选择injected web3,连接我们的环境。
在这里插入图片描述
在这里插入图片描述

可以发现已经连上了我们的区块链并且使用我们的区块链账户。
我们做的系统是跨链交易所,此处假设为两条链之间进行跨链资产转移,那么A链的原生代币跨链到B链时就变成了B链的ERC20代币,因此我们这个过程中每条链需要用到三个合约——原生代币管理合约、ERC20合约(可通用化)、跨链桥合约。

3.6.2 在Ropsten上部署并初始化合约

(1)部署并初始化跨链桥合约
选择部署合约的账户(user2),部署合约时输入该链对应的原生代币。此链为Ropsten,因此原生代币为RETH。
在这里插入图片描述

成功部署后可以获得该合约的地址,右侧可以复制合约地址

在这里插入图片描述

(2)部署并初始化原生代币合约(为了区分父子合约地址,部署代币合约使用的地址为uesr3)
为合约初始化,输入父合约的地址——跨链桥地址
在这里插入图片描述

(3)部署并初始化ERC20合约(此处仍使用uesr3账号部署合约)
输入初始化总代币数量(此处用于测试存取款功能),输入父合约地址——跨链桥地址,点击transact部署合约
在这里插入图片描述

(4)往跨链合约添加代币合约地址
在代币合约部署前我们无法知晓代币合约地址,因此我们需要在代币合约部署完成后将地址写入到跨链合约中。
我们部署原生代币RETH的合约:输入原生代币名称——RETH,再把刚刚部署完成的合约地址复制粘贴过来,点击transact提交交易。
同样的,部署BETH代币(BETH为Rinkeby链的原生代币,跨链的话则在Ropsten链上创造一种与之等价的ERC20代币,此处也称之为BERH)的操作也是如此。

注意:此操作并非任何人都可以随意操作,因此合约中设置为只有合约创建者才可调用此函数,此合约创建者为user2,故需要点击小狐狸钱包将账户切换至user2。
在这里插入图片描述
在这里插入图片描述

调用跨链合约中的ERC20_CIBTRACT函数查看是否添加合约地址成功。

在这里插入图片描述
在这里插入图片描述

3.6.3 在Rinkeby上部署并初始化合约

Rinkeby上的合约与Ropsten属于镜像合约,只需要修改几个参数即可***(
①修改跨链桥合约的fromchainID,NAVITVE_CHAINID为本链id;
②将ERC20合约中的name,symbol,decimal更改为对应的ERC20代币即可,此合约修改这几个参数即可 实现通用化;
③将原生代币合约(NAVIVE_ETH.sol)的代币属性修改成改链的原生代币)***。

注意:部署时切换到Rinkeby链,其余部署方式参考Ropsten部署与初始化(1)~(4)。

在这里插入图片描述

3.6.4 监听器初始化设置

设置自己的私钥、发送交易账户地址(私钥与账户地址必须对应):
在这里插入图片描述

连接到区块链,输入要连接区块链的RPC_URL:
在这里插入图片描述
(如果无法连接上,可能节点有问题,可以取infura官网创个账户,他会分配一个节点给你,将那个节点地址粘贴过来即可)

合约实例化:
a)导入合约abi字节码。点击remix编译器左侧箭头处即可复制当前合约的字节码。在在VS编译器(用于编写监听程序)中创建Ropsten_bridge.abi文件,将abi码粘贴过去即完成导入。
在这里插入图片描述

b)输入要监听的合约地址:
在这里插入图片描述
在这里插入图片描述

c)输入要监听的事件的十六进制编号:
在这里插入图片描述
在这里插入图片描述

只要写了event事件,topics可以在remix抛出的logs中找到,如:

在这里插入图片描述

4 跨链实验

合约中的其他功能读者可以自行测试,由于篇幅有限,此处仅示例跨链过程

4.1 Ropsten跨原生代币RETH至Rinkeby

已知跨链交易是在A链锁住代币,在B链释放对应的代币。

(1)在跨链开始前,先启动链下监听器
在这里插入图片描述

(2)首先从Ropsten链将原生代币RETH跨链到Rinkeby中——user3输入跨链币种RETH,目标链4(Rinkeby),目标地址(user3在Rinkeby中的地址),最后输入要跨链金额(右图即0.01eth)
在这里插入图片描述
在这里插入图片描述

(3)在跨链前先查看user3在Ropsten上余额——0.1681RETH
在这里插入图片描述

Rinkeby网络中的RETH余额——10^19Wei(10RETH)
右边钱包可知当前处于rinkeby网络
在这里插入图片描述

(4)记录完跨链前数据,开始跨链:
在这里插入图片描述

(5)跨链成功后,监视器返回发送交易成功的信息
在这里插入图片描述

(6)检查Ropsten链上是否减少了对应的原生代币(少的0.0002RETH用于gas消耗)

在这里插入图片描述

(7)检查目标链上是否增加了等量的ERC20_Ropsten代币(将网络切换到Rinkeby)

在这里插入图片描述

4.2 Rinkeby跨ERC20代币RETH至Ropsten

(1)在Ropsten中,RETH是原生代币,但在Rinkeby中BETH才是原生代币,因此在Rinkeby中RETH为ERC20代币,即RopstenETH在Rinkeby中的等价替代品。由于是ERC20代币,在交易所中进行转账时需要先给交易所提供代理金额
在这里插入图片描述

(2)使用Rinkeby链中的跨链桥中的general_cross_transfer函数进行ERC20资产跨链
在这里插入图片描述

(3)跨链前检查Ropsten中RETH余额
在这里插入图片描述

(4)开始跨链
在这里插入图片描述

(5)跨链成功后,监视器返回发送交易成功的信息

在这里插入图片描述

(6)检查ropsten链是否增加了对应的RETH
在这里插入图片描述

这篇关于单公证人模式实现测试链间跨链【入门教程(下)】的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/676847

相关文章

SpringBoot3实现Gzip压缩优化的技术指南

《SpringBoot3实现Gzip压缩优化的技术指南》随着Web应用的用户量和数据量增加,网络带宽和页面加载速度逐渐成为瓶颈,为了减少数据传输量,提高用户体验,我们可以使用Gzip压缩HTTP响应,... 目录1、简述2、配置2.1 添加依赖2.2 配置 Gzip 压缩3、服务端应用4、前端应用4.1 N

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Python FastAPI+Celery+RabbitMQ实现分布式图片水印处理系统

《PythonFastAPI+Celery+RabbitMQ实现分布式图片水印处理系统》这篇文章主要为大家详细介绍了PythonFastAPI如何结合Celery以及RabbitMQ实现简单的分布式... 实现思路FastAPI 服务器Celery 任务队列RabbitMQ 作为消息代理定时任务处理完整

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

Java实现时间与字符串互相转换详解

《Java实现时间与字符串互相转换详解》这篇文章主要为大家详细介绍了Java中实现时间与字符串互相转换的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、日期格式化为字符串(一)使用预定义格式(二)自定义格式二、字符串解析为日期(一)解析ISO格式字符串(二)解析自定义