区块链研究实验室-如何让以太坊智能合约与脱链服务互通
李悔之2015
发表于 2022-12-21 03:38:01
89
0
0
简而言之,您可以在处理事务时使智能合约发出事件。一旦事务被挖掘,事件数据就可以读取。在下图中,您可以看到您的DApp发出的事件可以从您的API的任何服务中检索。
目前有两种方法可以获取事件数据:: X2 L, Z; w2 H# s4 c1 t: L
目前有两种方法可以获取事件数据:0 G/ i) \+ x6 c0 C% t }
选项1:使用getTransactionReceipt
选项2:为事件注册监听器
您可以使用选项1来获取事件数据,但是可能会存在潜在问题,因为在区块被挖掘之前收据是空的。你可以提出调查,但这很难做到。6 G: X; d# y8 J F8 N1 B
另外,根据网络流量和gasPrice设置,交易可能长时间不会被打包。请注意,事件只有在被挖掘成区块后才会被传播。这意味着可以为任何一个区块调用多个事件。/ ~' g$ B E& \9 j6 s6 [& X9 C
solidity事件
在solidity里创建事件需要两个步骤:定义事件,发出事件。
pragma solidity ^0.4.18;0 `( t3 A5 }8 [1 b) Y6 H
contract MyContract { ( z {! x/ u. ] m/ y
event Transfer(address indexed from, address indexed to, uint256 value); /* This is an event */
function MyContract() {} /* a constructor */
function transfer(address _to) public {
Transfer(msg.sender, _to, msg.value); }}
注意事件Transfer是如何在第5行创建的,然后在函数传输中使用它发出Transfer事件(第10行)。 但会告诉你如何使用一个事件,如果我们用0x123调用传递函数,并从0x4444的地址发送1以太,则事件触发传输事件并应返回[0x4444,0x123,BigNumber(1以太)]。) I" u% D A, c5 k3 E
监听事件
以太坊实现了他们的JSON RPC规范,但任何人都可以用他们选择的语言实现自己的库。 我们将使用Truffle和原始web3 Javascript API。 监听事件的一般流程如下:
1、加载合同组件
2、配置web3和使用truffle提供程序9 t- v4 J7 @0 j9 l- _/ `
3、识别网络id7 y4 v- V4 l( X6 G6 ]8 h/ [
4、从合同组件中获取正确的合同地址1 [ y, p0 `) E$ i
5、获取JavaScript合约实例/ H5 g% P* V" r9 c4 V
6、使用可选配置调用contractInstace.NAME_OF_EVENT
下面我们有3个样本使用Truffle,Web3在1.0 beta之前,Web3在1.0 beta之后。
Truffle API; s* f: j3 q& K4 l' p8 c) T! s. ]+ X, ]
Truffle是web3的一个很好的包装器,它提供了一些便利功能,使用更加轻松。 看一下下面的代码示例:
const Web3 = require('web3') // Web3 0.20.4 or web3 1 beta) o' ^+ A& Z0 B" c, }* a( S p/ s
const truffleContract = require("truffle-contract")7 Z6 }+ k' U I C R# U7 r
const contractArtifact = require('./build/contracts/TutorialToken.json')
const providerUrl = 'http://localhost:8545'; O! l$ X. {# S/ r# Y9 L
const provider = new Web3.providers.HttpProvider(providerUrl)
const contract = truffleContract(contractArtifact)
contract.setProvider(provider)& P6 F. j: _( f" {. Y
// dirty hack for web3@1.0.0 support for localhost testrpc, see https://github.com/trufflesuite/truffle-contract/issues/56#issuecomment331084530$ o9 e8 h, z' H4 H
if (typeof contract.currentProvider.sendAsync !== "function") {
contract.currentProvider.sendAsync = function() { , {% I9 \; m( P% F; l
return contract.currentProvider.send.apply(
contract.currentProvider, . f/ t5 ]3 `) {8 E2 q' P
arguments : g; L2 k- X$ T8 r/ V
);
}; }2 H: ]: i0 B. n, R" v& [* z
contract.deployed() .then(contractInstance => { 7 b! y0 z+ ? t4 Y
const event = contractInstance.Transfer(null, {fromBlock: 0}, (err, res) => { 4 s+ C, h8 s4 {& s; {
if(err) {
throw Error(err) * e7 h5 r, {/ p& C* e
} })
event.watch(function(error, result){ " K- K5 J5 j. h2 B7 ~; [5 P
if (error) { return console.log(error) } ' Z' ~9 O6 d* F6 L2 N# c2 k
if (!error) {
// DO ALL YOUR WORK HERE!
let { args: { from, to, value }, blockNumber } = result
console.log(`----BlockNumber (${blockNumber})----`)
console.log(`from = ${from}`)
console.log(`to = ${to}`) $ c1 i/ C( {! |. ]* l
console.log(`value = ${value}`)
console.log(`----BlockNumber (${blockNumber})----`) ) {( Q& [ u4 F0 w0 H7 e/ I. R# V! j! u
}
}) })
.catch(e => {
console.error('Catastrophic Error!') ' T7 R+ R6 N7 m7 Z. X8 X! n
console.error(e) - ^+ z1 ?8 S# r! t' g6 z3 s* Z3 ?
}) 0 J' J- t) W! X0 B0 f
前9行需要一些模块,并设置truffle和web3能够正常网络通信。 第12-19行作为hack使其与web3一起使用,可以忽略。 工作从第21行开始,带有contract.deployed()。 这是一个很好的truffle辅助函数,可以从创建并保存到./build/contracts的工件中选择正确的合同地址。 返回contractInstance(第23行)后,我们通过调用contractInstance.NAME_OF_EVENT来“创建”该事件。 存在更多选项,可在此处找到。 我从0块中选择,因为它是我们自己的testrpc。 对于您的合同,最好选择更接近合同创建时的块编号。 F3 [% \3 `' ~; C
Raw Web3(0.20.4及以下)( d' h/ p. p6 F) X- x' T" E
对于第二个示例,我们将直接使用Web3。 除了上一个示例中的hack之外,代码流程类似。
const Web3 = require('web3') // Works with web3 0.20.4
const contractArtifact = require('./build/contracts/TutorialToken.json')
const web3 = new Web3()
const providerUrl = 'http://localhost:8545'$ W7 N; k1 B( }) I& V
const provider = new Web3.providers.HttpProvider(providerUrl)web3.setProvider(provider)
const networkId = web3.version.networkconst , N6 Y0 Y- x0 T! d, Y
contractAddr = contractArtifact.networks[networkId].address( V$ O! S% H7 [4 j, Z- O
const TutorialToken = web3.eth.contract(contractArtifact.abi, contractAddr)
const contract = TutorialToken.at(contractAddr)/ H! F; y: E- b! Q9 Y
const event = contract.Transfer()* M3 x0 s* ~' Y0 Z1 p" `
event.watch(function(error, result){
if (error) { return console.error(error) }
let { args: { from, to, value }, blockNumber } = result
console.log(`----BlockNumber (${blockNumber})----`) 7 D( X8 a% r/ o, N1 y9 U7 J
console.log(`from = ${from}`) 8 w) K2 B: b1 R* K5 S' N3 ]# }+ @
console.log(`to = ${to}`)
console.log(`value = ${value}`)
console.log(`----BlockNumber (${blockNumber})----`) }); 2 ^# ]" t( g$ H/ t7 z- Z4 @: |/ w
最重要的区别是第9-12行,我们手动从组件中抓取正确的网络。 我们没有很好的truffle.deployed功能来实现这一步。 现在我们在truffle示例中执行与之前相同的操作,并通过调用contract.NAME_OF_EVENT来创建事件。 在这个版本中我没有放一个块号,但是你可以传递与truffle版相同的东西。) n3 {% ^$ S0 a$ v3 ~( U( S/ o) Y
Raw Web3 1.0.0 beta
Web3确实为API带来了大量更改。 一个重大变化是我们必须使用webSockProvider来注册事件而不是httpProvider。% }; ?5 m2 z. E1 t
const Web3 = require('web3') // Works with web3 1.0.0-beta27
const contractArtifact = require('./build/contracts/TutorialToken.json')& t7 q, B: }7 ]! r# T1 r
const web3 = new Web3()0 M' ]6 c! _5 j
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.06 V7 h9 y2 o2 h q; ^; C% x
const provider = new Web3.providers.WebsocketProvider(providerUrl)! Y/ f% `) L& A4 \3 F
web3.setProvider(provider)
web3.eth.net.getId() .then(networkId => { 8 U% c8 r U4 x9 f; r" z/ _
const contractAddr = contractArtifact.networks[networkId].address
const TutorialToken = new web3.eth.Contract(contractArtifact.abi, contractAddr) ; |/ G" I; W* i2 w; L
TutorialToken.events.Transfer({fromBlock: 0},
function(error, event){ console.log(error) })
.on('data', (log) => { 9 k( `2 s+ c* m2 {# n/ @
let {
returnValues: { from, to, value }, blockNumber } = log - j* b/ s& C* k3 v+ c
console.log(`----BlockNumber (${blockNumber})----`) $ W) x7 B6 Q4 b' A2 K
console.log(`from = ${from}`) 5 N! t) R' l0 k: S3 E8 d" S
console.log(`to = ${to}`) $ Y! ^3 p/ o% R6 O; K
console.log(`value = ${value}`)
console.log(`----BlockNumber (${blockNumber})----`)
}) : V* R! \, w( T2 b% H7 S1 P
.on('changed', (log) => { 3 w, ]! T6 N/ ^$ ]2 b- p
console.log(`Changed: ${log}`)
}) .on('error', (log) => {
console.log(`error: ${log}`) + U j$ v9 m1 O. q
}) })
此外,获取networkId也已更新。 但过程是一样的。 我们使用web3.eth.net.getId()获取networkId,然后从Artifact中选择正确的合同地址。 初始化合同现在是一个类构造函数,在web3.eth.Contract中有一个大写C. 最后,我们创建了事件监听器。 我们现在必须自己处理“数据”,“更改”和“错误”事件。 之前它包含在前面2个示例中的watch功能中。
总结0 a5 X7 d' i' \. s$ Q+ k# x+ w% N! s
您应该了解如何将您的offchain服务连接到onchain事件。 这意味着您现在可以将区块链集成到现有基础架构中。
成为第一个吐槽的人