// Copyright 2020 The go-ethereum Authors // This file is part of go-ethereum. // // go-ethereum is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // go-ethereum is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with go-ethereum. If not, see . package t8ntool import ( "crypto/ecdsa" "encoding/json" "errors" "fmt" "math/big" "os" "path" "path/filepath" "github.com/holiman/uint256" "github.com/ledgerwatch/erigon/cmd/rpcdaemon/commands" "github.com/ledgerwatch/erigon/common" "github.com/ledgerwatch/erigon/common/hexutil" "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/crypto" "github.com/ledgerwatch/erigon/params" "github.com/ledgerwatch/erigon/rlp" "github.com/ledgerwatch/erigon/tests" "github.com/ledgerwatch/log/v3" "github.com/urfave/cli" ) const ( ErrorEVM = 2 ErrorVMConfig = 3 ErrorMissingBlockhash = 4 ErrorJson = 10 ErrorIO = 11 stdinSelector = "stdin" ) type NumberedError struct { errorCode int err error } func NewError(errorCode int, err error) *NumberedError { return &NumberedError{errorCode, err} } func (n *NumberedError) Error() string { return fmt.Sprintf("ERROR(%d): %v", n.errorCode, n.err.Error()) } func (n *NumberedError) Code() int { return n.errorCode } type input struct { Alloc core.GenesisAlloc `json:"alloc,omitempty"` Env *stEnv `json:"env,omitempty"` Txs []*txWithKey `json:"txs,omitempty"` } func Main(ctx *cli.Context) error { log.Root().SetHandler(log.LvlFilterHandler(log.LvlInfo, log.StderrHandler)) /* // Configure the go-ethereum logger glogger := log.NewGlogHandler(log.StreamHandler(os.Stderr, log.TerminalFormat(false))) glogger.Verbosity(log.Lvl(ctx.Int(VerbosityFlag.Name))) log.Root().SetHandler(glogger) */ var ( err error tracer vm.Tracer baseDir = "" ) var getTracer func(txIndex int, txHash common.Hash) (vm.Tracer, error) // If user specified a basedir, make sure it exists if ctx.IsSet(OutputBasedir.Name) { if base := ctx.String(OutputBasedir.Name); len(base) > 0 { err2 := os.MkdirAll(base, 0755) // //rw-r--r-- if err2 != nil { return NewError(ErrorIO, fmt.Errorf("failed creating output basedir: %v", err2)) } baseDir = base } } if ctx.Bool(TraceFlag.Name) { // Configure the EVM logger logConfig := &vm.LogConfig{ DisableStack: ctx.Bool(TraceDisableStackFlag.Name), DisableMemory: ctx.Bool(TraceDisableMemoryFlag.Name), DisableReturnData: ctx.Bool(TraceDisableReturnDataFlag.Name), Debug: true, } var prevFile *os.File // This one closes the last file defer func() { if prevFile != nil { prevFile.Close() } }() getTracer = func(txIndex int, txHash common.Hash) (vm.Tracer, error) { if prevFile != nil { prevFile.Close() } traceFile, err2 := os.Create(path.Join(baseDir, fmt.Sprintf("trace-%d-%v.jsonl", txIndex, txHash.String()))) if err2 != nil { return nil, NewError(ErrorIO, fmt.Errorf("failed creating trace-file: %v", err2)) } prevFile = traceFile return vm.NewJSONLogger(logConfig, traceFile), nil } } else { getTracer = func(txIndex int, txHash common.Hash) (tracer vm.Tracer, err error) { return nil, nil } } // We need to load three things: alloc, env and transactions. May be either in // stdin input or in files. // Check if anything needs to be read from stdin var ( prestate Prestate txs types.Transactions // txs to apply allocStr = ctx.String(InputAllocFlag.Name) envStr = ctx.String(InputEnvFlag.Name) txStr = ctx.String(InputTxsFlag.Name) inputData = &input{} ) // Figure out the prestate alloc if allocStr == stdinSelector || envStr == stdinSelector || txStr == stdinSelector { decoder := json.NewDecoder(os.Stdin) decoder.Decode(inputData) //nolint:errcheck } if allocStr != stdinSelector { inFile, err1 := os.Open(allocStr) if err1 != nil { return NewError(ErrorIO, fmt.Errorf("failed reading alloc file: %v", err1)) } defer inFile.Close() decoder := json.NewDecoder(inFile) if err = decoder.Decode(&inputData.Alloc); err != nil { return NewError(ErrorJson, fmt.Errorf("failed unmarshaling alloc-file: %v", err)) } } prestate.Pre = inputData.Alloc // Set the block environment if envStr != stdinSelector { inFile, err1 := os.Open(envStr) if err1 != nil { return NewError(ErrorIO, fmt.Errorf("failed reading env file: %v", err1)) } defer inFile.Close() decoder := json.NewDecoder(inFile) var env stEnv if err = decoder.Decode(&env); err != nil { return NewError(ErrorJson, fmt.Errorf("failed unmarshaling env-file: %v", err)) } inputData.Env = &env } prestate.Env = *inputData.Env vmConfig := vm.Config{ Tracer: tracer, Debug: (tracer != nil), } // Construct the chainconfig var chainConfig *params.ChainConfig if cConf, extraEips, err1 := tests.GetChainConfig(ctx.String(ForknameFlag.Name)); err1 != nil { return NewError(ErrorVMConfig, fmt.Errorf("failed constructing chain configuration: %v", err1)) } else { //nolint:golint chainConfig = cConf vmConfig.ExtraEips = extraEips } // Set the chain id chainConfig.ChainID = big.NewInt(ctx.Int64(ChainIDFlag.Name)) var txsWithKeys []*txWithKey if txStr != stdinSelector { inFile, err1 := os.Open(txStr) if err1 != nil { return NewError(ErrorIO, fmt.Errorf("failed reading txs file: %v", err1)) } defer inFile.Close() decoder := json.NewDecoder(inFile) if err = decoder.Decode(&txsWithKeys); err != nil { return NewError(ErrorJson, fmt.Errorf("failed unmarshaling txs-file: %v", err)) } } else { txsWithKeys = inputData.Txs } // We may have to sign the transactions. signer := types.MakeSigner(chainConfig, prestate.Env.Number) if txs, err = signUnsignedTransactions(txsWithKeys, *signer); err != nil { return NewError(ErrorJson, fmt.Errorf("failed signing transactions: %v", err)) } // Sanity check, to not `panic` in state_transition if chainConfig.IsLondon(prestate.Env.Number) { if prestate.Env.BaseFee == nil { return NewError(ErrorVMConfig, errors.New("EIP-1559 config but missing 'currentBaseFee' in env section")) } } // Run the test and aggregate the result _, result, err1 := prestate.Apply(vmConfig, chainConfig, txs, ctx.Int64(RewardFlag.Name), getTracer) if err1 != nil { return err1 } body, _ := rlp.EncodeToBytes(txs) // Dump the excution result collector := make(Alloc) // TODO: Where DumpToCollector is declared? //state.DumpToCollector(collector, false, false, false, nil, -1) return dispatchOutput(ctx, baseDir, result, collector, body) } // txWithKey is a helper-struct, to allow us to use the types.Transaction along with // a `secretKey`-field, for input type txWithKey struct { key *ecdsa.PrivateKey tx types.Transaction } func (t *txWithKey) UnmarshalJSON(input []byte) error { // Read the secretKey, if present type sKey struct { Key *common.Hash `json:"secretKey"` } var key sKey if err := json.Unmarshal(input, &key); err != nil { return err } if key.Key != nil { k := key.Key.Hex()[2:] if ecdsaKey, err := crypto.HexToECDSA(k); err == nil { t.key = ecdsaKey } else { return err } } gasPrice, value := uint256.NewInt(0), uint256.NewInt(0) var overflow bool // Now, read the transaction itself var txJson commands.RPCTransaction if err := json.Unmarshal(input, &txJson); err != nil { return err } if txJson.Value != nil { value, overflow = uint256.FromBig((*big.Int)(txJson.Value)) if overflow { return fmt.Errorf("value field caused an overflow (uint256)") } } if txJson.GasPrice != nil { gasPrice, overflow = uint256.FromBig((*big.Int)(txJson.GasPrice)) if overflow { return fmt.Errorf("gasPrice field caused an overflow (uint256)") } } // assemble transaction t.tx = types.NewTransaction(uint64(txJson.Nonce), *txJson.To, value, uint64(txJson.Gas), gasPrice, txJson.Input) return nil } // signUnsignedTransactions converts the input txs to canonical transactions. // // The transactions can have two forms, either // 1. unsigned or // 2. signed // For (1), r, s, v, need so be zero, and the `secretKey` needs to be set. // If so, we sign it here and now, with the given `secretKey` // If the condition above is not met, then it's considered a signed transaction. // // To manage this, we read the transactions twice, first trying to read the secretKeys, // and secondly to read them with the standard tx json format func signUnsignedTransactions(txs []*txWithKey, signer types.Signer) (types.Transactions, error) { var signedTxs []types.Transaction for i, txWithKey := range txs { tx := txWithKey.tx key := txWithKey.key v, r, s := tx.RawSignatureValues() if key != nil && v.IsZero() && r.IsZero() && s.IsZero() { // This transaction needs to be signed signed, err := types.SignTx(tx, signer, key) if err != nil { return nil, NewError(ErrorJson, fmt.Errorf("tx %d: failed to sign tx: %v", i, err)) } signedTxs = append(signedTxs, signed) } else { // Already signed signedTxs = append(signedTxs, tx) } } return signedTxs, nil } type Alloc map[common.Address]core.GenesisAccount func (g Alloc) OnRoot(common.Hash) {} func (g Alloc) OnAccount(addr common.Address, dumpAccount state.DumpAccount) { balance, _ := new(big.Int).SetString(dumpAccount.Balance, 10) var storage map[common.Hash]common.Hash if dumpAccount.Storage != nil { storage = make(map[common.Hash]common.Hash) for k, v := range dumpAccount.Storage { storage[common.HexToHash(k)] = common.HexToHash(v) } } genesisAccount := core.GenesisAccount{ Code: dumpAccount.Code, Storage: storage, Balance: balance, Nonce: dumpAccount.Nonce, } g[addr] = genesisAccount } // saveFile marshalls the object to the given file func saveFile(baseDir, filename string, data interface{}) error { b, err := json.MarshalIndent(data, "", " ") if err != nil { return NewError(ErrorJson, fmt.Errorf("failed marshalling output: %v", err)) } location := filepath.Join(baseDir, filename) if err = os.WriteFile(location, b, 0644); err != nil { //nolint:gosec return NewError(ErrorIO, fmt.Errorf("failed writing output: %v", err)) } log.Info("Wrote file", "file", location) return nil } // dispatchOutput writes the output data to either stderr or stdout, or to the specified // files func dispatchOutput(ctx *cli.Context, baseDir string, result *ExecutionResult, alloc Alloc, body hexutil.Bytes) error { stdOutObject := make(map[string]interface{}) stdErrObject := make(map[string]interface{}) dispatch := func(baseDir, fName, name string, obj interface{}) error { switch fName { case "stdout": stdOutObject[name] = obj case "stderr": stdErrObject[name] = obj case "": // don't save default: // save to file if err := saveFile(baseDir, fName, obj); err != nil { return err } } return nil } if err := dispatch(baseDir, ctx.String(OutputAllocFlag.Name), "alloc", alloc); err != nil { return err } if err := dispatch(baseDir, ctx.String(OutputResultFlag.Name), "result", result); err != nil { return err } if err := dispatch(baseDir, ctx.String(OutputBodyFlag.Name), "body", body); err != nil { return err } if len(stdOutObject) > 0 { b, err := json.MarshalIndent(stdOutObject, "", " ") if err != nil { return NewError(ErrorJson, fmt.Errorf("failed marshalling output: %v", err)) } os.Stdout.Write(b) } if len(stdErrObject) > 0 { b, err := json.MarshalIndent(stdErrObject, "", " ") if err != nil { return NewError(ErrorJson, fmt.Errorf("failed marshalling output: %v", err)) } os.Stderr.Write(b) } return nil }