区块链研究实验室-如何让以太坊智能合约与脱链服务互通
李悔之2015
发表于 2022-12-21 03:38:01
108
0
0
简而言之,您可以在处理事务时使智能合约发出事件。一旦事务被挖掘,事件数据就可以读取。在下图中,您可以看到您的DApp发出的事件可以从您的API的任何服务中检索。4 d# u* `5 P1 K& O* v
8 B( E) G" z( D* ^6 _& i+ D
目前有两种方法可以获取事件数据:5 s* s% T: D- z2 R
目前有两种方法可以获取事件数据:# J& J7 s5 P: Q& O% r, T3 C
选项1:使用getTransactionReceipt; o% _3 ^* }; V s% e8 ?
选项2:为事件注册监听器
您可以使用选项1来获取事件数据,但是可能会存在潜在问题,因为在区块被挖掘之前收据是空的。你可以提出调查,但这很难做到。8 u' {2 J( O A3 O0 ` K
另外,根据网络流量和gasPrice设置,交易可能长时间不会被打包。请注意,事件只有在被挖掘成区块后才会被传播。这意味着可以为任何一个区块调用多个事件。
solidity事件& Y" m- j5 N0 R0 E
在solidity里创建事件需要两个步骤:定义事件,发出事件。
pragma solidity ^0.4.18;
contract MyContract {
event Transfer(address indexed from, address indexed to, uint256 value); /* This is an event */ * d8 [0 ^: }2 {4 d% Z
function MyContract() {} /* a constructor */
function transfer(address _to) public { % T* C1 o1 T9 Z) ]5 _ P
Transfer(msg.sender, _to, msg.value); }}
注意事件Transfer是如何在第5行创建的,然后在函数传输中使用它发出Transfer事件(第10行)。 但会告诉你如何使用一个事件,如果我们用0x123调用传递函数,并从0x4444的地址发送1以太,则事件触发传输事件并应返回[0x4444,0x123,BigNumber(1以太)]。
监听事件, I, f8 `6 C" K* [
以太坊实现了他们的JSON RPC规范,但任何人都可以用他们选择的语言实现自己的库。 我们将使用Truffle和原始web3 Javascript API。 监听事件的一般流程如下:
1、加载合同组件
2、配置web3和使用truffle提供程序) C# z- I- `8 M! Q9 z
3、识别网络id
4、从合同组件中获取正确的合同地址
5、获取JavaScript合约实例
6、使用可选配置调用contractInstace.NAME_OF_EVENT7 z( `! k3 S- e# ^
下面我们有3个样本使用Truffle,Web3在1.0 beta之前,Web3在1.0 beta之后。8 u( @4 O Q0 B2 D) T+ j* l- y
Truffle API
Truffle是web3的一个很好的包装器,它提供了一些便利功能,使用更加轻松。 看一下下面的代码示例:
const Web3 = require('web3') // Web3 0.20.4 or web3 1 beta
const truffleContract = require("truffle-contract")" U' t2 h7 U2 z- x9 d& r5 Z
const contractArtifact = require('./build/contracts/TutorialToken.json'). f; Q! |( z2 M
const providerUrl = 'http://localhost:8545'8 \& @; X) s) { D Q
const provider = new Web3.providers.HttpProvider(providerUrl)
const contract = truffleContract(contractArtifact)
contract.setProvider(provider)
// dirty hack for web3@1.0.0 support for localhost testrpc, see https://github.com/trufflesuite/truffle-contract/issues/56#issuecomment331084530
if (typeof contract.currentProvider.sendAsync !== "function") { , O3 s6 H# W4 ^/ l4 r/ {5 d6 b
contract.currentProvider.sendAsync = function() {
return contract.currentProvider.send.apply(
contract.currentProvider,
arguments & b- l3 K, i) H" W5 f; [
); 9 I1 f2 _1 b: r, P
}; }
contract.deployed() .then(contractInstance => { & |# ]2 a* k% G+ w1 Z
const event = contractInstance.Transfer(null, {fromBlock: 0}, (err, res) => { # x( D- k% c! w
if(err) { 3 [- U3 {; e/ g- F0 X1 H5 a
throw Error(err)
} }) & N8 j1 F9 O% |& u6 G n4 B# D
event.watch(function(error, result){
if (error) { return console.log(error) } ) |3 M- T3 X: W; f: [# X( {% R
if (!error) { ! O# u8 P, F; b( p8 O& V( b
// DO ALL YOUR WORK HERE!
let { args: { from, to, value }, blockNumber } = result : S' ^: C# | z7 L+ F: C/ {3 I* \
console.log(`----BlockNumber (${blockNumber})----`) : ^$ \, e6 j3 x6 Q
console.log(`from = ${from}`)
console.log(`to = ${to}`) ' n- o/ G% O- [
console.log(`value = ${value}`) 6 Y- S0 B0 z$ u' N0 b9 \# b
console.log(`----BlockNumber (${blockNumber})----`)
} : j7 u( |" Y/ J M; ]- g o
}) })
.catch(e => {
console.error('Catastrophic Error!') 4 q; r9 n5 S# Y6 h6 i% t# R
console.error(e)
})
前9行需要一些模块,并设置truffle和web3能够正常网络通信。 第12-19行作为hack使其与web3一起使用,可以忽略。 工作从第21行开始,带有contract.deployed()。 这是一个很好的truffle辅助函数,可以从创建并保存到./build/contracts的工件中选择正确的合同地址。 返回contractInstance(第23行)后,我们通过调用contractInstance.NAME_OF_EVENT来“创建”该事件。 存在更多选项,可在此处找到。 我从0块中选择,因为它是我们自己的testrpc。 对于您的合同,最好选择更接近合同创建时的块编号。
Raw Web3(0.20.4及以下)
对于第二个示例,我们将直接使用Web3。 除了上一个示例中的hack之外,代码流程类似。
const Web3 = require('web3') // Works with web3 0.20.4
const contractArtifact = require('./build/contracts/TutorialToken.json')' p. g d5 O7 n% Y: p
const web3 = new Web3(); m) P) W0 ^6 ~2 r5 ^9 \
const providerUrl = 'http://localhost:8545'
const provider = new Web3.providers.HttpProvider(providerUrl)web3.setProvider(provider)
const networkId = web3.version.networkconst $ b$ a9 X$ y, C. h% n5 {3 S
contractAddr = contractArtifact.networks[networkId].address5 u' ^7 X# v( b9 q/ `/ r
const TutorialToken = web3.eth.contract(contractArtifact.abi, contractAddr)8 C( z* Z3 G& l' {
const contract = TutorialToken.at(contractAddr)
const event = contract.Transfer()
event.watch(function(error, result){
if (error) { return console.error(error) } f" ?* Z$ H2 ]
let { args: { from, to, value }, blockNumber } = result
console.log(`----BlockNumber (${blockNumber})----`) % J$ R+ i! q* @$ k. e
console.log(`from = ${from}`) * t/ k( r2 N `5 j9 M; g0 l5 a
console.log(`to = ${to}`) 3 P2 ^3 C( ]; p" d% t
console.log(`value = ${value}`)
console.log(`----BlockNumber (${blockNumber})----`) }); & U1 O. K5 H6 x# r/ b, D
最重要的区别是第9-12行,我们手动从组件中抓取正确的网络。 我们没有很好的truffle.deployed功能来实现这一步。 现在我们在truffle示例中执行与之前相同的操作,并通过调用contract.NAME_OF_EVENT来创建事件。 在这个版本中我没有放一个块号,但是你可以传递与truffle版相同的东西。) N* d3 _( i- s. Z @! [, `( s
Raw Web3 1.0.0 beta
Web3确实为API带来了大量更改。 一个重大变化是我们必须使用webSockProvider来注册事件而不是httpProvider。$ u& i; I; U7 v I, J
const Web3 = require('web3') // Works with web3 1.0.0-beta27
const contractArtifact = require('./build/contracts/TutorialToken.json')9 w! b7 s' _/ k# L3 ~, n9 m
const web3 = new Web3()
const providerUrl = 'ws://localhost:8545' // requires # https://github.com/trufflesuite/ganache-cli/releases/tag/v7.0.0-beta.0 or https://github.com/trufflesuite/ganache/releases/tag/v1.1.0-beta.0
const provider = new Web3.providers.WebsocketProvider(providerUrl)/ ]% V$ g6 F$ Y3 ~8 V4 t
web3.setProvider(provider)
web3.eth.net.getId() .then(networkId => {
const contractAddr = contractArtifact.networks[networkId].address % y$ r" ^4 o# S7 L; Q
const TutorialToken = new web3.eth.Contract(contractArtifact.abi, contractAddr) 0 ]: X7 W7 m* P) ]2 q Q. ]3 M+ N
TutorialToken.events.Transfer({fromBlock: 0}, $ b! j: H& O! J* e, k$ l
function(error, event){ console.log(error) }) # U# c7 f4 I: s+ z- C0 w' y' G
.on('data', (log) => {
let { 3 k, J }5 x7 ~% E
returnValues: { from, to, value }, blockNumber } = log
console.log(`----BlockNumber (${blockNumber})----`)
console.log(`from = ${from}`)
console.log(`to = ${to}`)
console.log(`value = ${value}`)
console.log(`----BlockNumber (${blockNumber})----`) 8 h# F4 k: k- t/ a
}) ) u( u% w; p+ ^/ h9 G0 t) Q% ?
.on('changed', (log) => {
console.log(`Changed: ${log}`)
}) .on('error', (log) => { 1 R3 j0 I* h+ P: e( R- Q& H
console.log(`error: ${log}`)
}) }) ; Q* z6 D, E0 W f! k
此外,获取networkId也已更新。 但过程是一样的。 我们使用web3.eth.net.getId()获取networkId,然后从Artifact中选择正确的合同地址。 初始化合同现在是一个类构造函数,在web3.eth.Contract中有一个大写C. 最后,我们创建了事件监听器。 我们现在必须自己处理“数据”,“更改”和“错误”事件。 之前它包含在前面2个示例中的watch功能中。
总结
您应该了解如何将您的offchain服务连接到onchain事件。 这意味着您现在可以将区块链集成到现有基础架构中。
成为第一个吐槽的人