erigon-pulse/cmd/rpcdaemon/commands/trace_api.go
Giulio rebuffo 33316fec6a
Implemented trace_filter (#998)
* added message for --rpc

* more messages for other unused flags

* implemented trace_filter

* lint

* cleanup

* cleanup
2020-08-29 16:50:24 +01:00

227 lines
6.3 KiB
Go

package commands
import (
"bytes"
"context"
"fmt"
"sort"
"github.com/ledgerwatch/turbo-geth/common"
"github.com/ledgerwatch/turbo-geth/common/dbutils"
"github.com/ledgerwatch/turbo-geth/common/hexutil"
"github.com/ledgerwatch/turbo-geth/core/rawdb"
"github.com/ledgerwatch/turbo-geth/eth"
"github.com/ledgerwatch/turbo-geth/ethdb"
"github.com/ledgerwatch/turbo-geth/turbo/adapter"
"github.com/ledgerwatch/turbo-geth/turbo/transactions"
)
// TraceFilterRequest represents the arguments for trace_filter.
type TraceFilterRequest struct {
FromBlock *hexutil.Uint64 `json:"fromBlock"`
ToBlock *hexutil.Uint64 `json:"toBlock"`
FromAddress []*common.Address `json:"fromAddress"`
ToAddress []*common.Address `json:"toAddress"`
After *uint64 `json:"after"`
Count *uint64 `json:"count"`
}
// TraceAPI
type TraceAPI interface {
Filter(ctx context.Context, req TraceFilterRequest) ([]interface{}, error)
}
// TraceAPIImpl is implementation of the TraceAPI interface based on remote Db access
type TraceAPIImpl struct {
db ethdb.KV
dbReader ethdb.Getter
maxTraces uint64
}
// NewPrivateDebugAPI returns PrivateDebugAPIImpl instance
func NewTraceAPI(db ethdb.KV, dbReader ethdb.Getter, maxTraces uint64) *TraceAPIImpl {
return &TraceAPIImpl{
db: db,
dbReader: dbReader,
maxTraces: maxTraces,
}
}
func retrieveHistory(tx ethdb.Tx, addr *common.Address, fromBlock uint64, toBlock uint64) ([]uint64, error) {
addrBytes := addr.Bytes()
ca := tx.Cursor(dbutils.AccountsHistoryBucket).Prefix(addrBytes)
var blockNumbers []uint64
for k, v, err := ca.First(); k != nil; k, v, err = ca.Next() {
if err != nil {
return nil, err
}
numbers, _, err := dbutils.WrapHistoryIndex(v).Decode()
if err != nil {
return nil, err
}
if numbers[0] > toBlock {
break
}
if numbers[len(numbers)-1] < fromBlock {
continue
}
blockNumbers = append(blockNumbers, numbers...)
}
// Remove dublicates
return blockNumbers, nil
}
func isAddressInFilter(addr *common.Address, filter []*common.Address) bool {
if filter == nil {
return true
}
i := sort.Search(len(filter), func(i int) bool {
return bytes.Equal(filter[i].Bytes(), addr.Bytes())
})
return i != len(filter)
}
// Filter Implements trace_filter
func (api *TraceAPIImpl) Filter(ctx context.Context, req TraceFilterRequest) ([]interface{}, error) {
var filteredTransactionsHash []common.Hash
resp := []interface{}{}
var maxTracesCount uint64
sort.Slice(req.FromAddress, func(i int, j int) bool {
return bytes.Compare(req.FromAddress[i].Bytes(), req.FromAddress[j].Bytes()) == -1
})
sort.Slice(req.ToAddress, func(i int, j int) bool {
return bytes.Compare(req.ToAddress[i].Bytes(), req.ToAddress[j].Bytes()) == -1
})
var fromBlock uint64
var toBlock uint64
if req.FromBlock == nil {
fromBlock = 0
} else {
fromBlock = uint64(*req.FromBlock)
}
if req.ToBlock == nil {
headNumber := rawdb.ReadHeaderNumber(api.dbReader, rawdb.ReadHeadHeaderHash(api.dbReader))
toBlock = *headNumber
} else {
toBlock = uint64(*req.ToBlock)
}
if fromBlock > toBlock {
return nil, fmt.Errorf("invalid parameters: toBlock must be greater than fromBlock")
}
if req.Count == nil {
maxTracesCount = api.maxTraces
} else {
maxTracesCount = *req.Count
if req.After != nil {
maxTracesCount += *req.After
}
if maxTracesCount > api.maxTraces {
maxTracesCount = api.maxTraces
}
}
if err := api.db.View(ctx, func(tx ethdb.Tx) error {
if req.FromAddress != nil || req.ToAddress != nil { // use address history index to retrieve matching transactions
var historyFilter []*common.Address
isFromAddress := req.FromAddress != nil
if isFromAddress {
historyFilter = req.FromAddress
} else {
historyFilter = req.ToAddress
}
for _, addr := range historyFilter {
addrBytes := addr.Bytes()
blockNumbers, err := retrieveHistory(tx, addr, fromBlock, toBlock)
if err != nil {
return err
}
for _, num := range blockNumbers {
block := rawdb.ReadBlockByNumber(api.dbReader, num)
senders := rawdb.ReadSenders(api.dbReader, block.Hash(), num)
txs := block.Transactions()
for i, tx := range txs {
if uint64(len(filteredTransactionsHash)) == maxTracesCount {
if uint64(len(filteredTransactionsHash)) == api.maxTraces {
return fmt.Errorf("too many traces found")
}
return nil
}
var to *common.Address
if tx.To() == nil {
to = &common.Address{}
} else {
to = tx.To()
}
if isFromAddress {
if !isAddressInFilter(to, req.ToAddress) {
continue
}
if bytes.Equal(senders[i].Bytes(), addrBytes) {
filteredTransactionsHash = append(filteredTransactionsHash, tx.Hash())
}
} else if bytes.Equal(to.Bytes(), addrBytes) {
filteredTransactionsHash = append(filteredTransactionsHash, tx.Hash())
}
}
}
}
} else if req.FromBlock != nil || req.ToBlock != nil { // iterate over blocks
for blockNum := fromBlock; blockNum < toBlock+1; blockNum++ {
block := rawdb.ReadBlockByNumber(api.dbReader, blockNum)
blockTransactions := block.Transactions()
for _, tx := range blockTransactions {
if uint64(len(filteredTransactionsHash)) == maxTracesCount {
if uint64(len(filteredTransactionsHash)) == api.maxTraces {
return fmt.Errorf("too many traces found")
}
return nil
}
filteredTransactionsHash = append(filteredTransactionsHash, tx.Hash())
}
}
} else {
return fmt.Errorf("invalid parameters")
}
return nil
}); err != nil {
return nil, err
}
getter := adapter.NewBlockGetter(api.dbReader)
chainContext := adapter.NewChainContext(api.dbReader)
genesisHash := rawdb.ReadBlockByNumber(api.dbReader, 0).Hash()
chainConfig := rawdb.ReadChainConfig(api.dbReader, genesisHash)
traceType := "callTracer"
for _, hash := range filteredTransactionsHash {
_, blockHash, _, txIndex := rawdb.ReadTransaction(api.dbReader, hash)
msg, vmctx, ibs, _, err := transactions.ComputeTxEnv(ctx, getter, chainConfig, chainContext, api.db, blockHash, txIndex)
if err != nil {
return nil, err
}
trace, err := transactions.TraceTx(ctx, msg, vmctx, ibs, &eth.TraceConfig{Tracer: &traceType})
if err != nil {
return nil, err
}
resp = append(resp, trace)
}
return resp, nil
}