status-go/services/rpcfilters/latest_logs.go

71 lines
1.8 KiB
Go

package rpcfilters
import (
"context"
"math/big"
"time"
ethereum "github.com/ethereum/go-ethereum"
"github.com/ethereum/go-ethereum/common/hexutil"
"github.com/ethereum/go-ethereum/core/types"
"github.com/ethereum/go-ethereum/log"
"github.com/ethereum/go-ethereum/rpc"
)
// ContextCaller provides CallContext method as ethereums rpc.Client.
type ContextCaller interface {
CallContext(ctx context.Context, result interface{}, method string, args ...interface{}) error
}
func pollLogs(client ContextCaller, f *logsFilter, timeout, period time.Duration) {
query := func() {
ctx, cancel := context.WithTimeout(f.ctx, timeout)
defer cancel()
logs, err := getLogs(ctx, client, f.criteria())
if err != nil {
log.Error("Error fetch logs", "criteria", f.crit, "error", err)
return
}
if err := f.add(logs); err != nil {
log.Error("Error adding logs", "logs", logs, "error", err)
}
}
query()
latest := time.NewTicker(period)
defer latest.Stop()
for {
select {
case <-latest.C:
query()
case <-f.done:
log.Debug("Filter was stopped", "ID", f.id, "crit", f.crit)
return
}
}
}
func getLogs(ctx context.Context, client ContextCaller, crit ethereum.FilterQuery) (rst []types.Log, err error) {
return rst, client.CallContext(ctx, &rst, "eth_getLogs", toFilterArg(crit))
}
func toFilterArg(q ethereum.FilterQuery) interface{} {
arg := map[string]interface{}{
"fromBlock": toBlockNumArg(q.FromBlock),
"toBlock": toBlockNumArg(q.ToBlock),
"address": q.Addresses,
"topics": q.Topics,
}
if q.FromBlock == nil {
arg["fromBlock"] = "0x0"
}
return arg
}
func toBlockNumArg(number *big.Int) string {
if number == nil || number.Int64() == rpc.LatestBlockNumber.Int64() {
return "latest"
} else if number.Int64() == rpc.PendingBlockNumber.Int64() {
return "pending"
}
return hexutil.EncodeBig(number)
}