区块链研究实验室-如何让以太坊智能合约与脱链服务互通
李悔之2015
发表于 2022-12-21 03:38:01
87
0
0
简而言之,您可以在处理事务时使智能合约发出事件。一旦事务被挖掘,事件数据就可以读取。在下图中,您可以看到您的DApp发出的事件可以从您的API的任何服务中检索。
8 `4 L# p- l! `' Q: Y
目前有两种方法可以获取事件数据:
目前有两种方法可以获取事件数据:" J+ a8 x% P6 b
选项1:使用getTransactionReceipt
选项2:为事件注册监听器# ?* X: j! v& C( Q1 y5 |
您可以使用选项1来获取事件数据,但是可能会存在潜在问题,因为在区块被挖掘之前收据是空的。你可以提出调查,但这很难做到。
另外,根据网络流量和gasPrice设置,交易可能长时间不会被打包。请注意,事件只有在被挖掘成区块后才会被传播。这意味着可以为任何一个区块调用多个事件。
solidity事件& g. ]( {9 o/ h# w6 O
在solidity里创建事件需要两个步骤:定义事件,发出事件。, C* f, k" u- H3 b8 d1 y
pragma solidity ^0.4.18;
contract MyContract {
event Transfer(address indexed from, address indexed to, uint256 value); /* This is an event */ ' d0 ]: Y- Q8 ?1 ^0 A5 k. p. H. ?
function MyContract() {} /* a constructor */ 1 l/ X: K3 J6 \
function transfer(address _to) public { 6 Z9 H1 N! i0 n8 N9 o3 r0 r
Transfer(msg.sender, _to, msg.value); }} {; x7 Q6 L+ ^: q: ?
注意事件Transfer是如何在第5行创建的,然后在函数传输中使用它发出Transfer事件(第10行)。 但会告诉你如何使用一个事件,如果我们用0x123调用传递函数,并从0x4444的地址发送1以太,则事件触发传输事件并应返回[0x4444,0x123,BigNumber(1以太)]。- a; G/ I- `* Q ?3 e o" e
监听事件
以太坊实现了他们的JSON RPC规范,但任何人都可以用他们选择的语言实现自己的库。 我们将使用Truffle和原始web3 Javascript API。 监听事件的一般流程如下:' w' y5 k. ^4 j' [1 I" X# i- q
1、加载合同组件# ?1 ^) i; y8 v4 ^
2、配置web3和使用truffle提供程序
3、识别网络id
4、从合同组件中获取正确的合同地址
5、获取JavaScript合约实例: R- ]- ?( J8 K9 ?' g, m" B& M
6、使用可选配置调用contractInstace.NAME_OF_EVENT
下面我们有3个样本使用Truffle,Web3在1.0 beta之前,Web3在1.0 beta之后。) E9 w& d* j8 x$ \4 \. H7 c
Truffle API
Truffle是web3的一个很好的包装器,它提供了一些便利功能,使用更加轻松。 看一下下面的代码示例:- U1 u/ F$ ]/ w' f6 _1 n/ ]; n& E
const Web3 = require('web3') // Web3 0.20.4 or web3 1 beta/ ]$ D! V9 E- Y& Q. ?
const truffleContract = require("truffle-contract")
const contractArtifact = require('./build/contracts/TutorialToken.json')
const providerUrl = 'http://localhost:8545'
const provider = new Web3.providers.HttpProvider(providerUrl)
const contract = truffleContract(contractArtifact). o8 T: N9 F# V u7 w: U" R; }2 o& t# P
contract.setProvider(provider)7 q8 e. m z: t) J% m* ^
// dirty hack for web3@1.0.0 support for localhost testrpc, see https://github.com/trufflesuite/truffle-contract/issues/56#issuecomment331084530, V3 b" ?" F9 u2 Q+ B
if (typeof contract.currentProvider.sendAsync !== "function") { 1 ]+ U- h( [& F# A; k
contract.currentProvider.sendAsync = function() {
return contract.currentProvider.send.apply(
contract.currentProvider, 3 {/ P/ X5 d: f: c# t. [; ]" h- r
arguments $ Z* j7 A) n8 w7 V* k$ w
); + W+ A+ y4 G- J7 ]2 x, K% ]
}; }
contract.deployed() .then(contractInstance => {
const event = contractInstance.Transfer(null, {fromBlock: 0}, (err, res) => { % b& G9 Y( f6 G, I0 T1 b2 x
if(err) { $ e7 c. q7 } r2 d( {2 ]
throw Error(err) 3 k4 a$ O+ M, o
} })
event.watch(function(error, result){ $ L1 j( C: u! f9 [' S' u
if (error) { return console.log(error) }
if (!error) { $ ?3 W! P$ @: V$ h: Z
// DO ALL YOUR WORK HERE! # R+ x; @2 b7 `4 _- m5 o
let { args: { from, to, value }, blockNumber } = result H/ _$ t1 b9 w: r* [
console.log(`----BlockNumber (${blockNumber})----`)
console.log(`from = ${from}`) ( T7 i! h6 ~/ j. w$ S& B
console.log(`to = ${to}`) " R% R G0 k( z. s- c! G
console.log(`value = ${value}`) 0 T; `+ j, [% b& M% |
console.log(`----BlockNumber (${blockNumber})----`) ! O# a5 ` Y4 @& i; H4 v! B' D5 j
} - _# K1 ^0 F6 A% {7 X8 Y- ^
}) })
.catch(e => { 1 o9 w ` |: T; ?
console.error('Catastrophic Error!') & M* n2 m7 E. B/ V3 U: U2 h
console.error(e) ) C- F ?/ k4 m7 \+ o' c
})
前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及以下)( Y; T, Y, Z6 S, M+ ] b3 v# }
对于第二个示例,我们将直接使用Web3。 除了上一个示例中的hack之外,代码流程类似。# @1 M* E# F# C: A; c+ P- g3 W
const Web3 = require('web3') // Works with web3 0.20.45 K3 R" B4 q. i' S# N) R
const contractArtifact = require('./build/contracts/TutorialToken.json')
const web3 = new Web3()0 p4 P3 X- L5 z& u) v8 d
const providerUrl = 'http://localhost:8545'
const provider = new Web3.providers.HttpProvider(providerUrl)web3.setProvider(provider)
const networkId = web3.version.networkconst
contractAddr = contractArtifact.networks[networkId].address
const TutorialToken = web3.eth.contract(contractArtifact.abi, contractAddr)
const contract = TutorialToken.at(contractAddr)1 B! y/ n3 V5 f8 s2 n: n6 I! W
const event = contract.Transfer()1 _" Y! D% J$ i, K# s
event.watch(function(error, result){
if (error) { return console.error(error) }
let { args: { from, to, value }, blockNumber } = result
console.log(`----BlockNumber (${blockNumber})----`)
console.log(`from = ${from}`)
console.log(`to = ${to}`) ' L! I. J) \2 N3 G. A: i, w$ A
console.log(`value = ${value}`)
console.log(`----BlockNumber (${blockNumber})----`) }); & |, x# _4 ^+ F7 |' O
最重要的区别是第9-12行,我们手动从组件中抓取正确的网络。 我们没有很好的truffle.deployed功能来实现这一步。 现在我们在truffle示例中执行与之前相同的操作,并通过调用contract.NAME_OF_EVENT来创建事件。 在这个版本中我没有放一个块号,但是你可以传递与truffle版相同的东西。, u' u3 y$ g- m2 @
Raw Web3 1.0.0 beta* |. Y; Z6 R' d6 S
Web3确实为API带来了大量更改。 一个重大变化是我们必须使用webSockProvider来注册事件而不是httpProvider。
const Web3 = require('web3') // Works with web3 1.0.0-beta27/ a; q# X/ d; X% X2 y" [
const contractArtifact = require('./build/contracts/TutorialToken.json'), _: @; q$ c# ~1 x
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& C q Q5 O- b$ F$ y. |! ^
const provider = new Web3.providers.WebsocketProvider(providerUrl)! Z& M% h+ e, n4 \7 ~" P# z
web3.setProvider(provider)5 b+ ?0 G; q+ z5 Q+ ?0 n
web3.eth.net.getId() .then(networkId => {
const contractAddr = contractArtifact.networks[networkId].address / E! h4 f9 t& g9 H3 G" c: ]* c
const TutorialToken = new web3.eth.Contract(contractArtifact.abi, contractAddr)
TutorialToken.events.Transfer({fromBlock: 0},
function(error, event){ console.log(error) })
.on('data', (log) => {
let { 1 k- O; J. `8 A1 W9 ?; x
returnValues: { from, to, value }, blockNumber } = log
console.log(`----BlockNumber (${blockNumber})----`)
console.log(`from = ${from}`)
console.log(`to = ${to}`) 1 G9 w3 f" @! _0 D" o Y
console.log(`value = ${value}`)
console.log(`----BlockNumber (${blockNumber})----`) , _1 }+ s' ?$ j, v/ _
})
.on('changed', (log) => { / e: f7 K3 _, M* x: [/ c
console.log(`Changed: ${log}`)
}) .on('error', (log) => {
console.log(`error: ${log}`)
}) })
此外,获取networkId也已更新。 但过程是一样的。 我们使用web3.eth.net.getId()获取networkId,然后从Artifact中选择正确的合同地址。 初始化合同现在是一个类构造函数,在web3.eth.Contract中有一个大写C. 最后,我们创建了事件监听器。 我们现在必须自己处理“数据”,“更改”和“错误”事件。 之前它包含在前面2个示例中的watch功能中。4 H5 ^5 N3 w4 Y" M9 M
总结
您应该了解如何将您的offchain服务连接到onchain事件。 这意味着您现在可以将区块链集成到现有基础架构中。
成为第一个吐槽的人