feat(internal/ethapi): add eth_getStorageValues method #32591 (#2116)

This commit is contained in:
Daniel Liu 2026-03-06 13:52:36 +08:00 committed by GitHub
parent 45aadcd311
commit 436166e447
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 158 additions and 0 deletions

View file

@ -55,6 +55,7 @@ import (
const (
defaultGasPrice = 50 * params.Shannon
// statuses of candidates
statusMasternode = "MASTERNODE"
statusSlashed = "SLASHED"
@ -66,6 +67,10 @@ const (
fieldEpoch = "epoch"
)
// maxGetStorageSlots is the maximum total number of storage slots that can
// be requested in a single eth_getStorageValues call.
const maxGetStorageSlots = 1024
var errEmptyHeader = errors.New("empty header")
// EthereumAPI provides an API to access Ethereum related information.
@ -517,6 +522,41 @@ func (api *BlockChainAPI) GetStorageAt(ctx context.Context, address common.Addre
return res[:], state.Error()
}
// GetStorageValues returns multiple storage slot values for multiple accounts
// at the given block.
func (api *BlockChainAPI) GetStorageValues(ctx context.Context, requests map[common.Address][]common.Hash, blockNrOrHash rpc.BlockNumberOrHash) (map[common.Address][]hexutil.Bytes, error) {
// Count total slots requested.
var totalSlots int
for _, keys := range requests {
totalSlots += len(keys)
if totalSlots > maxGetStorageSlots {
return nil, &clientLimitExceededError{message: fmt.Sprintf("too many slots (max %d)", maxGetStorageSlots)}
}
}
if totalSlots == 0 {
return nil, &invalidParamsError{message: "empty request"}
}
state, _, err := api.b.StateAndHeaderByNumberOrHash(ctx, blockNrOrHash)
if state == nil || err != nil {
return nil, err
}
result := make(map[common.Address][]hexutil.Bytes, len(requests))
for addr, keys := range requests {
vals := make([]hexutil.Bytes, len(keys))
for i, key := range keys {
v := state.GetState(addr, key)
vals[i] = v[:]
}
if err := state.Error(); err != nil {
return nil, err
}
result[addr] = vals
}
return result, nil
}
// GetBlockReceipts returns the block receipts for the given block hash or number or tag.
func (api *BlockChainAPI) GetBlockReceipts(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) ([]map[string]interface{}, error) {
block, err := api.b.BlockByNumberOrHash(ctx, blockNrOrHash)

View file

@ -17,15 +17,20 @@
package ethapi
import (
"context"
"encoding/json"
"hash"
"math/big"
"testing"
"github.com/XinFinOrg/XDPoSChain/common"
"github.com/XinFinOrg/XDPoSChain/core"
"github.com/XinFinOrg/XDPoSChain/core/rawdb"
"github.com/XinFinOrg/XDPoSChain/core/state"
"github.com/XinFinOrg/XDPoSChain/core/types"
"github.com/XinFinOrg/XDPoSChain/crypto/keccak"
"github.com/XinFinOrg/XDPoSChain/params"
"github.com/XinFinOrg/XDPoSChain/rpc"
"github.com/stretchr/testify/require"
)
@ -271,3 +276,110 @@ func TestRPCMarshalBlock(t *testing.T) {
require.JSONEqf(t, tc.want, string(out), "test %d", i)
}
}
type storageBackendMock struct {
*backendMock
stateDB *state.StateDB
header *types.Header
}
func (b *storageBackendMock) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) {
return b.stateDB, b.header, nil
}
func TestGetStorageValues(t *testing.T) {
t.Parallel()
var (
addr1 = common.HexToAddress("0x1111")
addr2 = common.HexToAddress("0x2222")
slot0 = common.Hash{}
slot1 = common.BigToHash(big.NewInt(1))
slot2 = common.BigToHash(big.NewInt(2))
val0 = common.BigToHash(big.NewInt(42))
val1 = common.BigToHash(big.NewInt(100))
val2 = common.BigToHash(big.NewInt(200))
genesis = &core.Genesis{
Config: params.MergedTestChainConfig,
Alloc: types.GenesisAlloc{
addr1: {
Balance: big.NewInt(params.Ether),
Storage: map[common.Hash]common.Hash{
slot0: val0,
slot1: val1,
},
},
addr2: {
Balance: big.NewInt(params.Ether),
Storage: map[common.Hash]common.Hash{
slot2: val2,
},
},
},
}
)
db := rawdb.NewMemoryDatabase()
block := genesis.MustCommit(db)
stateDB, err := state.New(block.Root(), state.NewDatabase(db))
if err != nil {
t.Fatalf("failed to create state db: %v", err)
}
backend := &storageBackendMock{
backendMock: newBackendMock(),
stateDB: stateDB,
header: block.Header(),
}
api := NewBlockChainAPI(backend, nil)
latest := rpc.BlockNumberOrHashWithNumber(rpc.LatestBlockNumber)
// Happy path: multiple addresses, multiple slots.
result, err := api.GetStorageValues(context.Background(), map[common.Address][]common.Hash{
addr1: {slot0, slot1},
addr2: {slot2},
}, latest)
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
if len(result) != 2 {
t.Fatalf("expected 2 addresses in result, got %d", len(result))
}
if got := common.BytesToHash(result[addr1][0]); got != val0 {
t.Errorf("addr1 slot0: want %x, got %x", val0, got)
}
if got := common.BytesToHash(result[addr1][1]); got != val1 {
t.Errorf("addr1 slot1: want %x, got %x", val1, got)
}
if got := common.BytesToHash(result[addr2][0]); got != val2 {
t.Errorf("addr2 slot2: want %x, got %x", val2, got)
}
// Missing slot returns zero.
result, err = api.GetStorageValues(context.Background(), map[common.Address][]common.Hash{
addr1: {common.HexToHash("0xff")},
}, latest)
if err != nil {
t.Fatalf("unexpected error: %v", err)
}
if got := common.BytesToHash(result[addr1][0]); got != (common.Hash{}) {
t.Errorf("missing slot: want zero, got %x", got)
}
// Empty request returns error.
_, err = api.GetStorageValues(context.Background(), map[common.Address][]common.Hash{}, latest)
if err == nil {
t.Fatal("expected error for empty request")
}
// Exceeding slot limit returns error.
tooMany := make([]common.Hash, maxGetStorageSlots+1)
for i := range tooMany {
tooMany[i] = common.BigToHash(big.NewInt(int64(i)))
}
_, err = api.GetStorageValues(context.Background(), map[common.Address][]common.Hash{
addr1: tooMany,
}, latest)
if err == nil {
t.Fatal("expected error for exceeding slot limit")
}
}

View file

@ -543,6 +543,12 @@ web3._extend({
params: 2,
inputFormatter: [web3._extend.formatters.inputAddressFormatter, web3._extend.formatters.inputBlockNumberFormatter]
}),
new web3._extend.Method({
name: 'getStorageValues',
call: 'eth_getStorageValues',
params: 2,
inputFormatter: [null, web3._extend.formatters.inputBlockNumberFormatter]
}),
new web3._extend.Method({
name: 'createAccessList',
call: 'eth_createAccessList',