mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-19 00:54:12 +00:00
527 lines
14 KiB
Go
527 lines
14 KiB
Go
package commands
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"math/big"
|
|
"time"
|
|
|
|
"github.com/holiman/uint256"
|
|
jsoniter "github.com/json-iterator/go"
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/log/v3"
|
|
|
|
"github.com/ledgerwatch/erigon/core/vm/evmtypes"
|
|
|
|
"github.com/ledgerwatch/erigon/common/hexutil"
|
|
"github.com/ledgerwatch/erigon/common/math"
|
|
"github.com/ledgerwatch/erigon/core"
|
|
"github.com/ledgerwatch/erigon/core/rawdb"
|
|
"github.com/ledgerwatch/erigon/core/state"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/core/vm"
|
|
"github.com/ledgerwatch/erigon/eth/tracers"
|
|
"github.com/ledgerwatch/erigon/rpc"
|
|
"github.com/ledgerwatch/erigon/turbo/adapter/ethapi"
|
|
"github.com/ledgerwatch/erigon/turbo/rpchelper"
|
|
"github.com/ledgerwatch/erigon/turbo/transactions"
|
|
)
|
|
|
|
// TraceBlockByNumber implements debug_traceBlockByNumber. Returns Geth style block traces.
|
|
func (api *PrivateDebugAPIImpl) TraceBlockByNumber(ctx context.Context, blockNum rpc.BlockNumber, config *tracers.TraceConfig, stream *jsoniter.Stream) error {
|
|
return api.traceBlock(ctx, rpc.BlockNumberOrHashWithNumber(blockNum), config, stream)
|
|
}
|
|
|
|
// TraceBlockByHash implements debug_traceBlockByHash. Returns Geth style block traces.
|
|
func (api *PrivateDebugAPIImpl) TraceBlockByHash(ctx context.Context, hash common.Hash, config *tracers.TraceConfig, stream *jsoniter.Stream) error {
|
|
return api.traceBlock(ctx, rpc.BlockNumberOrHashWithHash(hash, true), config, stream)
|
|
}
|
|
|
|
func (api *PrivateDebugAPIImpl) traceBlock(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash, config *tracers.TraceConfig, stream *jsoniter.Stream) error {
|
|
tx, err := api.db.BeginRo(ctx)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
defer tx.Rollback()
|
|
var (
|
|
block *types.Block
|
|
number rpc.BlockNumber
|
|
numberOk bool
|
|
hash common.Hash
|
|
hashOk bool
|
|
)
|
|
if number, numberOk = blockNrOrHash.Number(); numberOk {
|
|
block, err = api.blockByRPCNumber(number, tx)
|
|
} else if hash, hashOk = blockNrOrHash.Hash(); hashOk {
|
|
block, err = api.blockByHashWithSenders(tx, hash)
|
|
} else {
|
|
return fmt.Errorf("invalid arguments; neither block nor hash specified")
|
|
}
|
|
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
|
|
if block == nil {
|
|
if numberOk {
|
|
return fmt.Errorf("invalid arguments; block with number %d not found", number)
|
|
}
|
|
return fmt.Errorf("invalid arguments; block with hash %x not found", hash)
|
|
}
|
|
|
|
// if we've pruned this history away for this block then just return early
|
|
// to save any red herring errors
|
|
err = api.BaseAPI.checkPruneHistory(tx, block.NumberU64())
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
|
|
if config == nil {
|
|
config = &tracers.TraceConfig{}
|
|
}
|
|
|
|
if config.BorTraceEnabled == nil {
|
|
config.BorTraceEnabled = newBoolPtr(false)
|
|
}
|
|
|
|
var excessDataGas *big.Int
|
|
parentBlock, err := api.blockByHashWithSenders(tx, block.ParentHash())
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
if parentBlock != nil {
|
|
excessDataGas = parentBlock.ExcessDataGas()
|
|
}
|
|
chainConfig, err := api.chainConfig(tx)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
engine := api.engine()
|
|
|
|
_, blockCtx, _, ibs, _, err := transactions.ComputeTxEnv(ctx, engine, block, chainConfig, api._blockReader, tx, 0, api.historyV3(tx))
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
|
|
signer := types.MakeSigner(chainConfig, block.NumberU64())
|
|
rules := chainConfig.Rules(block.NumberU64(), block.Time())
|
|
stream.WriteArrayStart()
|
|
|
|
borTx, _, _, _ := rawdb.ReadBorTransactionForBlock(tx, block)
|
|
txns := block.Transactions()
|
|
if borTx != nil && *config.BorTraceEnabled {
|
|
txns = append(txns, borTx)
|
|
}
|
|
|
|
for idx, txn := range txns {
|
|
stream.WriteObjectStart()
|
|
stream.WriteObjectField("result")
|
|
select {
|
|
default:
|
|
case <-ctx.Done():
|
|
stream.WriteNil()
|
|
return ctx.Err()
|
|
}
|
|
ibs.SetTxContext(txn.Hash(), block.Hash(), idx)
|
|
msg, _ := txn.AsMessage(*signer, block.BaseFee(), rules)
|
|
|
|
if msg.FeeCap().IsZero() && engine != nil {
|
|
syscall := func(contract common.Address, data []byte) ([]byte, error) {
|
|
return core.SysCallContract(contract, data, chainConfig, ibs, block.Header(), engine, true /* constCall */, excessDataGas)
|
|
}
|
|
msg.SetIsFree(engine.IsServiceTransaction(msg.From(), syscall))
|
|
}
|
|
|
|
txCtx := evmtypes.TxContext{
|
|
TxHash: txn.Hash(),
|
|
Origin: msg.From(),
|
|
GasPrice: msg.GasPrice(),
|
|
}
|
|
|
|
if borTx != nil && idx == len(txns)-1 {
|
|
if *config.BorTraceEnabled {
|
|
config.BorTx = newBoolPtr(true)
|
|
}
|
|
}
|
|
|
|
err = transactions.TraceTx(ctx, msg, blockCtx, txCtx, ibs, config, chainConfig, stream, api.evmCallTimeout)
|
|
if err == nil {
|
|
err = ibs.FinalizeTx(rules, state.NewNoopWriter())
|
|
}
|
|
stream.WriteObjectEnd()
|
|
|
|
// if we have an error we want to output valid json for it before continuing after clearing down potential writes to the stream
|
|
if err != nil {
|
|
stream.WriteMore()
|
|
stream.WriteObjectStart()
|
|
err = rpc.HandleError(err, stream)
|
|
stream.WriteObjectEnd()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if idx != len(txns)-1 {
|
|
stream.WriteMore()
|
|
}
|
|
stream.Flush()
|
|
}
|
|
stream.WriteArrayEnd()
|
|
stream.Flush()
|
|
return nil
|
|
}
|
|
|
|
// TraceTransaction implements debug_traceTransaction. Returns Geth style transaction traces.
|
|
func (api *PrivateDebugAPIImpl) TraceTransaction(ctx context.Context, hash common.Hash, config *tracers.TraceConfig, stream *jsoniter.Stream) error {
|
|
tx, err := api.db.BeginRo(ctx)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
defer tx.Rollback()
|
|
chainConfig, err := api.chainConfig(tx)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
// Retrieve the transaction and assemble its EVM context
|
|
blockNum, ok, err := api.txnLookup(ctx, tx, hash)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
if !ok {
|
|
stream.WriteNil()
|
|
return nil
|
|
}
|
|
|
|
// check pruning to ensure we have history at this block level
|
|
err = api.BaseAPI.checkPruneHistory(tx, blockNum)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
|
|
// Private API returns 0 if transaction is not found.
|
|
if blockNum == 0 && chainConfig.Bor != nil {
|
|
blockNumPtr, err := rawdb.ReadBorTxLookupEntry(tx, hash)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
if blockNumPtr == nil {
|
|
stream.WriteNil()
|
|
return nil
|
|
}
|
|
blockNum = *blockNumPtr
|
|
}
|
|
block, err := api.blockByNumberWithSenders(tx, blockNum)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
if block == nil {
|
|
stream.WriteNil()
|
|
return nil
|
|
}
|
|
var txnIndex uint64
|
|
var txn types.Transaction
|
|
for i, transaction := range block.Transactions() {
|
|
if transaction.Hash() == hash {
|
|
txnIndex = uint64(i)
|
|
txn = transaction
|
|
break
|
|
}
|
|
}
|
|
if txn == nil {
|
|
var borTx types.Transaction
|
|
borTx, _, _, _, err = rawdb.ReadBorTransaction(tx, hash)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if borTx != nil {
|
|
stream.WriteNil()
|
|
return nil
|
|
}
|
|
stream.WriteNil()
|
|
return fmt.Errorf("transaction %#x not found", hash)
|
|
}
|
|
engine := api.engine()
|
|
|
|
msg, blockCtx, txCtx, ibs, _, err := transactions.ComputeTxEnv(ctx, engine, block, chainConfig, api._blockReader, tx, int(txnIndex), api.historyV3(tx))
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
// Trace the transaction and return
|
|
return transactions.TraceTx(ctx, msg, blockCtx, txCtx, ibs, config, chainConfig, stream, api.evmCallTimeout)
|
|
}
|
|
|
|
func (api *PrivateDebugAPIImpl) TraceCall(ctx context.Context, args ethapi.CallArgs, blockNrOrHash rpc.BlockNumberOrHash, config *tracers.TraceConfig, stream *jsoniter.Stream) error {
|
|
dbtx, err := api.db.BeginRo(ctx)
|
|
if err != nil {
|
|
return fmt.Errorf("create ro transaction: %v", err)
|
|
}
|
|
defer dbtx.Rollback()
|
|
|
|
chainConfig, err := api.chainConfig(dbtx)
|
|
if err != nil {
|
|
return fmt.Errorf("read chain config: %v", err)
|
|
}
|
|
engine := api.engine()
|
|
|
|
blockNumber, hash, _, err := rpchelper.GetBlockNumber(blockNrOrHash, dbtx, api.filters)
|
|
if err != nil {
|
|
return fmt.Errorf("get block number: %v", err)
|
|
}
|
|
|
|
err = api.BaseAPI.checkPruneHistory(dbtx, blockNumber)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
stateReader, err := rpchelper.CreateStateReader(ctx, dbtx, blockNrOrHash, 0, api.filters, api.stateCache, api.historyV3(dbtx), chainConfig.ChainName)
|
|
if err != nil {
|
|
return fmt.Errorf("create state reader: %v", err)
|
|
}
|
|
header, err := api._blockReader.Header(context.Background(), dbtx, hash, blockNumber)
|
|
if err != nil {
|
|
return fmt.Errorf("could not fetch header %d(%x): %v", blockNumber, hash, err)
|
|
}
|
|
if header == nil {
|
|
return fmt.Errorf("block %d(%x) not found", blockNumber, hash)
|
|
}
|
|
ibs := state.New(stateReader)
|
|
|
|
if config != nil && config.StateOverrides != nil {
|
|
if err := config.StateOverrides.Override(ibs); err != nil {
|
|
return fmt.Errorf("override state: %v", err)
|
|
}
|
|
}
|
|
|
|
var baseFee *uint256.Int
|
|
if header != nil && header.BaseFee != nil {
|
|
var overflow bool
|
|
baseFee, overflow = uint256.FromBig(header.BaseFee)
|
|
if overflow {
|
|
return fmt.Errorf("header.BaseFee uint256 overflow")
|
|
}
|
|
}
|
|
msg, err := args.ToMessage(api.GasCap, baseFee)
|
|
if err != nil {
|
|
return fmt.Errorf("convert args to msg: %v", err)
|
|
}
|
|
|
|
blockCtx := transactions.NewEVMBlockContext(engine, header, blockNrOrHash.RequireCanonical, dbtx, api._blockReader)
|
|
txCtx := core.NewEVMTxContext(msg)
|
|
// Trace the transaction and return
|
|
return transactions.TraceTx(ctx, msg, blockCtx, txCtx, ibs, config, chainConfig, stream, api.evmCallTimeout)
|
|
}
|
|
|
|
func (api *PrivateDebugAPIImpl) TraceCallMany(ctx context.Context, bundles []Bundle, simulateContext StateContext, config *tracers.TraceConfig, stream *jsoniter.Stream) error {
|
|
var (
|
|
hash common.Hash
|
|
replayTransactions types.Transactions
|
|
evm *vm.EVM
|
|
blockCtx evmtypes.BlockContext
|
|
txCtx evmtypes.TxContext
|
|
overrideBlockHash map[uint64]common.Hash
|
|
baseFee uint256.Int
|
|
)
|
|
|
|
if config == nil {
|
|
config = &tracers.TraceConfig{}
|
|
}
|
|
|
|
overrideBlockHash = make(map[uint64]common.Hash)
|
|
tx, err := api.db.BeginRo(ctx)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
defer tx.Rollback()
|
|
chainConfig, err := api.chainConfig(tx)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
if len(bundles) == 0 {
|
|
stream.WriteNil()
|
|
return fmt.Errorf("empty bundles")
|
|
}
|
|
empty := true
|
|
for _, bundle := range bundles {
|
|
if len(bundle.Transactions) != 0 {
|
|
empty = false
|
|
}
|
|
}
|
|
|
|
if empty {
|
|
stream.WriteNil()
|
|
return fmt.Errorf("empty bundles")
|
|
}
|
|
|
|
defer func(start time.Time) { log.Trace("Tracing CallMany finished", "runtime", time.Since(start)) }(time.Now())
|
|
|
|
blockNum, hash, _, err := rpchelper.GetBlockNumber(simulateContext.BlockNumber, tx, api.filters)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
|
|
err = api.BaseAPI.checkPruneHistory(tx, blockNum)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
block, err := api.blockByNumberWithSenders(tx, blockNum)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
|
|
// -1 is a default value for transaction index.
|
|
// If it's -1, we will try to replay every single transaction in that block
|
|
transactionIndex := -1
|
|
|
|
if simulateContext.TransactionIndex != nil {
|
|
transactionIndex = *simulateContext.TransactionIndex
|
|
}
|
|
|
|
if transactionIndex == -1 {
|
|
transactionIndex = len(block.Transactions())
|
|
}
|
|
|
|
replayTransactions = block.Transactions()[:transactionIndex]
|
|
|
|
stateReader, err := rpchelper.CreateStateReader(ctx, tx, rpc.BlockNumberOrHashWithNumber(rpc.BlockNumber(blockNum-1)), 0, api.filters, api.stateCache, api.historyV3(tx), chainConfig.ChainName)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
|
|
st := state.New(stateReader)
|
|
|
|
parent := block.Header()
|
|
|
|
if parent == nil {
|
|
stream.WriteNil()
|
|
return fmt.Errorf("block %d(%x) not found", blockNum, hash)
|
|
}
|
|
|
|
getHash := func(i uint64) common.Hash {
|
|
if hash, ok := overrideBlockHash[i]; ok {
|
|
return hash
|
|
}
|
|
hash, err := rawdb.ReadCanonicalHash(tx, i)
|
|
if err != nil {
|
|
log.Debug("Can't get block hash by number", "number", i, "only-canonical", true)
|
|
}
|
|
return hash
|
|
}
|
|
|
|
if parent.BaseFee != nil {
|
|
baseFee.SetFromBig(parent.BaseFee)
|
|
}
|
|
|
|
blockCtx = evmtypes.BlockContext{
|
|
CanTransfer: core.CanTransfer,
|
|
Transfer: core.Transfer,
|
|
GetHash: getHash,
|
|
Coinbase: parent.Coinbase,
|
|
BlockNumber: parent.Number.Uint64(),
|
|
Time: parent.Time,
|
|
Difficulty: new(big.Int).Set(parent.Difficulty),
|
|
GasLimit: parent.GasLimit,
|
|
BaseFee: &baseFee,
|
|
}
|
|
|
|
// Get a new instance of the EVM
|
|
evm = vm.NewEVM(blockCtx, txCtx, st, chainConfig, vm.Config{Debug: false})
|
|
signer := types.MakeSigner(chainConfig, blockNum)
|
|
rules := chainConfig.Rules(blockNum, blockCtx.Time)
|
|
|
|
// Setup the gas pool (also for unmetered requests)
|
|
// and apply the message.
|
|
gp := new(core.GasPool).AddGas(math.MaxUint64).AddDataGas(math.MaxUint64)
|
|
for idx, txn := range replayTransactions {
|
|
st.SetTxContext(txn.Hash(), block.Hash(), idx)
|
|
msg, err := txn.AsMessage(*signer, block.BaseFee(), rules)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
txCtx = core.NewEVMTxContext(msg)
|
|
evm = vm.NewEVM(blockCtx, txCtx, evm.IntraBlockState(), chainConfig, vm.Config{Debug: false})
|
|
// Execute the transaction message
|
|
_, err = core.ApplyMessage(evm, msg, gp, true /* refunds */, false /* gasBailout */)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
_ = st.FinalizeTx(rules, state.NewNoopWriter())
|
|
|
|
}
|
|
|
|
// after replaying the txns, we want to overload the state
|
|
if config.StateOverrides != nil {
|
|
err = config.StateOverrides.Override(evm.IntraBlockState().(*state.IntraBlockState))
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
}
|
|
|
|
stream.WriteArrayStart()
|
|
for bundle_index, bundle := range bundles {
|
|
stream.WriteArrayStart()
|
|
// first change blockContext
|
|
blockHeaderOverride(&blockCtx, bundle.BlockOverride, overrideBlockHash)
|
|
for txn_index, txn := range bundle.Transactions {
|
|
if txn.Gas == nil || *(txn.Gas) == 0 {
|
|
txn.Gas = (*hexutil.Uint64)(&api.GasCap)
|
|
}
|
|
msg, err := txn.ToMessage(api.GasCap, blockCtx.BaseFee)
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
txCtx = core.NewEVMTxContext(msg)
|
|
ibs := evm.IntraBlockState().(*state.IntraBlockState)
|
|
ibs.SetTxContext(common.Hash{}, parent.Hash(), txn_index)
|
|
err = transactions.TraceTx(ctx, msg, blockCtx, txCtx, evm.IntraBlockState(), config, chainConfig, stream, api.evmCallTimeout)
|
|
|
|
if err != nil {
|
|
stream.WriteNil()
|
|
return err
|
|
}
|
|
|
|
_ = ibs.FinalizeTx(rules, state.NewNoopWriter())
|
|
|
|
if txn_index < len(bundle.Transactions)-1 {
|
|
stream.WriteMore()
|
|
}
|
|
}
|
|
stream.WriteArrayEnd()
|
|
|
|
if bundle_index < len(bundles)-1 {
|
|
stream.WriteMore()
|
|
}
|
|
blockCtx.BlockNumber++
|
|
blockCtx.Time++
|
|
}
|
|
stream.WriteArrayEnd()
|
|
return nil
|
|
}
|
|
|
|
func newBoolPtr(bb bool) *bool {
|
|
b := bb
|
|
return &b
|
|
}
|