-
Notifications
You must be signed in to change notification settings - Fork 18
/
Copy pathETH_RPC_Requester.go
370 lines (335 loc) · 12.8 KB
/
ETH_RPC_Requester.go
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
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
package main
import (
"errors"
"eth-relay/model"
"eth-relay/tool"
"fmt"
"github.com/ethereum/go-ethereum/accounts/keystore"
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/rlp"
"github.com/ethereum/go-ethereum/rpc"
"github.com/ethereum/go-ethereum/common/hexutil"
"math/big"
)
type ETHRPCRequester struct {
nonceManager *NonceManager // nonce 管理者实例
client *ETHRPCClient // 小写字母开头,私有的 rpc 客户端实例
}
func NewETHRPCRequester(nodeUrl string) *ETHRPCRequester {
requester := ÐRPCRequester{}
// 实例化 nonce 管理器
requester.nonceManager = NewNonceManager()
// 实例化 rpc 客户端
requester.client = NewETHRPCClient(nodeUrl)
return requester
}
// 根据交易的 hash 值获取对应交易的信息
func (r *ETHRPCRequester) GetTransactionByHash(txHash string) (model.Transaction, error) {
methodName := "eth_getTransactionByHash"
result := model.Transaction{}
// 下面 call 的 result 传入的是 model.Transaction 结构体的引用,
// 这样内部所设置的值在函数执行完之后才能依然有效果
err := r.client.GetRpc().Call(&result, methodName, txHash)
return result, err
}
// 根据交易 hash 字符串数组批量获取对应的交易信息
func (r *ETHRPCRequester) GetTransactions(txHashs []string) ([]*model.Transaction, error) {
name := "eth_getTransactionByHash"
// 结果数组存储的是每个请求的结果指针,也就是引用
rets := []*model.Transaction{}
// 获取 hash 数组的长度,方便在循环中逐个实例化 BatchElem
size := len(txHashs)
reqs := []rpc.BatchElem{}
for i := 0; i < size; i++ {
ret := model.Transaction{}
// 实例化每个 BatchElem
req := rpc.BatchElem{
Method: name,
Args: []interface{}{txHashs[i]},
// &ret 传入单个请求的结果引用,这样是保证它在函数内部被修改值后,回到函数外来,值仍有效
Result: &ret,
}
reqs = append(reqs, req) // 将每个 BatchElem 添加到 BatchElem 数组
rets = append(rets, &ret) // 每个请求的结果引用添加到结果数组中
}
err := r.client.GetRpc().BatchCall(reqs) // 传入 BatchElem 数组,发起批量请求
return rets, err
}
// 单条查询:根据以太坊地址,查询以太坊 eth 的余额
func (r *ETHRPCRequester) GetETHBalance(address string) (string, error) {
name := "eth_getBalance"
result := ""
// 对应文档,第一个参数就是要被查询的以太坊地址
// 第二个参数就是 latest
err := r.client.GetRpc().Call(&result, name, address, "latest")
if err != nil {
return "", err
}
if result == "" {
return "", errors.New("eth balance is null")
}
// 因为查询所返回的结果是一个16进制的字符串,
// 为了方便阅读,我们在下面使用 go 的大数处理将其转为 10 进制,
// 并防止数位溢出
ten, _ := new(big.Int).SetString(result[2:], 16)
return ten.String(), nil
}
// 批量查询:根据以太坊地址数组,查询以太坊 eth 的余额
func (r *ETHRPCRequester) GetETHBalances(addresss []string) ([]string, error) {
name := "eth_getBalance"
// 结果数组存储的是每个请求的结果指针,也就是引用
rets := []*string{}
// 获取 addresss 数组的长度,方便在循环中逐个实例化 BatchElem
size := len(addresss)
reqs := []rpc.BatchElem{}
for i := 0; i < size; i++ {
ret := ""
// 实例化每个 BatchElem
req := rpc.BatchElem{
Method: name,
Args: []interface{}{addresss[i], "latest"},
// &ret 传入单个请求的结果引用,这样是保证它在函数内部被修改值后,回到函数外来,值仍有效
Result: &ret,
}
reqs = append(reqs, req) // 将每个 BatchElem 添加到 BatchElem 数组
rets = append(rets, &ret) // 每个请求的结果引用添加到结果数组中
}
err := r.client.GetRpc().BatchCall(reqs) // 传入 BatchElem 数组,发起批量请求
if err != nil {
return nil, err
}
// 查询每个请求有没有错误
for _, req := range reqs {
if req.Error != nil {
return nil, req.Error // 返回错误
}
}
finalRet := []string{}
for _, item := range rets {
ten, _ := new(big.Int).SetString((*item)[2:], 16)
finalRet = append(finalRet, ten.String())
}
return finalRet, err
}
// ERC20BalanceRpcReq 是查询 ERC20 代币的参数集合结构体
type ERC20BalanceRpcReq struct {
ContractAddress string // 合约的以太坊地址
UserAddress string // 用户的以太坊地址
ContractDecimal int // 合约所对应代币的数位
}
// 批量查询:根据以太坊地址数组,查询 ERC20 代币的余额
func (r *ETHRPCRequester) GetERC20Balances(paramArr []ERC20BalanceRpcReq) ([]string, error) {
name := "eth_call"
methodId := "0x70a08231" // 这个就是 balanceOf 的 methodId
// 结果数组存储的是每个请求的结果指针,也就是引用
rets := []*string{}
// 获取参数数组的长度,方便在循环中逐个实例化 BatchElem
size := len(paramArr)
reqs := []rpc.BatchElem{}
for i := 0; i < size; i++ {
ret := ""
arg := &model.CallArg{}
userAddress := paramArr[i].UserAddress
// 下面是针对访问 balanceOf 时的必须参数,查询余额是不需要油费的,但是发现一些版本的节点又需要指定这个参数,所以下面还是指定一个
arg.Gas = hexutil.EncodeUint64(300000)
arg.To = common.HexToAddress(paramArr[i].ContractAddress)
// data 参数的组合格式见 “交易参数的说明” 小节中的详解
arg.Data = methodId + "000000000000000000000000" + userAddress[2:]
// 实例化每个 BatchElem
req := rpc.BatchElem{
Method: name,
Args: []interface{}{arg, "latest"},
// &ret 传入单个请求的结果引用,这样是保证它在函数内部被修改值后,回到函数外来,值仍有效
Result: &ret,
}
reqs = append(reqs, req) // 将每个 BatchElem 添加到 BatchElem 数组
rets = append(rets, &ret) // 每个请求的结果引用添加到结果数组中
}
err := r.client.GetRpc().BatchCall(reqs) // 传入 BatchElem 数组,发起批量请求
if err != nil {
return nil, err
}
// 查询每个请求有没有错误
for _, req := range reqs {
if req.Error != nil {
return nil, req.Error // 返回错误
}
}
finalRet := []string{}
for _, item := range rets {
if *item == "" {
continue
}
ten, _ := new(big.Int).SetString((*item)[2:], 16)
finalRet = append(finalRet, ten.String())
}
return finalRet, err
}
// 创建以太坊钱包
func (r *ETHRPCRequester) CreateETHWallet(password string) (string, error) {
if password == "" {
return "", errors.New("password cant empty")
}
if len(password) < 6 {
return "", errors.New("password's len must more than 6 words")
}
keydir := "./keystores" // 它是用来存储所创建了的钱包的 keystore 文件的目录
// StandardScryptN 是Scrypt加密算法的标准N参数
// StandardScryptP 是Scrypt加密算法的标准P参数
ks := keystore.NewKeyStore(keydir, keystore.StandardScryptN, keystore.StandardScryptP)
wallet, err := ks.NewAccount(password) // 传入密码,创建钱包
if err != nil {
return "0x", err
}
return wallet.Address.String(), nil
}
// 发送交易,根据入参 transaction 的不同变量设置,达到发送不同种类的交易
func (r *ETHRPCRequester) SendTransaction(address string, transaction *types.Transaction) (string, error) {
// 签名交易数据
signTx, err := tool.SignETHTransaction(address, transaction)
if err != nil {
return "", fmt.Errorf("签名失败! %s", err.Error())
}
// rlp 序列化
txRlpData, err := rlp.EncodeToBytes(signTx)
if nil != err {
return "", fmt.Errorf("rlp 序列化失败! %s", err.Error())
}
// 下面调用以太坊的 rpc 接口
txHash := ""
methodName := "eth_sendRawTransaction"
err = r.client.client.Call(&txHash, methodName, common.ToHex(txRlpData))
if err != nil {
return "", fmt.Errorf("发送交易失败! %s", err.Error())
}
return txHash, nil // 返回交易 hash
}
// 获取地址的 nonce 值
func (r *ETHRPCRequester) GetNonce(address string) (uint64, error) {
methodName := "eth_getTransactionCount"
nonce := ""
// 因为我们要查询最新的,根据基于 etTransactionCount 情况下的区块号关系,选取 pending
err := r.client.client.Call(&nonce, methodName, address, "pending")
if err != nil {
return 0, fmt.Errorf("发送交易失败! %s", err.Error())
}
n, _ := new(big.Int).SetString(nonce[2:], 16) // 采用大数类型将 16 进制的结果转为 10 进制
return n.Uint64(), nil // 返回交易 hash
}
// 发送 ETH 交易,或称转账 ETH
// 参数分别是:交易发起地址,交易接收地址,ETH数量,油费设置
func (r *ETHRPCRequester) SendETHTransaction(fromStr, toStr, valueStr string, gasLimit, gasPrice uint64) (string, error) {
if !common.IsHexAddress(fromStr) || !common.IsHexAddress(toStr) {
return "", errors.New("invalid address")
}
to := common.HexToAddress(toStr) // 将字符串类型的转为 address 类型的
gasPrice_ := new(big.Int).SetUint64(gasPrice)
// value 乘上 10^decimal,得出真实的转账值,ETH 的小数位是 18 位
realV := tool.GetRealDecimalValue(valueStr, 18)
if realV == "" {
return "", errors.New("invalid value")
}
amount, _ := new(big.Int).SetString(realV, 10)
// 获取 nonce
nonce := r.nonceManager.GetNonce(fromStr)
if nonce == nil {
// nonce 不存在,开始访问节点获取
n, err := r.GetNonce(fromStr)
if err != nil {
return "", fmt.Errorf("获取 nonce 失败 %s", err.Error())
}
nonce = new(big.Int).SetUint64(n)
}
// 构建 data,因为 eth 交易转账类型,data 是空,我们设置空字符串即可
data := []byte("")
// 构建交易结构体
transaction := types.NewTransaction(
nonce.Uint64(),
to,
amount,
gasLimit,
gasPrice_,
data)
return r.SendTransaction(fromStr, transaction)
}
// 发送 ERC20 代币交易,或称转账 ERC20 代币
// 参数分别是:
// 交易发起地址,代币合约地址,交易接收地址,代币数量,油费设置,代币的 decimal 值
func (r *ETHRPCRequester) SendERC20Transaction(
fromStr, contact, receiver, valueStr string, gasLimit, gasPrice uint64, decimal int) (string, error) {
if !common.IsHexAddress(fromStr) ||
!common.IsHexAddress(contact) ||
!common.IsHexAddress(receiver) {
return "", errors.New("invalid address")
}
to := common.HexToAddress(contact) // 将合约 contact 字符串类型的转为 address 类型的
gasPrice_ := new(big.Int).SetUint64(gasPrice)
// 结构体中的 value 字段为 0
amount := new(big.Int).SetInt64(0)
// 获取 nonce
nonce := r.nonceManager.GetNonce(fromStr)
if nonce == nil {
// nonce 不存在,开始访问节点获取
n, err := r.GetNonce(fromStr)
if err != nil {
return "", fmt.Errorf("获取 nonce 失败 %s", err.Error())
}
nonce = new(big.Int).SetUint64(n)
}
// 构建 data,真实的 value 转账数值由 data 携带
data := tool.BuildERC20TransferData(valueStr, receiver, decimal)
dataBytes := []byte(data)
// 构建交易结构体
transaction := types.NewTransaction(
nonce.Uint64(),
to,
amount,
gasLimit,
gasPrice_,
dataBytes)
return r.SendTransaction(fromStr, transaction)
}
// 获取以太坊最新生成区块的区块号
func (r *ETHRPCRequester) GetLatestBlockNumber() (*big.Int, error) {
methodName := "eth_blockNumber"
number := ""
// eth_blockNumber 不需要参数
err := r.client.client.Call(&number, methodName)
if err != nil {
return nil, fmt.Errorf("获取最新区块号失败! %s", err.Error())
}
ten,_ := new(big.Int).SetString(number[2:],16)
return ten, nil
}
// 根据区块号,获取区块信息
func (r *ETHRPCRequester) GetBlockInfoByNumber(blockNumber *big.Int) (*model.FullBlock, error) {
number := fmt.Sprintf("%#x", blockNumber)
methodName := "eth_getBlockByNumber"
fullBlock := model.FullBlock{}
// eth_getBlockByNumber 的第二个参数:
// 如果是 true,则返回完整的区块信息,false 则 transaction 部分只返回交易hash数组
err := r.client.client.Call(&fullBlock, methodName, number, true)
if err != nil {
return nil, fmt.Errorf("get block info failed! %s", err.Error())
}
if fullBlock.Number == "" {
return nil, fmt.Errorf("block info is empty %s",blockNumber.String())
}
return &fullBlock, nil
}
// 根据区块 hash,获取区块信息
func (r *ETHRPCRequester) GetBlockInfoByHash(blockHash string) (*model.FullBlock, error) {
methodName := "eth_getBlockByHash"
fullBlock := model.FullBlock{}
// eth_getBlockByHash 的第二个参数:
// 如果是 true,则返回完整的区块信息,false 则 transaction 部分只返回交易hash数组
err := r.client.client.Call(&fullBlock, methodName, blockHash, true)
if err != nil {
return nil, fmt.Errorf("get block info failed! %s", err.Error())
}
if fullBlock.Number == "" {
return nil, fmt.Errorf("block info is empty %s",blockHash)
}
return &fullBlock, nil
}