-
Notifications
You must be signed in to change notification settings - Fork 255
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Co-authored-by: lcy1317 <213183464.seu.edu.cn> Co-authored-by: XingQiang Bai <[email protected]>
- Loading branch information
1 parent
016f06e
commit e3bdf64
Showing
2 changed files
with
167 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,165 @@ | ||
# 合约事件推送 | ||
|
||
标签:``go-sdk`` ``事件订阅`` ``Event`` | ||
|
||
---- | ||
## 1. 功能简介 | ||
|
||
合约事件推送功能提供了合约事件的异步推送机制,客户端向节点发送注册请求,在请求中携带客户端关注的合约事件的参数,节点根据请求参数对请求区块范围的`Event Log`进行过滤,将结果分次推送给客户端。 | ||
|
||
## 2. 交互协议 | ||
|
||
客户端与节点的交互基于[`Channel`](https://fisco-bcos-documentation.readthedocs.io/zh_CN/latest/docs/design/protocol_description.html#id4)协议。交互分为三个阶段:注册请求,节点回复,`Event Log`数据推送。 | ||
|
||
### 注册请求 | ||
|
||
客户端向节点发送事件推送的注册请求: | ||
|
||
```Json | ||
// request sample: | ||
{ | ||
"fromBlock": "latest", | ||
"toBlock": "latest", | ||
"addresses": [ | ||
"0x1f3119043aaee8e5080967e6f80eb3135086c138" | ||
], | ||
"topics": [ | ||
"0x91c95f04198617c60eaf2180fbca88fc192db379657df0e412a9f7dd4ebbe95d" | ||
], | ||
"groupID": "1", | ||
"filterID": "bb31e4ec086c48e18f21cb994e2e5967" | ||
} | ||
``` | ||
|
||
- filerID:字符串类型,每次请求唯一,标记一次注册任务 | ||
- groupID:字符串类型,群组ID | ||
- fromBlock:整形字符串,初始区块。“latest” 当前块高 | ||
- toBlock:整形字符串,最终区块。“latest” 处理至当前块高时,继续等待新区块 | ||
- addresses:字符串或者字符串数组:字符串表示单个合约地址,数组为多个合约地址,数组可以为空 | ||
- topics:字符串类型或者数组类型:字符串表示单个topic,数组为多个topic,数组可以为空 | ||
|
||
### 节点回复 | ||
|
||
节点接受客户端注册请求时,会对请求参数进行校验,将是否成功接受该注册请求结果回复给客户端。 | ||
|
||
```Json | ||
// response sample: | ||
{ | ||
"filterID": "bb31e4ec086c48e18f21cb994e2e5967", | ||
"result": 0 | ||
} | ||
``` | ||
|
||
- filterID:字符串类型,每次请求唯一,标记一次注册任务 | ||
- result:整形,返回结果。0成功,其余为失败状态码 | ||
|
||
### Event Log数据推送 | ||
|
||
节点验证客户端注册请求成功之后,根据客户端请求参数条件,向客户端推送`Event Log`数据。 | ||
|
||
```Json | ||
// event log push sample: | ||
{ | ||
"filterID": "bb31e4ec086c48e18f21cb994e2e5967", | ||
"result": 0, | ||
"logs": [ | ||
|
||
] | ||
} | ||
``` | ||
|
||
- filterID:字符串类型,每次请求唯一,标记一次注册任务 | ||
- result:整形 0:`Event Log`数据推送 1:推送完成。客户端一次注册请求对应节点的数据推送会有多次(请求区块范围比较大或者等待新的区块),`result`字段为1时说明节点推送已经结束 | ||
- logs:Log对象数组,result为0时有效 | ||
|
||
## 3. Go SDK教程 | ||
|
||
### 注册接口 | ||
|
||
在FISCO-BCOS Go SDK中`client.Client`类提供合约事件的注册接口,用户可以调用`SubscribeEventLogs`向节点发送注册请求,并设置回调函数。 | ||
|
||
```go | ||
func (c *Client) SubscribeEventLogs(eventLogParams types.EventLogParams, handler func(int, []types.Log)) error { | ||
return c.apiHandler.SubscribeEventLogs(eventLogParams, handler) | ||
} | ||
``` | ||
|
||
#### `types.EventLogParams`注册参数 | ||
|
||
事件回调请求注册的参数: | ||
|
||
```go | ||
type EventLogParams struct { | ||
FromBlock string `json:"fromBlock"` // 订阅起始区块 latest表示最新区块 | ||
ToBlock string `json:"toBlock"` // 订阅结束区块 latest表示最新区块 | ||
Addresses []string `json:"addresses"` // 订阅的合约地址 | ||
Topics []string `json:"topics"` // 订阅的事件 | ||
GroupID string `json:"groupID"` | ||
FilterID string `json:"filterID"` | ||
} | ||
``` | ||
|
||
|
||
#### topic计算 | ||
|
||
Go语言可以使用`github.com/ethereum/go-ethereum/common`包根据合约中定义的事件进行Topic的计算: | ||
|
||
```go | ||
topic = common.BytesToHash(crypto.Keccak256([]byte("testEventLog(address,string,uint256)"))).Hex() | ||
``` | ||
|
||
*需要注意的是uint及int为uint256及int256的别名,即使合约中使用的类型为uint及int,在topic的计算中仍需使用uint256与int256* | ||
|
||
## 4. 示例 | ||
|
||
这里以一个测试事件的合约`eventTest`为例说明,给出Go SDK获取合约事件推送的配置过程。 | ||
|
||
```solidity | ||
pragma solidity >=0.4.24 <0.6.11; | ||
contract eventTest { | ||
event testEventLog(address bidder,string message,uint value); | ||
function testEvent() public { // testEvent函数仅用来触发一个事件 | ||
// 触发一个事件 | ||
emit testEventLog(msg.sender,"successfully subscribe",3); | ||
} | ||
} | ||
``` | ||
|
||
在Go SDK中给出了[`subscriber.go`](https://github.com/FISCO-BCOS/go-sdk/blob/master/examples/eventLog/sub/subscriber.go)合约样例 | ||
|
||
*需要注意的是Topic及Address不可置空,置空将无法接收到相应事件。* | ||
|
||
```go | ||
func main() { | ||
...... // 省略 | ||
endpoint := "127.0.0.1:20200" // 部署的Fisco链的ip:channel_port | ||
privateKey, _ := hex.DecodeString("145e247e170ba3afd6ae97e88f00dbc976c2345d511b0f6713355d19d8b80b58") //用户私钥 | ||
config := &conf.Config{IsHTTP: false, ChainID: 1, CAFile: "ca.crt", Key: "sdk.key", Cert: "sdk.crt", | ||
IsSMCrypto: false, GroupID: 1, PrivateKey: privateKey, NodeURL: endpoint} | ||
var c *client.Client | ||
var err error | ||
...... // 省略 | ||
var eventLogParams types.EventLogParams | ||
eventLogParams.FromBlock = "1" | ||
eventLogParams.ToBlock = "latest" // 当前配置订阅了从创世区块至最新区块的链上所有事件 | ||
eventLogParams.GroupID = "1" | ||
var topics = make([]string, 1) | ||
topics[0] = common.BytesToHash(crypto.Keccak256([]byte("testEventLog(address,string,uint256)"))).Hex() // 事件的Topic计算,注意使用uint256及int256替代uint及int | ||
eventLogParams.Topics = topics | ||
var addresses = make([]string, 1) | ||
addresses[0] = "0xd2cf82e18f3d2c5cae0de87d29994be622f3fdd3" // 所订阅事件对应的合约地址 | ||
|
||
eventLogParams.Addresses = addresses | ||
timeout := 30 * time.Second | ||
queryTicker := time.NewTicker(timeout) | ||
defer queryTicker.Stop() | ||
done := make(chan bool) | ||
err = c.SubscribeEventLogs(eventLogParams, func(status int, logs []types.Log) { // 调用SubscribeEventLogs函数订阅事件 | ||
...... // 省略 | ||
}) | ||
...... // 省略 | ||
} | ||
``` | ||
|
||
|
||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -25,4 +25,5 @@ Go SDK | |
api.md | ||
console.md | ||
contractExamples.md | ||
amopExamples.md | ||
amopExamples.md | ||
event_sub.md |