package transactions import ( "context" "encoding/hex" "encoding/json" "errors" "fmt" "time" jsoniter "github.com/json-iterator/go" "github.com/ledgerwatch/log/v3" "github.com/ledgerwatch/erigon-lib/chain" libcommon "github.com/ledgerwatch/erigon-lib/common" "github.com/ledgerwatch/erigon-lib/kv" ethereum "github.com/ledgerwatch/erigon" "github.com/ledgerwatch/erigon/consensus" "github.com/ledgerwatch/erigon/core" "github.com/ledgerwatch/erigon/core/state" "github.com/ledgerwatch/erigon/core/types" "github.com/ledgerwatch/erigon/core/vm" "github.com/ledgerwatch/erigon/core/vm/evmtypes" "github.com/ledgerwatch/erigon/eth/stagedsync" "github.com/ledgerwatch/erigon/eth/tracers" "github.com/ledgerwatch/erigon/eth/tracers/logger" "github.com/ledgerwatch/erigon/polygon/bor/statefull" "github.com/ledgerwatch/erigon/turbo/rpchelper" "github.com/ledgerwatch/erigon/turbo/services" ) type BlockGetter interface { // GetBlockByHash retrieves a block from the database by hash, caching it if found. GetBlockByHash(hash libcommon.Hash) (*types.Block, error) // GetBlock retrieves a block from the database by hash and number, // caching it if found. GetBlock(hash libcommon.Hash, number uint64) *types.Block } // ComputeTxEnv returns the execution environment of a certain transaction. func ComputeTxEnv(ctx context.Context, engine consensus.EngineReader, block *types.Block, cfg *chain.Config, headerReader services.HeaderReader, dbtx kv.Tx, txIndex int, historyV3 bool) (core.Message, evmtypes.BlockContext, evmtypes.TxContext, *state.IntraBlockState, state.StateReader, error) { reader, err := rpchelper.CreateHistoryStateReader(dbtx, block.NumberU64(), txIndex, historyV3, cfg.ChainName) if err != nil { return nil, evmtypes.BlockContext{}, evmtypes.TxContext{}, nil, nil, err } // Create the parent state database statedb := state.New(reader) if txIndex == 0 && len(block.Transactions()) == 0 { return nil, evmtypes.BlockContext{}, evmtypes.TxContext{}, statedb, reader, nil } getHeader := func(hash libcommon.Hash, n uint64) *types.Header { h, _ := headerReader.HeaderByNumber(ctx, dbtx, n) return h } header := block.HeaderNoCopy() blockContext := core.NewEVMBlockContext(header, core.GetHashFn(header, getHeader), engine, nil) // Recompute transactions up to the target index. signer := types.MakeSigner(cfg, block.NumberU64(), block.Time()) if historyV3 { rules := cfg.Rules(blockContext.BlockNumber, blockContext.Time) txn := block.Transactions()[txIndex] statedb.SetTxContext(txn.Hash(), block.Hash(), txIndex) msg, _ := txn.AsMessage(*signer, block.BaseFee(), rules) if msg.FeeCap().IsZero() && engine != nil { syscall := func(contract libcommon.Address, data []byte) ([]byte, error) { return core.SysCallContract(contract, data, cfg, statedb, header, engine, true /* constCall */) } msg.SetIsFree(engine.IsServiceTransaction(msg.From(), syscall)) } TxContext := core.NewEVMTxContext(msg) return msg, blockContext, TxContext, statedb, reader, nil } vmenv := vm.NewEVM(blockContext, evmtypes.TxContext{}, statedb, cfg, vm.Config{}) rules := vmenv.ChainRules() consensusHeaderReader := stagedsync.NewChainReaderImpl(cfg, dbtx, nil, nil) logger := log.New("tracing") core.InitializeBlockExecution(engine.(consensus.Engine), consensusHeaderReader, header, cfg, statedb, logger) for idx, txn := range block.Transactions() { select { default: case <-ctx.Done(): return nil, evmtypes.BlockContext{}, evmtypes.TxContext{}, nil, nil, ctx.Err() } statedb.SetTxContext(txn.Hash(), block.Hash(), idx) // Assemble the transaction call message and return if the requested offset msg, _ := txn.AsMessage(*signer, block.BaseFee(), rules) if msg.FeeCap().IsZero() && engine != nil { syscall := func(contract libcommon.Address, data []byte) ([]byte, error) { return core.SysCallContract(contract, data, cfg, statedb, header, engine, true /* constCall */) } msg.SetIsFree(engine.IsServiceTransaction(msg.From(), syscall)) } TxContext := core.NewEVMTxContext(msg) if idx == txIndex { return msg, blockContext, TxContext, statedb, reader, nil } vmenv.Reset(TxContext, statedb) // Not yet the searched for transaction, execute on top of the current state if _, err := core.ApplyMessage(vmenv, msg, new(core.GasPool).AddGas(txn.GetGas()).AddBlobGas(txn.GetBlobGas()), true /* refunds */, false /* gasBailout */); err != nil { return nil, evmtypes.BlockContext{}, evmtypes.TxContext{}, nil, nil, fmt.Errorf("transaction %x failed: %w", txn.Hash(), err) } // Ensure any modifications are committed to the state // Only delete empty objects if EIP161 (part of Spurious Dragon) is in effect _ = statedb.FinalizeTx(rules, reader.(*state.PlainState)) if idx+1 == len(block.Transactions()) { // Return the state from evaluating all txs in the block, note no msg or TxContext in this case return nil, blockContext, evmtypes.TxContext{}, statedb, reader, nil } } return nil, evmtypes.BlockContext{}, evmtypes.TxContext{}, nil, nil, fmt.Errorf("transaction index %d out of range for block %x", txIndex, block.Hash()) } // TraceTx configures a new tracer according to the provided configuration, and // executes the given message in the provided environment. The return value will // be tracer dependent. func TraceTx( ctx context.Context, message core.Message, blockCtx evmtypes.BlockContext, txCtx evmtypes.TxContext, ibs evmtypes.IntraBlockState, config *tracers.TraceConfig, chainConfig *chain.Config, stream *jsoniter.Stream, callTimeout time.Duration, ) error { // Assemble the structured logger or the JavaScript tracer var ( tracer vm.EVMLogger err error ) var streaming bool switch { case config != nil && config.Tracer != nil: // Define a meaningful timeout of a single transaction trace timeout := callTimeout if config.Timeout != nil { if timeout, err = time.ParseDuration(*config.Timeout); err != nil { stream.WriteNil() return err } } // Construct the JavaScript tracer to execute with cfg := json.RawMessage("{}") if config != nil && config.TracerConfig != nil { cfg = *config.TracerConfig } if tracer, err = tracers.New(*config.Tracer, &tracers.Context{ TxHash: txCtx.TxHash, }, cfg); err != nil { stream.WriteNil() return err } // Handle timeouts and RPC cancellations deadlineCtx, cancel := context.WithTimeout(ctx, timeout) go func() { <-deadlineCtx.Done() tracer.(tracers.Tracer).Stop(errors.New("execution timeout")) }() defer cancel() streaming = false case config == nil: tracer = logger.NewJsonStreamLogger(nil, ctx, stream) streaming = true default: tracer = logger.NewJsonStreamLogger(config.LogConfig, ctx, stream) streaming = true } // Run the transaction with tracing enabled. vmenv := vm.NewEVM(blockCtx, txCtx, ibs, chainConfig, vm.Config{Debug: true, Tracer: tracer}) var refunds = true if config != nil && config.NoRefunds != nil && *config.NoRefunds { refunds = false } if streaming { stream.WriteObjectStart() stream.WriteObjectField("structLogs") stream.WriteArrayStart() } var result *core.ExecutionResult if config != nil && config.BorTx != nil && *config.BorTx { callmsg := prepareCallMessage(message) result, err = statefull.ApplyBorMessage(*vmenv, callmsg) } else { result, err = core.ApplyMessage(vmenv, message, new(core.GasPool).AddGas(message.Gas()).AddBlobGas(message.BlobGas()), refunds, false /* gasBailout */) } if err != nil { if streaming { stream.WriteArrayEnd() stream.WriteObjectEnd() stream.WriteMore() stream.WriteObjectField("resultHack") // higher-level func will assing it to NULL } else { stream.WriteNil() } return fmt.Errorf("tracing failed: %w", err) } // Depending on the tracer type, format and return the output if streaming { stream.WriteArrayEnd() stream.WriteMore() stream.WriteObjectField("gas") stream.WriteUint64(result.UsedGas) stream.WriteMore() stream.WriteObjectField("failed") stream.WriteBool(result.Failed()) stream.WriteMore() // If the result contains a revert reason, return it. returnVal := hex.EncodeToString(result.Return()) if len(result.Revert()) > 0 { returnVal = hex.EncodeToString(result.Revert()) } stream.WriteObjectField("returnValue") stream.WriteString(returnVal) stream.WriteObjectEnd() } else { if r, err1 := tracer.(tracers.Tracer).GetResult(); err1 == nil { stream.Write(r) } else { return err1 } } return nil } func prepareCallMessage(msg core.Message) statefull.Callmsg { return statefull.Callmsg{ CallMsg: ethereum.CallMsg{ From: msg.From(), To: msg.To(), Gas: msg.Gas(), GasPrice: msg.GasPrice(), Value: msg.Value(), Data: msg.Data(), AccessList: msg.AccessList(), }} }