erigon-pulse/cmd/devnettest/services/event.go

131 lines
3.2 KiB
Go
Raw Normal View History

package services
import (
"context"
"fmt"
2022-08-10 12:04:13 +00:00
"github.com/ledgerwatch/erigon/cmd/devnettest/utils"
"github.com/ledgerwatch/erigon/common"
"github.com/ledgerwatch/erigon/rpc"
)
const numberOfIterations = 128
var ch = make(chan interface{})
// subscribe connects to a websocket and returns the subscription handler and a channel buffer
func subscribe(client *rpc.Client, method string, args ...interface{}) (*rpc.ClientSubscription, error) {
var (
namespace string
subMethod string
splitErr error
)
namespace, subMethod, splitErr = utils.NamespaceAndSubMethodFromMethod(method)
if splitErr != nil {
return nil, fmt.Errorf("cannot get namespace and submethod from method: %v", splitErr)
}
var arr = []interface{}{subMethod}
arr = append(arr, args...)
sub, err := client.Subscribe(context.Background(), namespace, ch, arr...)
if err != nil {
return nil, fmt.Errorf("client failed to subscribe: %v", err)
}
return sub, nil
}
// subscribeToNewHeadsAndSearch makes a ws subscription for eth_newHeads and searches each new header for the tx hash
func subscribeToNewHeadsAndSearch(client *rpc.Client, method string, hash common.Hash) (uint64, error) {
sub, err := subscribe(client, method)
if err != nil {
return uint64(0), fmt.Errorf("error subscribing to newHeads: %v", err)
}
defer unSubscribe(sub)
var (
blockCount int
blockN uint64
)
mark:
for {
select {
case v := <-ch:
blockCount++
blockNumber := v.(map[string]interface{})["number"]
num, foundTx, err := blockHasHash(client, hash, blockNumber.(string))
if err != nil {
return uint64(0), fmt.Errorf("could not verify if current block contains the tx hash: %v", err)
}
if foundTx || blockCount == numberOfIterations {
blockN = num
break mark
}
case err := <-sub.Err():
return uint64(0), fmt.Errorf("subscription error from client: %v", err)
}
}
return blockN, nil
}
// Logs dials a websocket connection and listens for log events by calling subscribeToLogs
func Logs(addresses, topics []string) error {
client, clientErr := rpc.DialWebsocket(context.Background(), "ws://127.0.0.1:8545", "")
if clientErr != nil {
return fmt.Errorf("failed to dial websocket: %v", clientErr)
}
if err := subscribeToLogs(client, "eth_logs", addresses, topics); err != nil {
return fmt.Errorf("failed to subscribe to logs: %v", err)
}
return nil
}
// subscribeToLogs makes a ws subscription for eth_subscribeLogs
func subscribeToLogs(client *rpc.Client, method string, addresses []string, topics []string) error {
params := map[string][]string{
"address": addresses,
"topics": topics,
}
_, err := subscribe(client, method, params)
if err != nil {
return fmt.Errorf("error subscribing to logs: %v", err)
}
//defer unSubscribe(sub)
//var count int
//ForLoop:
// for {
// select {
// case v := <-ch:
// count++
// _map := v.(map[string]interface{})
// for k, val := range _map {
// fmt.Printf("%s: %+v, ", k, val)
// }
// fmt.Println()
// fmt.Println()
// if count == numberOfIterations {
// break ForLoop
// }
// case err := <-sub.Err():
// return fmt.Errorf("subscription error from client: %v", err)
// }
// }
return nil
}
func unSubscribe(sub *rpc.ClientSubscription) {
sub.Unsubscribe()
for len(ch) > 0 {
<-ch
}
}