W3-2 合约事件
W3-2 合约事件
理解事件
作用
- 合约与外部世界的重要接⼝,通知外部世界链上状态的变化
- 事件有时也作为便宜的存储
编码特征
- 使⽤关键字 event 定义事件,事件不需要实现
- 使⽤关键字 emit 触发事件
- 事件中使⽤
indexed
修饰,表示对这个字段建⽴索引,⽅便外部对该字段过滤查找
pragma solidity ^0.8.0;
contract testEvent {
constructor() {}
event Deposit(address indexed _from, uint _value);
function deposit(uint value) public {
emit Deposit(msg.sender, value);
}
}
保存事件/数据上链
数据编码过程
事件签名为第一个 Topic ,在黄皮书种称作 LOG0
索引事件进行 ABI 编码后用 Topic 部分 (topic[1,2,3..]) 记录下来,详细见👇代码
⾮索引事件进行 ABI 编码后作为 Data 部分保存下来
- 代码示例
1 2 3 4 5 6
// 假定 events 事件格式 event Transfer (address indexed _from, address index _to, uint256 value); // 去除关键词 events,indexed 和 参数名称 _from, _to , value let transferTopic = ethers.utils.keccak256( ethers.utils.toUtf8Bytes("Transfer(address,address,uint256)")); // 等价于 ethers.utils.id("Transfer(address,address,uint256)")
代码示例
事件解码 decoded
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
var ethers = require("ethers")
// 定义要解码的事件的ABI
const eventAbi = [
{
"anonymous": false,
"inputs": [
{
"indexed": true,
"name": "from",
"type": "address"
},
{
"indexed": true,
"name": "to",
"type": "address"
},
{
"indexed": false,
"name": "value",
"type": "uint256"
}
],
"name": "Transfer",
"type": "event"
}
]
// 创建一个新的ethers.ethers.utils.Interface对象
const contractInterface = new ethers.utils.Interface(eventAbi);
// 要解码的topics数组
const topics = [
"0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef",
"0x0000000000000000000000004f9fbb3f1e99b56e0fe2892e623ed36a76fc605d",
"0x000000000000000000000000b8a9fada75c6d891fb77a7988ff9bad9e485ca1c"
];
const data = '0x00000000000000000000000000000000000000000000087f552f6cf833c0c200'
const decodedLog = contractInterface.parseLog({data, topics});
console.log(decodedLog)
事件编码 encoded
1
2
3
4
5
// 以事件中的 data 部分,类型为 unit256,数值为 40127806604948680000000 进行编码
var ethers = require("ethers")
const abiCoder = new ethers.utils.AbiCoder();
const encodedLog = abiCoder.encode(["uint256"], ["40127806604948680000000"]);
console.log(encodedLog)
算法
监听事件
- 监听(实时)事件(使⽤ webSocket 链接)
1
2
3
4
5
6
let erc20 = new ethers.Contract(erc20地址, ABI, provider);
// indexed 的字段可以作为参数加⼊到过滤器中
let filter = erc20.filters.Transfer()
wssprovider.on(filter, (log) => {
console.log(log);
})
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
event = {
blockNumber: 9,
blockHash: '0x5eaa8438d80d30e521b6c48db8c22d5b40255cd472d71cfd387c314486a7e679',
transactionIndex: 0,
removed: false,
address: '0x5FbDB2315678afecb367f032d93F642f64180aa3',
data: '0x00000000000000000000000000000000000000000000000000000000000003e8',
topics: [
'0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef',
'0x000000000000000000000000f39fd6e51aad88f6f4ce6ab8827279cfffb92266',
'0x00000000000000000000000070997970c51812dc3a010c7d01b50e0d17dc79c8'
],
transactionHash: '0x1eda2652cde3d9b3d8dbb1e0441a2139ebd25ba25a97fab3158406a695e33ba9',
logIndex: 0
}
获取事件
获取历史事件
调用方法
- https://docs.ethers.io/v5/api/providers/provider/#Provider-getLogs
- https://docs.ethers.io/v5/concepts/events/#events–filters
1
2
3
4
5
6
7
8
let erc20 = new ethers.Contract(erc20地址, ABI, provider);
let filter = erc20.filters.Transfer()
// 选择区块区间
filter.fromBlock = 1000;
filter.toBlock = 2000; // "latest";
// 获取⽇志
let logs = await provider.getLogs(filter);
let logs = await erc20.queryFilter(filterTo, -10, "latest");
处理事件
解析事件
- 使⽤ ethers.utils.Interface 进⾏编解码
1 2 3 4
const TransferEvent = new ethers.utils.Interface(["event Transfer(address indexed from,address indexed to,uint256 value)"]); const data = TransferEvent.parseLog(logs[i]) console.log("from:" + data.args.from) console.log("from:" + data.args.to)
- Interface (ethers.io)
相关文章
This post is licensed under CC BY 4.0 by the author.