package commands import ( "context" "errors" "fmt" "math/big" "sync" "github.com/holiman/uint256" "github.com/ledgerwatch/erigon-lib/kv" "github.com/ledgerwatch/erigon/cmd/rpcdaemon/commands" "github.com/ledgerwatch/erigon/common" "github.com/ledgerwatch/erigon/common/hexutil" "github.com/ledgerwatch/erigon/consensus/ethash" "github.com/ledgerwatch/erigon/core" "github.com/ledgerwatch/erigon/core/rawdb" "github.com/ledgerwatch/erigon/core/types" "github.com/ledgerwatch/erigon/core/vm" "github.com/ledgerwatch/erigon/params" "github.com/ledgerwatch/erigon/rpc" "github.com/ledgerwatch/erigon/turbo/rpchelper" "github.com/ledgerwatch/erigon/turbo/transactions" "github.com/ledgerwatch/log/v3" "github.com/wmitsuda/otterscan/erigon_internal/ethapi" ) // API_LEVEL Must be incremented every time new additions are made const API_LEVEL = 8 type TransactionsWithReceipts struct { Txs []*commands.RPCTransaction `json:"txs"` Receipts []map[string]interface{} `json:"receipts"` FirstPage bool `json:"firstPage"` LastPage bool `json:"lastPage"` } type OtterscanAPI interface { GetApiLevel() uint8 GetInternalOperations(ctx context.Context, hash common.Hash) ([]*InternalOperation, error) SearchTransactionsBefore(ctx context.Context, addr common.Address, blockNum uint64, pageSize uint16) (*TransactionsWithReceipts, error) SearchTransactionsAfter(ctx context.Context, addr common.Address, blockNum uint64, pageSize uint16) (*TransactionsWithReceipts, error) GetBlockDetails(ctx context.Context, number rpc.BlockNumber) (map[string]interface{}, error) GetBlockDetailsByHash(ctx context.Context, hash common.Hash) (map[string]interface{}, error) GetBlockTransactions(ctx context.Context, number rpc.BlockNumber, pageNumber uint8, pageSize uint8) (map[string]interface{}, error) HasCode(ctx context.Context, address common.Address, blockNrOrHash rpc.BlockNumberOrHash) (bool, error) TraceTransaction(ctx context.Context, hash common.Hash) ([]*TraceEntry, error) GetTransactionError(ctx context.Context, hash common.Hash) (hexutil.Bytes, error) GetTransactionBySenderAndNonce(ctx context.Context, addr common.Address, nonce uint64) (*common.Hash, error) GetContractCreator(ctx context.Context, addr common.Address) (*ContractCreatorData, error) } type OtterscanAPIImpl struct { *BaseAPIUtils db kv.RoDB } func NewOtterscanAPI(base *BaseAPIUtils, db kv.RoDB) *OtterscanAPIImpl { return &OtterscanAPIImpl{ BaseAPIUtils: base, db: db, } } func (api *OtterscanAPIImpl) GetApiLevel() uint8 { return API_LEVEL } // TODO: dedup from eth_txs.go#GetTransactionByHash func (api *OtterscanAPIImpl) getTransactionByHash(ctx context.Context, tx kv.Tx, hash common.Hash) (types.Transaction, *types.Block, common.Hash, uint64, uint64, error) { // https://infura.io/docs/ethereum/json-rpc/eth-getTransactionByHash blockNum, ok, err := api.txnLookup(ctx, tx, hash) if err != nil { return nil, nil, common.Hash{}, 0, 0, err } if !ok { return nil, nil, common.Hash{}, 0, 0, nil } block, err := api.blockByNumberWithSenders(tx, blockNum) if err != nil { return nil, nil, common.Hash{}, 0, 0, err } if block == nil { return nil, nil, common.Hash{}, 0, 0, nil } blockHash := block.Hash() var txnIndex uint64 var txn types.Transaction for i, transaction := range block.Transactions() { if transaction.Hash() == hash { txn = transaction txnIndex = uint64(i) break } } // Add GasPrice for the DynamicFeeTransaction // var baseFee *big.Int // if chainConfig.IsLondon(blockNum) && blockHash != (common.Hash{}) { // baseFee = block.BaseFee() // } // if no transaction was found then we return nil if txn == nil { return nil, nil, common.Hash{}, 0, 0, nil } return txn, block, blockHash, blockNum, txnIndex, nil } func (api *OtterscanAPIImpl) runTracer(ctx context.Context, tx kv.Tx, hash common.Hash, tracer vm.Tracer) (*core.ExecutionResult, error) { txn, block, blockHash, _, txIndex, err := api.getTransactionByHash(ctx, tx, hash) if err != nil { return nil, err } if txn == nil { return nil, fmt.Errorf("transaction %#x not found", hash) } chainConfig, err := api.chainConfig(tx) if err != nil { return nil, err } getHeader := func(hash common.Hash, number uint64) *types.Header { return rawdb.ReadHeader(tx, hash, number) } msg, blockCtx, txCtx, ibs, _, err := transactions.ComputeTxEnv(ctx, block, chainConfig, getHeader, ethash.NewFaker(), tx, blockHash, txIndex) if err != nil { return nil, err } var vmConfig vm.Config if tracer == nil { vmConfig = vm.Config{} } else { vmConfig = vm.Config{Debug: true, Tracer: tracer} } vmenv := vm.NewEVM(blockCtx, txCtx, ibs, chainConfig, vmConfig) result, err := core.ApplyMessage(vmenv, msg, new(core.GasPool).AddGas(msg.Gas()), true, false /* gasBailout */) if err != nil { return nil, fmt.Errorf("tracing failed: %v", err) } return result, nil } func (api *OtterscanAPIImpl) GetInternalOperations(ctx context.Context, hash common.Hash) ([]*InternalOperation, error) { tx, err := api.db.BeginRo(ctx) if err != nil { return nil, err } defer tx.Rollback() tracer := NewOperationsTracer(ctx) if _, err := api.runTracer(ctx, tx, hash, tracer); err != nil { return nil, err } return tracer.Results, nil } // Search transactions that touch a certain address. // // It searches back a certain block (excluding); the results are sorted descending. // // The pageSize indicates how many txs may be returned. If there are less txs than pageSize, // they are just returned. But it may return a little more than pageSize if there are more txs // than the necessary to fill pageSize in the last found block, i.e., let's say you want pageSize == 25, // you already found 24 txs, the next block contains 4 matches, then this function will return 28 txs. func (api *OtterscanAPIImpl) SearchTransactionsBefore(ctx context.Context, addr common.Address, blockNum uint64, pageSize uint16) (*TransactionsWithReceipts, error) { dbtx, err := api.db.BeginRo(ctx) if err != nil { return nil, err } defer dbtx.Rollback() log.Info("got cursor") callFromCursor, err := dbtx.Cursor(kv.CallFromIndex) if err != nil { return nil, err } defer callFromCursor.Close() log.Info("call from cur") callToCursor, err := dbtx.Cursor(kv.CallToIndex) if err != nil { return nil, err } defer callToCursor.Close() log.Info("cur to call") chainConfig, err := api.chainConfig(dbtx) if err != nil { return nil, err } isFirstPage := false if blockNum == 0 { isFirstPage = true } else { // Internal search code considers blockNum [including], so adjust the value blockNum-- } // Initialize search cursors at the first shard >= desired block number callFromProvider := NewCallCursorBackwardBlockProvider(callFromCursor, addr, blockNum) callToProvider := NewCallCursorBackwardBlockProvider(callToCursor, addr, blockNum) callFromToProvider := newCallFromToBlockProvider(false, callFromProvider, callToProvider) txs := make([]*commands.RPCTransaction, 0, pageSize) receipts := make([]map[string]interface{}, 0, pageSize) resultCount := uint16(0) hasMore := true for { if resultCount >= pageSize || !hasMore { break } var results []*TransactionsWithReceipts results, hasMore, err = api.traceBlocks(ctx, addr, chainConfig, pageSize, resultCount, callFromToProvider) if err != nil { return nil, err } for _, r := range results { if r == nil { return nil, errors.New("internal error during search tracing") } for i := len(r.Txs) - 1; i >= 0; i-- { txs = append(txs, r.Txs[i]) } for i := len(r.Receipts) - 1; i >= 0; i-- { receipts = append(receipts, r.Receipts[i]) } resultCount += uint16(len(r.Txs)) if resultCount >= pageSize { break } } } return &TransactionsWithReceipts{txs, receipts, isFirstPage, !hasMore}, nil } // Search transactions that touch a certain address. // // It searches forward a certain block (excluding); the results are sorted descending. // // The pageSize indicates how many txs may be returned. If there are less txs than pageSize, // they are just returned. But it may return a little more than pageSize if there are more txs // than the necessary to fill pageSize in the last found block, i.e., let's say you want pageSize == 25, // you already found 24 txs, the next block contains 4 matches, then this function will return 28 txs. func (api *OtterscanAPIImpl) SearchTransactionsAfter(ctx context.Context, addr common.Address, blockNum uint64, pageSize uint16) (*TransactionsWithReceipts, error) { dbtx, err := api.db.BeginRo(ctx) if err != nil { return nil, err } defer dbtx.Rollback() callFromCursor, err := dbtx.Cursor(kv.CallFromIndex) if err != nil { return nil, err } defer callFromCursor.Close() callToCursor, err := dbtx.Cursor(kv.CallToIndex) if err != nil { return nil, err } defer callToCursor.Close() chainConfig, err := api.chainConfig(dbtx) if err != nil { return nil, err } isLastPage := false if blockNum == 0 { isLastPage = true } else { // Internal search code considers blockNum [including], so adjust the value blockNum++ } // Initialize search cursors at the first shard >= desired block number callFromProvider := NewCallCursorForwardBlockProvider(callFromCursor, addr, blockNum) callToProvider := NewCallCursorForwardBlockProvider(callToCursor, addr, blockNum) callFromToProvider := newCallFromToBlockProvider(true, callFromProvider, callToProvider) txs := make([]*commands.RPCTransaction, 0, pageSize) receipts := make([]map[string]interface{}, 0, pageSize) resultCount := uint16(0) hasMore := true for { if resultCount >= pageSize || !hasMore { break } var results []*TransactionsWithReceipts results, hasMore, err = api.traceBlocks(ctx, addr, chainConfig, pageSize, resultCount, callFromToProvider) if err != nil { return nil, err } for _, r := range results { if r == nil { return nil, errors.New("internal error during search tracing") } txs = append(txs, r.Txs...) receipts = append(receipts, r.Receipts...) resultCount += uint16(len(r.Txs)) if resultCount >= pageSize { break } } } // Reverse results lentxs := len(txs) for i := 0; i < lentxs/2; i++ { txs[i], txs[lentxs-1-i] = txs[lentxs-1-i], txs[i] receipts[i], receipts[lentxs-1-i] = receipts[lentxs-1-i], receipts[i] } return &TransactionsWithReceipts{txs, receipts, !hasMore, isLastPage}, nil } func (api *OtterscanAPIImpl) traceBlocks(ctx context.Context, addr common.Address, chainConfig *params.ChainConfig, pageSize, resultCount uint16, callFromToProvider BlockProvider) ([]*TransactionsWithReceipts, bool, error) { var wg sync.WaitGroup // Estimate the common case of user address having at most 1 interaction/block and // trace N := remaining page matches as number of blocks to trace concurrently. // TODO: this is not optimimal for big contract addresses; implement some better heuristics. estBlocksToTrace := pageSize - resultCount results := make([]*TransactionsWithReceipts, estBlocksToTrace) totalBlocksTraced := 0 hasMore := true for i := 0; i < int(estBlocksToTrace); i++ { var nextBlock uint64 var err error nextBlock, hasMore, err = callFromToProvider() if err != nil { return nil, false, err } // TODO: nextBlock == 0 seems redundant with hasMore == false if !hasMore && nextBlock == 0 { break } wg.Add(1) totalBlocksTraced++ go api.searchTraceBlock(ctx, &wg, addr, chainConfig, i, nextBlock, results) } wg.Wait() return results[:totalBlocksTraced], hasMore, nil } func (api *OtterscanAPIImpl) delegateGetBlockByNumber(tx kv.Tx, b *types.Block, number rpc.BlockNumber, inclTx bool) (map[string]interface{}, error) { td, err := rawdb.ReadTd(tx, b.Hash(), b.NumberU64()) if err != nil { return nil, err } response, err := ethapi.RPCMarshalBlock(b, inclTx, inclTx) if !inclTx { delete(response, "transactions") // workaround for https://github.com/ledgerwatch/erigon/issues/4989#issuecomment-1218415666 } response["totalDifficulty"] = (*hexutil.Big)(td) response["transactionCount"] = b.Transactions().Len() if err == nil && number == rpc.PendingBlockNumber { // Pending blocks need to nil out a few fields for _, field := range []string{"hash", "nonce", "miner"} { response[field] = nil } } // Explicitly drop unwanted fields response["logsBloom"] = nil return response, err } // TODO: temporary workaround due to API breakage from watch_the_burn type internalIssuance struct { BlockReward string `json:"blockReward,omitempty"` UncleReward string `json:"uncleReward,omitempty"` Issuance string `json:"issuance,omitempty"` } func (api *OtterscanAPIImpl) delegateIssuance(tx kv.Tx, block *types.Block, chainConfig *params.ChainConfig) (internalIssuance, error) { if chainConfig.Ethash == nil { // Clique for example has no issuance return internalIssuance{}, nil } minerReward, uncleRewards := ethash.AccumulateRewards(chainConfig, block.Header(), block.Uncles()) issuance := minerReward for _, r := range uncleRewards { p := r // avoids warning? issuance.Add(&issuance, &p) } var ret internalIssuance ret.BlockReward = hexutil.EncodeBig(minerReward.ToBig()) ret.Issuance = hexutil.EncodeBig(issuance.ToBig()) issuance.Sub(&issuance, &minerReward) ret.UncleReward = hexutil.EncodeBig(issuance.ToBig()) return ret, nil } func (api *OtterscanAPIImpl) delegateBlockFees(ctx context.Context, tx kv.Tx, block *types.Block, senders []common.Address, chainConfig *params.ChainConfig) (uint64, error) { receipts, err := api.getReceipts(ctx, tx, chainConfig, block, senders) if err != nil { return 0, fmt.Errorf("getReceipts error: %v", err) } fees := uint64(0) for _, receipt := range receipts { txn := block.Transactions()[receipt.TransactionIndex] effectiveGasPrice := uint64(0) if !chainConfig.IsLondon(block.NumberU64()) { effectiveGasPrice = txn.GetPrice().Uint64() } else { baseFee, _ := uint256.FromBig(block.BaseFee()) gasPrice := new(big.Int).Add(block.BaseFee(), txn.GetEffectiveGasTip(baseFee).ToBig()) effectiveGasPrice = gasPrice.Uint64() } fees += effectiveGasPrice * receipt.GasUsed } return fees, nil } func (api *OtterscanAPIImpl) getBlockWithSenders(ctx context.Context, number rpc.BlockNumber, tx kv.Tx) (*types.Block, []common.Address, error) { if number == rpc.PendingBlockNumber { return api.pendingBlock(), nil, nil } n, hash, _, err := rpchelper.GetBlockNumber(rpc.BlockNumberOrHashWithNumber(number), tx, api.filters) if err != nil { return nil, nil, err } block, senders, err := api._blockReader.BlockWithSenders(ctx, tx, hash, n) return block, senders, err } func (api *OtterscanAPIImpl) GetBlockTransactions(ctx context.Context, number rpc.BlockNumber, pageNumber uint8, pageSize uint8) (map[string]interface{}, error) { tx, err := api.db.BeginRo(ctx) if err != nil { return nil, err } defer tx.Rollback() b, senders, err := api.getBlockWithSenders(ctx, number, tx) if err != nil { return nil, err } if b == nil { return nil, nil } chainConfig, err := api.chainConfig(tx) if err != nil { return nil, err } getBlockRes, err := api.delegateGetBlockByNumber(tx, b, number, true) if err != nil { return nil, err } // Receipts receipts, err := api.getReceipts(ctx, tx, chainConfig, b, senders) if err != nil { return nil, fmt.Errorf("getReceipts error: %v", err) } result := make([]map[string]interface{}, 0, len(receipts)) for _, receipt := range receipts { txn := b.Transactions()[receipt.TransactionIndex] marshalledRcpt := marshalReceipt(receipt, txn, chainConfig, b, txn.Hash(), true) marshalledRcpt["logs"] = nil marshalledRcpt["logsBloom"] = nil result = append(result, marshalledRcpt) } // Pruned block attrs prunedBlock := map[string]interface{}{} for _, k := range []string{"timestamp", "miner", "baseFeePerGas"} { prunedBlock[k] = getBlockRes[k] } // Crop tx input to 4bytes var txs = getBlockRes["transactions"].([]interface{}) for _, rawTx := range txs { rpcTx := rawTx.(*ethapi.RPCTransaction) if len(rpcTx.Input) >= 4 { rpcTx.Input = rpcTx.Input[:4] } } // Crop page pageEnd := b.Transactions().Len() - int(pageNumber)*int(pageSize) pageStart := pageEnd - int(pageSize) if pageEnd < 0 { pageEnd = 0 } if pageStart < 0 { pageStart = 0 } response := map[string]interface{}{} getBlockRes["transactions"] = getBlockRes["transactions"].([]interface{})[pageStart:pageEnd] response["fullblock"] = getBlockRes response["receipts"] = result[pageStart:pageEnd] return response, nil }