package commands import ( "context" "math/big" "github.com/holiman/uint256" libcommon "github.com/ledgerwatch/erigon-lib/common" "github.com/ledgerwatch/erigon/common/hexutil" "github.com/ledgerwatch/erigon/core/vm" ) func (api *OtterscanAPIImpl) TraceTransaction(ctx context.Context, hash libcommon.Hash) ([]*TraceEntry, error) { tx, err := api.db.BeginRo(ctx) if err != nil { return nil, err } defer tx.Rollback() tracer := NewTransactionTracer(ctx) if _, err := api.runTracer(ctx, tx, hash, tracer); err != nil { return nil, err } return tracer.Results, nil } type TraceEntry struct { Type string `json:"type"` Depth int `json:"depth"` From libcommon.Address `json:"from"` To libcommon.Address `json:"to"` Value *hexutil.Big `json:"value"` Input hexutil.Bytes `json:"input"` } type TransactionTracer struct { DefaultTracer ctx context.Context Results []*TraceEntry depth int // computed from CaptureStart, CaptureEnter, and CaptureExit calls } func NewTransactionTracer(ctx context.Context) *TransactionTracer { return &TransactionTracer{ ctx: ctx, Results: make([]*TraceEntry, 0), } } func (t *TransactionTracer) captureStartOrEnter(typ vm.OpCode, from, to libcommon.Address, precompile bool, input []byte, value *uint256.Int) { if precompile { return } inputCopy := make([]byte, len(input)) copy(inputCopy, input) _value := new(big.Int) if value != nil { _value.Set(value.ToBig()) } if typ == vm.CALL { t.Results = append(t.Results, &TraceEntry{"CALL", t.depth, from, to, (*hexutil.Big)(_value), inputCopy}) return } if typ == vm.STATICCALL { t.Results = append(t.Results, &TraceEntry{"STATICCALL", t.depth, from, to, nil, inputCopy}) return } if typ == vm.DELEGATECALL { t.Results = append(t.Results, &TraceEntry{"DELEGATECALL", t.depth, from, to, nil, inputCopy}) return } if typ == vm.CALLCODE { t.Results = append(t.Results, &TraceEntry{"CALLCODE", t.depth, from, to, (*hexutil.Big)(_value), inputCopy}) return } if typ == vm.CREATE { t.Results = append(t.Results, &TraceEntry{"CREATE", t.depth, from, to, (*hexutil.Big)(value.ToBig()), inputCopy}) return } if typ == vm.CREATE2 { t.Results = append(t.Results, &TraceEntry{"CREATE2", t.depth, from, to, (*hexutil.Big)(value.ToBig()), inputCopy}) return } if typ == vm.SELFDESTRUCT { last := t.Results[len(t.Results)-1] t.Results = append(t.Results, &TraceEntry{"SELFDESTRUCT", last.Depth + 1, from, to, (*hexutil.Big)(value.ToBig()), nil}) } } func (t *TransactionTracer) CaptureStart(env vm.VMInterface, from libcommon.Address, to libcommon.Address, precompile bool, create bool, input []byte, gas uint64, value *uint256.Int, code []byte) { t.depth = 0 t.captureStartOrEnter(vm.CALL, from, to, precompile, input, value) } func (t *TransactionTracer) CaptureEnter(typ vm.OpCode, from libcommon.Address, to libcommon.Address, precompile bool, create bool, input []byte, gas uint64, value *uint256.Int, code []byte) { t.depth++ t.captureStartOrEnter(typ, from, to, precompile, input, value) } func (t *TransactionTracer) CaptureExit(output []byte, usedGas uint64, err error) { t.depth-- }