2023-07-08 17:01:26 +00:00
|
|
|
package jsonrpc
|
2021-02-21 08:38:00 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2021-09-28 02:25:56 +00:00
|
|
|
"fmt"
|
2022-09-18 10:41:01 +00:00
|
|
|
"testing"
|
|
|
|
|
2023-11-23 05:00:41 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/common/hexutil"
|
|
|
|
|
2022-07-15 14:04:23 +00:00
|
|
|
"github.com/holiman/uint256"
|
2023-01-27 04:39:34 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
2023-01-13 18:12:18 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
|
2023-05-24 11:34:36 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv/kvcache"
|
2021-09-28 02:25:56 +00:00
|
|
|
"github.com/ledgerwatch/erigon/core"
|
|
|
|
"github.com/ledgerwatch/erigon/rpc"
|
2022-09-17 06:25:27 +00:00
|
|
|
"github.com/ledgerwatch/erigon/rpc/rpccfg"
|
2022-10-25 02:58:25 +00:00
|
|
|
"github.com/ledgerwatch/erigon/turbo/adapter/ethapi"
|
2023-08-05 21:33:10 +00:00
|
|
|
"github.com/ledgerwatch/erigon/turbo/stages/mock"
|
2023-01-13 18:12:18 +00:00
|
|
|
|
2021-06-29 10:00:22 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cmd/rpcdaemon/rpcdaemontest"
|
2023-05-08 16:52:31 +00:00
|
|
|
"github.com/ledgerwatch/log/v3"
|
2021-02-21 08:38:00 +00:00
|
|
|
)
|
|
|
|
|
2023-08-05 21:33:10 +00:00
|
|
|
func newBaseApiForTest(m *mock.MockSentry) *BaseAPI {
|
2023-06-15 06:11:51 +00:00
|
|
|
agg := m.HistoryV3Components()
|
|
|
|
stateCache := kvcache.New(kvcache.DefaultCoherentConfig)
|
|
|
|
return NewBaseApi(nil, stateCache, m.BlockReader, agg, false, rpccfg.DefaultEvmCallTimeout, m.Engine, m.Dirs)
|
|
|
|
}
|
|
|
|
|
2022-07-15 14:04:23 +00:00
|
|
|
func TestGetBalanceChangesInBlock(t *testing.T) {
|
|
|
|
assert := assert.New(t)
|
|
|
|
myBlockNum := rpc.BlockNumberOrHashWithNumber(0)
|
2022-09-05 14:31:00 +00:00
|
|
|
m, _, _ := rpcdaemontest.CreateTestSentry(t)
|
|
|
|
db := m.DB
|
2023-06-15 06:11:51 +00:00
|
|
|
api := NewErigonAPI(newBaseApiForTest(m), db, nil)
|
2022-07-15 14:04:23 +00:00
|
|
|
balances, err := api.GetBalanceChangesInBlock(context.Background(), myBlockNum)
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("calling GetBalanceChangesInBlock resulted in an error: %v", err)
|
|
|
|
}
|
2023-01-27 04:39:34 +00:00
|
|
|
expected := map[common.Address]*hexutil.Big{
|
|
|
|
common.HexToAddress("0x0D3ab14BBaD3D99F4203bd7a11aCB94882050E7e"): (*hexutil.Big)(uint256.NewInt(200000000000000000).ToBig()),
|
|
|
|
common.HexToAddress("0x703c4b2bD70c169f5717101CaeE543299Fc946C7"): (*hexutil.Big)(uint256.NewInt(300000000000000000).ToBig()),
|
|
|
|
common.HexToAddress("0x71562b71999873DB5b286dF957af199Ec94617F7"): (*hexutil.Big)(uint256.NewInt(9000000000000000000).ToBig()),
|
2022-07-15 14:04:23 +00:00
|
|
|
}
|
|
|
|
assert.Equal(len(expected), len(balances))
|
|
|
|
for i := range balances {
|
|
|
|
assert.Contains(expected, i, "%s is not expected to be present in the output.", i)
|
|
|
|
assert.Equal(balances[i], expected[i], "the value for %s is expected to be %v, but got %v.", i, expected[i], balances[i])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-21 08:38:00 +00:00
|
|
|
func TestGetTransactionReceipt(t *testing.T) {
|
2022-09-18 10:41:01 +00:00
|
|
|
m, _, _ := rpcdaemontest.CreateTestSentry(t)
|
|
|
|
db := m.DB
|
2022-09-26 03:54:42 +00:00
|
|
|
agg := m.HistoryV3Components()
|
2021-09-29 01:36:25 +00:00
|
|
|
stateCache := kvcache.New(kvcache.DefaultCoherentConfig)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(NewBaseApi(nil, stateCache, m.BlockReader, agg, false, rpccfg.DefaultEvmCallTimeout, m.Engine, m.Dirs), db, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2021-02-21 08:38:00 +00:00
|
|
|
// Call GetTransactionReceipt for transaction which is not in the database
|
2023-01-27 04:39:34 +00:00
|
|
|
if _, err := api.GetTransactionReceipt(context.Background(), common.Hash{}); err != nil {
|
2021-02-21 08:38:00 +00:00
|
|
|
t.Errorf("calling GetTransactionReceipt with empty hash: %v", err)
|
|
|
|
}
|
|
|
|
}
|
2021-04-23 19:48:00 +00:00
|
|
|
|
|
|
|
func TestGetTransactionReceiptUnprotected(t *testing.T) {
|
2022-09-18 10:41:01 +00:00
|
|
|
m, _, _ := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2021-04-23 19:48:00 +00:00
|
|
|
// Call GetTransactionReceipt for un-protected transaction
|
2023-01-27 04:39:34 +00:00
|
|
|
if _, err := api.GetTransactionReceipt(context.Background(), common.HexToHash("0x3f3cb8a0e13ed2481f97f53f7095b9cbc78b6ffb779f2d3e565146371a8830ea")); err != nil {
|
2021-04-23 19:48:00 +00:00
|
|
|
t.Errorf("calling GetTransactionReceipt for unprotected tx: %v", err)
|
|
|
|
}
|
|
|
|
}
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
// EIP-1898 test cases
|
|
|
|
|
|
|
|
func TestGetStorageAt_ByBlockNumber_WithRequireCanonicalDefault(t *testing.T) {
|
|
|
|
assert := assert.New(t)
|
2022-09-18 10:41:01 +00:00
|
|
|
m, _, _ := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2023-01-27 04:39:34 +00:00
|
|
|
addr := common.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7")
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
result, err := api.GetStorageAt(context.Background(), addr, "0x0", rpc.BlockNumberOrHashWithNumber(0))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("calling GetStorageAt: %v", err)
|
|
|
|
}
|
|
|
|
|
2023-01-27 04:39:34 +00:00
|
|
|
assert.Equal(common.HexToHash("0x0").String(), result)
|
2021-09-28 02:25:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetStorageAt_ByBlockHash_WithRequireCanonicalDefault(t *testing.T) {
|
|
|
|
assert := assert.New(t)
|
|
|
|
m, _, _ := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2023-01-27 04:39:34 +00:00
|
|
|
addr := common.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7")
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
result, err := api.GetStorageAt(context.Background(), addr, "0x0", rpc.BlockNumberOrHashWithHash(m.Genesis.Hash(), false))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("calling GetStorageAt: %v", err)
|
|
|
|
}
|
|
|
|
|
2023-01-27 04:39:34 +00:00
|
|
|
assert.Equal(common.HexToHash("0x0").String(), result)
|
2021-09-28 02:25:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetStorageAt_ByBlockHash_WithRequireCanonicalTrue(t *testing.T) {
|
|
|
|
assert := assert.New(t)
|
|
|
|
m, _, _ := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2023-01-27 04:39:34 +00:00
|
|
|
addr := common.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7")
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
result, err := api.GetStorageAt(context.Background(), addr, "0x0", rpc.BlockNumberOrHashWithHash(m.Genesis.Hash(), true))
|
|
|
|
if err != nil {
|
|
|
|
t.Errorf("calling GetStorageAt: %v", err)
|
|
|
|
}
|
|
|
|
|
2023-01-27 04:39:34 +00:00
|
|
|
assert.Equal(common.HexToHash("0x0").String(), result)
|
2021-09-28 02:25:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetStorageAt_ByBlockHash_WithRequireCanonicalDefault_BlockNotFoundError(t *testing.T) {
|
|
|
|
m, _, _ := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2023-01-27 04:39:34 +00:00
|
|
|
addr := common.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7")
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
offChain, err := core.GenerateChain(m.ChainConfig, m.Genesis, m.Engine, m.DB, 1, func(i int, block *core.BlockGen) {
|
2023-06-23 03:07:42 +00:00
|
|
|
})
|
2021-09-28 02:25:56 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
offChainBlock := offChain.Blocks[0]
|
|
|
|
|
|
|
|
if _, err := api.GetStorageAt(context.Background(), addr, "0x0", rpc.BlockNumberOrHashWithHash(offChainBlock.Hash(), false)); err != nil {
|
|
|
|
if fmt.Sprintf("%v", err) != fmt.Sprintf("block %s not found", offChainBlock.Hash().String()[2:]) {
|
|
|
|
t.Errorf("wrong error: %v", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Error("error expected")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetStorageAt_ByBlockHash_WithRequireCanonicalTrue_BlockNotFoundError(t *testing.T) {
|
|
|
|
m, _, _ := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2023-01-27 04:39:34 +00:00
|
|
|
addr := common.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7")
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
offChain, err := core.GenerateChain(m.ChainConfig, m.Genesis, m.Engine, m.DB, 1, func(i int, block *core.BlockGen) {
|
2023-06-23 03:07:42 +00:00
|
|
|
})
|
2021-09-28 02:25:56 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
offChainBlock := offChain.Blocks[0]
|
|
|
|
|
|
|
|
if _, err := api.GetStorageAt(context.Background(), addr, "0x0", rpc.BlockNumberOrHashWithHash(offChainBlock.Hash(), true)); err != nil {
|
|
|
|
if fmt.Sprintf("%v", err) != fmt.Sprintf("block %s not found", offChainBlock.Hash().String()[2:]) {
|
|
|
|
t.Errorf("wrong error: %v", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Error("error expected")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetStorageAt_ByBlockHash_WithRequireCanonicalDefault_NonCanonicalBlock(t *testing.T) {
|
|
|
|
assert := assert.New(t)
|
|
|
|
m, _, orphanedChain := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2023-01-27 04:39:34 +00:00
|
|
|
addr := common.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7")
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
orphanedBlock := orphanedChain[0].Blocks[0]
|
|
|
|
|
|
|
|
result, err := api.GetStorageAt(context.Background(), addr, "0x0", rpc.BlockNumberOrHashWithHash(orphanedBlock.Hash(), false))
|
|
|
|
if err != nil {
|
2022-02-23 23:42:14 +00:00
|
|
|
if fmt.Sprintf("%v", err) != fmt.Sprintf("hash %s is not currently canonical", orphanedBlock.Hash().String()[2:]) {
|
|
|
|
t.Errorf("wrong error: %v", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Error("error expected")
|
2021-09-28 02:25:56 +00:00
|
|
|
}
|
|
|
|
|
2023-01-27 04:39:34 +00:00
|
|
|
assert.Equal(common.HexToHash("0x0").String(), result)
|
2021-09-28 02:25:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestGetStorageAt_ByBlockHash_WithRequireCanonicalTrue_NonCanonicalBlock(t *testing.T) {
|
|
|
|
m, _, orphanedChain := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2023-01-27 04:39:34 +00:00
|
|
|
addr := common.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7")
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
orphanedBlock := orphanedChain[0].Blocks[0]
|
|
|
|
|
|
|
|
if _, err := api.GetStorageAt(context.Background(), addr, "0x0", rpc.BlockNumberOrHashWithHash(orphanedBlock.Hash(), true)); err != nil {
|
|
|
|
if fmt.Sprintf("%v", err) != fmt.Sprintf("hash %s is not currently canonical", orphanedBlock.Hash().String()[2:]) {
|
|
|
|
t.Errorf("wrong error: %v", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Error("error expected")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCall_ByBlockHash_WithRequireCanonicalDefault_NonCanonicalBlock(t *testing.T) {
|
|
|
|
m, _, orphanedChain := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2023-01-27 04:39:34 +00:00
|
|
|
from := common.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7")
|
|
|
|
to := common.HexToAddress("0x0d3ab14bbad3d99f4203bd7a11acb94882050e7e")
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
orphanedBlock := orphanedChain[0].Blocks[0]
|
|
|
|
|
|
|
|
if _, err := api.Call(context.Background(), ethapi.CallArgs{
|
|
|
|
From: &from,
|
|
|
|
To: &to,
|
|
|
|
}, rpc.BlockNumberOrHashWithHash(orphanedBlock.Hash(), false), nil); err != nil {
|
|
|
|
if fmt.Sprintf("%v", err) != fmt.Sprintf("hash %s is not currently canonical", orphanedBlock.Hash().String()[2:]) {
|
|
|
|
/* Not sure. Here https://github.com/ethereum/EIPs/blob/master/EIPS/eip-1898.md it is not explicitly said that
|
|
|
|
eth_call should only work with canonical blocks.
|
|
|
|
But since there is no point in changing the state of non-canonical block, it ignores RequireCanonical. */
|
|
|
|
t.Errorf("wrong error: %v", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Error("error expected")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCall_ByBlockHash_WithRequireCanonicalTrue_NonCanonicalBlock(t *testing.T) {
|
|
|
|
m, _, orphanedChain := rpcdaemontest.CreateTestSentry(t)
|
2023-11-13 16:11:35 +00:00
|
|
|
api := NewEthAPI(newBaseApiForTest(m), m.DB, nil, nil, nil, 5000000, 100_000, false, 100_000, log.New())
|
2023-01-27 04:39:34 +00:00
|
|
|
from := common.HexToAddress("0x71562b71999873db5b286df957af199ec94617f7")
|
|
|
|
to := common.HexToAddress("0x0d3ab14bbad3d99f4203bd7a11acb94882050e7e")
|
2021-09-28 02:25:56 +00:00
|
|
|
|
|
|
|
orphanedBlock := orphanedChain[0].Blocks[0]
|
|
|
|
|
|
|
|
if _, err := api.Call(context.Background(), ethapi.CallArgs{
|
|
|
|
From: &from,
|
|
|
|
To: &to,
|
|
|
|
}, rpc.BlockNumberOrHashWithHash(orphanedBlock.Hash(), true), nil); err != nil {
|
|
|
|
if fmt.Sprintf("%v", err) != fmt.Sprintf("hash %s is not currently canonical", orphanedBlock.Hash().String()[2:]) {
|
|
|
|
t.Errorf("wrong error: %v", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
t.Error("error expected")
|
|
|
|
}
|
|
|
|
}
|