package jsonrpc import ( "context" "fmt" "github.com/ledgerwatch/erigon-lib/common/hexutil" "math/big" "testing" "time" "github.com/holiman/uint256" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" libcommon "github.com/ledgerwatch/erigon-lib/common" "github.com/ledgerwatch/erigon-lib/common/hexutility" "github.com/ledgerwatch/erigon-lib/gointerfaces/txpool" "github.com/ledgerwatch/erigon-lib/kv" "github.com/ledgerwatch/erigon-lib/kv/kvcache" "github.com/ledgerwatch/erigon/cmd/rpcdaemon/rpcdaemontest" "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/crypto" "github.com/ledgerwatch/erigon/params" "github.com/ledgerwatch/erigon/rpc" "github.com/ledgerwatch/erigon/rpc/rpccfg" "github.com/ledgerwatch/erigon/turbo/adapter/ethapi" "github.com/ledgerwatch/erigon/turbo/rpchelper" "github.com/ledgerwatch/erigon/turbo/stages/mock" "github.com/ledgerwatch/erigon/turbo/trie" "github.com/ledgerwatch/log/v3" ) func TestEstimateGas(t *testing.T) { m, _, _ := rpcdaemontest.CreateTestSentry(t) agg := m.HistoryV3Components() stateCache := kvcache.New(kvcache.DefaultCoherentConfig) ctx, conn := rpcdaemontest.CreateTestGrpcConn(t, mock.Mock(t)) mining := txpool.NewMiningClient(conn) ff := rpchelper.New(ctx, nil, nil, mining, func() {}, m.Log) api := NewEthAPI(NewBaseApi(ff, stateCache, m.BlockReader, agg, false, rpccfg.DefaultEvmCallTimeout, m.Engine, m.Dirs), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New()) var from = libcommon.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7") var to = libcommon.HexToAddress("0x0d3ab14bbad3d99f4203bd7a11acb94882050e7e") if _, err := api.EstimateGas(context.Background(), ðapi.CallArgs{ From: &from, To: &to, }, nil); err != nil { t.Errorf("calling EstimateGas: %v", err) } } func TestEthCallNonCanonical(t *testing.T) { m, _, _ := rpcdaemontest.CreateTestSentry(t) agg := m.HistoryV3Components() stateCache := kvcache.New(kvcache.DefaultCoherentConfig) api := NewEthAPI(NewBaseApi(nil, stateCache, m.BlockReader, agg, false, rpccfg.DefaultEvmCallTimeout, m.Engine, m.Dirs), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New()) var from = libcommon.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7") var to = libcommon.HexToAddress("0x0d3ab14bbad3d99f4203bd7a11acb94882050e7e") if _, err := api.Call(context.Background(), ethapi.CallArgs{ From: &from, To: &to, }, rpc.BlockNumberOrHashWithHash(libcommon.HexToHash("0x3fcb7c0d4569fddc89cbea54b42f163e0c789351d98810a513895ab44b47020b"), true), nil); err != nil { if fmt.Sprintf("%v", err) != "hash 3fcb7c0d4569fddc89cbea54b42f163e0c789351d98810a513895ab44b47020b is not currently canonical" { t.Errorf("wrong error: %v", err) } } } func TestEthCallToPrunedBlock(t *testing.T) { pruneTo := uint64(3) ethCallBlockNumber := rpc.BlockNumber(2) m, bankAddress, contractAddress := chainWithDeployedContract(t) doPrune(t, m.DB, pruneTo) api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New()) callData := hexutil.MustDecode("0x2e64cec1") callDataBytes := hexutility.Bytes(callData) if _, err := api.Call(context.Background(), ethapi.CallArgs{ From: &bankAddress, To: &contractAddress, Data: &callDataBytes, }, rpc.BlockNumberOrHashWithNumber(ethCallBlockNumber), nil); err != nil { t.Errorf("unexpected error: %v", err) } } func TestGetProof(t *testing.T) { var maxGetProofRewindBlockCount = 1 // Note, this is unsafe for parallel tests, but, this test is the only consumer for now m, bankAddr, contractAddr := chainWithDeployedContract(t) if m.HistoryV3 { t.Skip("not supported by Erigon3") } api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, maxGetProofRewindBlockCount, log.New()) key := func(b byte) libcommon.Hash { result := libcommon.Hash{} result[31] = b return result } tests := []struct { name string blockNum uint64 addr libcommon.Address storageKeys []libcommon.Hash stateVal uint64 expectedErr string }{ { name: "currentBlockNoState", addr: contractAddr, blockNum: 3, }, { name: "currentBlockEOA", addr: bankAddr, blockNum: 3, }, { name: "currentBlockNoAccount", addr: libcommon.HexToAddress("0xdeaddeaddeaddeaddeaddeaddeaddeaddeaddead0"), blockNum: 3, }, { name: "currentBlockWithState", addr: contractAddr, blockNum: 3, storageKeys: []libcommon.Hash{key(0), key(4), key(8), key(10)}, stateVal: 2, }, { name: "currentBlockWithMissingState", addr: contractAddr, storageKeys: []libcommon.Hash{libcommon.HexToHash("0xdeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddead")}, blockNum: 3, stateVal: 0, }, { name: "currentBlockEOAMissingState", addr: bankAddr, storageKeys: []libcommon.Hash{libcommon.HexToHash("0xdeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddead")}, blockNum: 3, stateVal: 0, }, { name: "currentBlockNoAccountMissingState", addr: libcommon.HexToAddress("0xdeaddeaddeaddeaddeaddeaddeaddeaddeaddead0"), storageKeys: []libcommon.Hash{libcommon.HexToHash("0xdeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddeaddead")}, blockNum: 3, stateVal: 0, }, { name: "olderBlockWithState", addr: contractAddr, blockNum: 2, storageKeys: []libcommon.Hash{key(1), key(5), key(9), key(13)}, stateVal: 1, }, { name: "tooOldBlock", addr: contractAddr, blockNum: 1, expectedErr: "requested block is too old, block must be within 1 blocks of the head block number (currently 3)", }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { proof, err := api.GetProof( context.Background(), tt.addr, tt.storageKeys, rpc.BlockNumberOrHashWithNumber(rpc.BlockNumber(tt.blockNum)), ) if tt.expectedErr != "" { require.EqualError(t, err, tt.expectedErr) require.Nil(t, proof) return } require.NoError(t, err) require.NotNil(t, proof) tx, err := m.DB.BeginRo(context.Background()) assert.NoError(t, err) defer tx.Rollback() header, err := api.headerByRPCNumber(rpc.BlockNumber(tt.blockNum), tx) require.NoError(t, err) require.Equal(t, tt.addr, proof.Address) err = trie.VerifyAccountProof(header.Root, proof) require.NoError(t, err) require.Equal(t, len(tt.storageKeys), len(proof.StorageProof)) for _, storageKey := range tt.storageKeys { found := false for _, storageProof := range proof.StorageProof { if storageProof.Key != storageKey { continue } found = true require.Equal(t, tt.stateVal, (*big.Int)(storageProof.Value).Uint64()) err = trie.VerifyStorageProof(proof.StorageHash, storageProof) require.NoError(t, err) } require.True(t, found, "did not find storage proof for key=%x", storageKey) } }) } } func TestGetBlockByTimestampLatestTime(t *testing.T) { ctx := context.Background() m, _, _ := rpcdaemontest.CreateTestSentry(t) tx, err := m.DB.BeginRo(ctx) if err != nil { t.Errorf("fail at beginning tx") } defer tx.Rollback() api := NewErigonAPI(newBaseApiForTest(m), m.DB, nil) latestBlock, err := m.BlockReader.CurrentBlock(tx) require.NoError(t, err) response, err := ethapi.RPCMarshalBlockDeprecated(latestBlock, true, false) if err != nil { t.Error("couldn't get the rpc marshal block") } if err == nil && rpc.BlockNumber(latestBlock.NumberU64()) == rpc.PendingBlockNumber { // Pending blocks need to nil out a few fields for _, field := range []string{"hash", "nonce", "miner"} { response[field] = nil } } block, err := api.GetBlockByTimestamp(ctx, rpc.Timestamp(latestBlock.Header().Time), false) if err != nil { t.Errorf("couldn't retrieve block %v", err) } if block["timestamp"] != response["timestamp"] || block["hash"] != response["hash"] { t.Errorf("Retrieved the wrong block.\nexpected block hash: %s expected timestamp: %d\nblock hash retrieved: %s timestamp retrieved: %d", response["hash"], response["timestamp"], block["hash"], block["timestamp"]) } } func TestGetBlockByTimestampOldestTime(t *testing.T) { ctx := context.Background() m, _, _ := rpcdaemontest.CreateTestSentry(t) tx, err := m.DB.BeginRo(ctx) if err != nil { t.Errorf("failed at beginning tx") } defer tx.Rollback() api := NewErigonAPI(newBaseApiForTest(m), m.DB, nil) oldestBlock, err := m.BlockReader.BlockByNumber(m.Ctx, tx, 0) if err != nil { t.Error("couldn't retrieve oldest block") } response, err := ethapi.RPCMarshalBlockDeprecated(oldestBlock, true, false) if err != nil { t.Error("couldn't get the rpc marshal block") } if err == nil && rpc.BlockNumber(oldestBlock.NumberU64()) == rpc.PendingBlockNumber { // Pending blocks need to nil out a few fields for _, field := range []string{"hash", "nonce", "miner"} { response[field] = nil } } block, err := api.GetBlockByTimestamp(ctx, rpc.Timestamp(oldestBlock.Header().Time), false) if err != nil { t.Errorf("couldn't retrieve block %v", err) } if block["timestamp"] != response["timestamp"] || block["hash"] != response["hash"] { t.Errorf("Retrieved the wrong block.\nexpected block hash: %s expected timestamp: %d\nblock hash retrieved: %s timestamp retrieved: %d", response["hash"], response["timestamp"], block["hash"], block["timestamp"]) } } func TestGetBlockByTimeHigherThanLatestBlock(t *testing.T) { ctx := context.Background() m, _, _ := rpcdaemontest.CreateTestSentry(t) tx, err := m.DB.BeginRo(ctx) if err != nil { t.Errorf("fail at beginning tx") } defer tx.Rollback() api := NewErigonAPI(newBaseApiForTest(m), m.DB, nil) latestBlock, err := m.BlockReader.CurrentBlock(tx) require.NoError(t, err) response, err := ethapi.RPCMarshalBlockDeprecated(latestBlock, true, false) if err != nil { t.Error("couldn't get the rpc marshal block") } if err == nil && rpc.BlockNumber(latestBlock.NumberU64()) == rpc.PendingBlockNumber { // Pending blocks need to nil out a few fields for _, field := range []string{"hash", "nonce", "miner"} { response[field] = nil } } block, err := api.GetBlockByTimestamp(ctx, rpc.Timestamp(latestBlock.Header().Time+999999999999), false) if err != nil { t.Errorf("couldn't retrieve block %v", err) } if block["timestamp"] != response["timestamp"] || block["hash"] != response["hash"] { t.Errorf("Retrieved the wrong block.\nexpected block hash: %s expected timestamp: %d\nblock hash retrieved: %s timestamp retrieved: %d", response["hash"], response["timestamp"], block["hash"], block["timestamp"]) } } func TestGetBlockByTimeMiddle(t *testing.T) { ctx := context.Background() m, _, _ := rpcdaemontest.CreateTestSentry(t) tx, err := m.DB.BeginRo(ctx) if err != nil { t.Errorf("fail at beginning tx") } defer tx.Rollback() api := NewErigonAPI(newBaseApiForTest(m), m.DB, nil) currentHeader := rawdb.ReadCurrentHeader(tx) oldestHeader, err := api._blockReader.HeaderByNumber(ctx, tx, 0) if err != nil { t.Errorf("error getting the oldest header %s", err) } if oldestHeader == nil { t.Error("couldn't find oldest header") } middleNumber := (currentHeader.Number.Uint64() + oldestHeader.Number.Uint64()) / 2 middleBlock, err := m.BlockReader.BlockByNumber(m.Ctx, tx, middleNumber) if err != nil { t.Error("couldn't retrieve middle block") } response, err := ethapi.RPCMarshalBlockDeprecated(middleBlock, true, false) if err != nil { t.Error("couldn't get the rpc marshal block") } if err == nil && rpc.BlockNumber(middleBlock.NumberU64()) == rpc.PendingBlockNumber { // Pending blocks need to nil out a few fields for _, field := range []string{"hash", "nonce", "miner"} { response[field] = nil } } block, err := api.GetBlockByTimestamp(ctx, rpc.Timestamp(middleBlock.Header().Time), false) if err != nil { t.Errorf("couldn't retrieve block %v", err) } if block["timestamp"] != response["timestamp"] || block["hash"] != response["hash"] { t.Errorf("Retrieved the wrong block.\nexpected block hash: %s expected timestamp: %d\nblock hash retrieved: %s timestamp retrieved: %d", response["hash"], response["timestamp"], block["hash"], block["timestamp"]) } } func TestGetBlockByTimestamp(t *testing.T) { ctx := context.Background() m, _, _ := rpcdaemontest.CreateTestSentry(t) tx, err := m.DB.BeginRo(ctx) if err != nil { t.Errorf("fail at beginning tx") } defer tx.Rollback() api := NewErigonAPI(newBaseApiForTest(m), m.DB, nil) highestBlockNumber := rawdb.ReadCurrentHeader(tx).Number pickedBlock, err := m.BlockReader.BlockByNumber(m.Ctx, tx, highestBlockNumber.Uint64()/3) if err != nil { t.Errorf("couldn't get block %v", pickedBlock.Number()) } if pickedBlock == nil { t.Error("couldn't retrieve picked block") } response, err := ethapi.RPCMarshalBlockDeprecated(pickedBlock, true, false) if err != nil { t.Error("couldn't get the rpc marshal block") } if err == nil && rpc.BlockNumber(pickedBlock.NumberU64()) == rpc.PendingBlockNumber { // Pending blocks need to nil out a few fields for _, field := range []string{"hash", "nonce", "miner"} { response[field] = nil } } block, err := api.GetBlockByTimestamp(ctx, rpc.Timestamp(pickedBlock.Header().Time), false) if err != nil { t.Errorf("couldn't retrieve block %v", err) } if block["timestamp"] != response["timestamp"] || block["hash"] != response["hash"] { t.Errorf("Retrieved the wrong block.\nexpected block hash: %s expected timestamp: %d\nblock hash retrieved: %s timestamp retrieved: %d", response["hash"], response["timestamp"], block["hash"], block["timestamp"]) } } // contractHexString is the output of compiling the following solidity contract: // // pragma solidity ^0.8.0; // // contract Box { // uint256 private _value0x0; // uint256 private _value0x1; // uint256 private _value0x2; // uint256 private _value0x3; // uint256 private _value0x4; // uint256 private _value0x5; // uint256 private _value0x6; // uint256 private _value0x7; // uint256 private _value0x8; // uint256 private _value0x9; // uint256 private _value0xa; // uint256 private _value0xb; // uint256 private _value0xc; // uint256 private _value0xd; // uint256 private _value0xe; // uint256 private _value0xf; // uint256 private _value0x10; // // // Emitted when the stored value changes // event ValueChanged(uint256 value); // // // Stores a new value in the contract // function store(uint256 value) public { // _value0x0 = value; // _value0x1 = value; // _value0x2 = value; // _value0x3 = value; // _value0x4 = value; // _value0x5 = value; // _value0x6 = value; // _value0x7 = value; // _value0x8 = value; // _value0x9 = value; // _value0xa = value; // _value0xb = value; // _value0xc = value; // _value0xd = value; // _value0xe = value; // _value0xf = value; // _value0x10 = value; // emit ValueChanged(value); // } // // // Reads the last stored value // function retrieve() public view returns (uint256) { // return _value0x0; // } // } // // You may produce this hex string by saving the contract into a file // Box.sol and invoking // // solc Box.sol --bin --abi --optimize // // This contract is a slight modification of Box.sol to use more storage nodes // and ensure the contract storage will contain at least 1 non-leaf node (by // storing 17 values). const contractHexString = "0x608060405234801561001057600080fd5b5061013f806100206000396000f3fe608060405234801561001057600080fd5b50600436106100365760003560e01c80632e64cec11461003b5780636057361d14610050575b600080fd5b60005460405190815260200160405180910390f35b61006361005e3660046100f0565b610065565b005b6000819055600181905560028190556003819055600481905560058190556006819055600781905560088190556009819055600a819055600b819055600c819055600d819055600e819055600f81905560108190556040518181527f93fe6d397c74fdf1402a8b72e47b68512f0510d7b98a4bc4cbdf6ac7108b3c599060200160405180910390a150565b60006020828403121561010257600080fd5b503591905056fea2646970667358221220031e17f1bd1d1dcbee088287a905b152410b180064c149763590a0bbc516d95e64736f6c63430008130033" var contractFuncSelector = crypto.Keccak256([]byte("store(uint256)"))[:4] // contractInvocationData returns data suitable for invoking the 'store' // function of the contract in contractHexString, note func contractInvocationData(val byte) []byte { return hexutil.MustDecode(fmt.Sprintf("0x%x00000000000000000000000000000000000000000000000000000000000000%02x", contractFuncSelector, val)) } func chainWithDeployedContract(t *testing.T) (*mock.MockSentry, libcommon.Address, libcommon.Address) { var ( signer = types.LatestSignerForChainID(nil) bankKey, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291") bankAddress = crypto.PubkeyToAddress(bankKey.PublicKey) bankFunds = big.NewInt(1e9) contract = hexutil.MustDecode(contractHexString) gspec = &types.Genesis{ Config: params.TestChainConfig, Alloc: types.GenesisAlloc{bankAddress: {Balance: bankFunds}}, } ) m := mock.MockWithGenesis(t, gspec, bankKey, false) db := m.DB var contractAddr libcommon.Address chain, err := core.GenerateChain(m.ChainConfig, m.Genesis, m.Engine, m.DB, 3, func(i int, block *core.BlockGen) { nonce := block.TxNonce(bankAddress) switch i { case 0: tx, err := types.SignTx(types.NewContractCreation(nonce, new(uint256.Int), 1e6, new(uint256.Int), contract), *signer, bankKey) assert.NoError(t, err) block.AddTx(tx) contractAddr = crypto.CreateAddress(bankAddress, nonce) case 1: txn, err := types.SignTx(types.NewTransaction(nonce, contractAddr, new(uint256.Int), 900000, new(uint256.Int), contractInvocationData(1)), *signer, bankKey) assert.NoError(t, err) block.AddTx(txn) case 2: txn, err := types.SignTx(types.NewTransaction(nonce, contractAddr, new(uint256.Int), 900000, new(uint256.Int), contractInvocationData(2)), *signer, bankKey) assert.NoError(t, err) block.AddTx(txn) } }) if err != nil { t.Fatalf("generate blocks: %v", err) } err = m.InsertChain(chain) assert.NoError(t, err) tx, err := db.BeginRo(context.Background()) if err != nil { t.Fatalf("read only db tx to read state: %v", err) } defer tx.Rollback() stateReader, err := rpchelper.CreateHistoryStateReader(tx, 1, 0, m.HistoryV3, "") assert.NoError(t, err) st := state.New(stateReader) assert.NoError(t, err) assert.False(t, st.Exist(contractAddr), "Contract should not exist at block #1") stateReader, err = rpchelper.CreateHistoryStateReader(tx, 2, 0, m.HistoryV3, "") assert.NoError(t, err) st = state.New(stateReader) assert.NoError(t, err) assert.True(t, st.Exist(contractAddr), "Contract should exist at block #2") return m, bankAddress, contractAddr } func doPrune(t *testing.T, db kv.RwDB, pruneTo uint64) { ctx := context.Background() tx, err := db.BeginRw(ctx) assert.NoError(t, err) logEvery := time.NewTicker(20 * time.Second) err = rawdb.PruneTableDupSort(tx, kv.AccountChangeSet, "", pruneTo, logEvery, ctx) assert.NoError(t, err) err = rawdb.PruneTableDupSort(tx, kv.StorageChangeSet, "", pruneTo, logEvery, ctx) assert.NoError(t, err) err = rawdb.PruneTable(tx, kv.Receipts, pruneTo, ctx, math.MaxInt32) assert.NoError(t, err) err = rawdb.PruneTable(tx, kv.Log, pruneTo, ctx, math.MaxInt32) assert.NoError(t, err) err = rawdb.PruneTableDupSort(tx, kv.CallTraceSet, "", pruneTo, logEvery, ctx) assert.NoError(t, err) err = tx.Commit() assert.NoError(t, err) }