2022-01-13 11:05:30 +00:00
|
|
|
package commands
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/binary"
|
|
|
|
"fmt"
|
|
|
|
|
2023-01-13 18:12:18 +00:00
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
2023-04-13 11:19:02 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/common/hexutility"
|
2022-11-20 03:58:20 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/common/length"
|
2022-01-13 11:05:30 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
2023-06-14 03:01:00 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv/order"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv/rawdbv3"
|
|
|
|
"github.com/ledgerwatch/erigon/common"
|
|
|
|
"github.com/ledgerwatch/erigon/core/rawdb"
|
2023-06-09 03:53:04 +00:00
|
|
|
"github.com/ledgerwatch/erigon/turbo/rpchelper"
|
2023-01-13 18:12:18 +00:00
|
|
|
|
2022-02-08 13:02:18 +00:00
|
|
|
"github.com/ledgerwatch/erigon/rpc"
|
2022-01-13 11:05:30 +00:00
|
|
|
)
|
|
|
|
|
2023-01-13 18:12:18 +00:00
|
|
|
var latestTag = libcommon.BytesToHash([]byte("latest"))
|
2022-02-08 13:02:18 +00:00
|
|
|
|
|
|
|
var ErrWrongTag = fmt.Errorf("listStorageKeys wrong block tag or number: must be '%s' ('latest')", latestTag)
|
|
|
|
|
2022-01-13 11:05:30 +00:00
|
|
|
// ParityAPI the interface for the parity_ RPC commands
|
|
|
|
type ParityAPI interface {
|
2023-04-13 11:19:02 +00:00
|
|
|
ListStorageKeys(ctx context.Context, account libcommon.Address, quantity int, offset *hexutility.Bytes, blockNumber rpc.BlockNumberOrHash) ([]hexutility.Bytes, error)
|
2022-01-13 11:05:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ParityAPIImpl data structure to store things needed for parity_ commands
|
|
|
|
type ParityAPIImpl struct {
|
2023-06-14 03:01:00 +00:00
|
|
|
*BaseAPI
|
2022-01-13 11:05:30 +00:00
|
|
|
db kv.RoDB
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewParityAPIImpl returns ParityAPIImpl instance
|
2023-06-14 03:01:00 +00:00
|
|
|
func NewParityAPIImpl(base *BaseAPI, db kv.RoDB) *ParityAPIImpl {
|
2022-01-13 11:05:30 +00:00
|
|
|
return &ParityAPIImpl{
|
2023-06-14 03:01:00 +00:00
|
|
|
BaseAPI: base,
|
|
|
|
db: db,
|
2022-01-13 11:05:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListStorageKeys implements parity_listStorageKeys. Returns all storage keys of the given address
|
2023-04-13 11:19:02 +00:00
|
|
|
func (api *ParityAPIImpl) ListStorageKeys(ctx context.Context, account libcommon.Address, quantity int, offset *hexutility.Bytes, blockNumberOrTag rpc.BlockNumberOrHash) ([]hexutility.Bytes, error) {
|
2022-02-08 13:02:18 +00:00
|
|
|
if err := api.checkBlockNumber(blockNumberOrTag); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-06-14 03:01:00 +00:00
|
|
|
keys := make([]hexutility.Bytes, 0)
|
2022-02-08 13:02:18 +00:00
|
|
|
|
2022-01-13 11:05:30 +00:00
|
|
|
tx, err := api.db.BeginRo(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("listStorageKeys cannot open tx: %w", err)
|
|
|
|
}
|
|
|
|
defer tx.Rollback()
|
2023-06-09 03:53:04 +00:00
|
|
|
a, err := rpchelper.NewLatestStateReader(tx).ReadAccountData(account)
|
2022-01-13 11:05:30 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
} else if a == nil {
|
|
|
|
return nil, fmt.Errorf("acc not found")
|
|
|
|
}
|
|
|
|
|
2023-06-14 03:01:00 +00:00
|
|
|
if api.historyV3(tx) {
|
|
|
|
bn := rawdb.ReadCurrentBlockNumber(tx)
|
|
|
|
minTxNum, err := rawdbv3.TxNums.Min(tx, *bn)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
from := account[:]
|
|
|
|
if offset != nil {
|
|
|
|
from = append(from, *offset...)
|
|
|
|
}
|
|
|
|
to, _ := kv.NextSubtree(account[:])
|
|
|
|
r, err := tx.(kv.TemporalTx).DomainRange(kv.StorageDomain, from, to, minTxNum, order.Asc, quantity)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for r.HasNext() {
|
|
|
|
k, _, err := r.Next()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
keys = append(keys, common.CopyBytes(k[20:]))
|
|
|
|
}
|
|
|
|
return keys, nil
|
|
|
|
}
|
2022-01-13 11:05:30 +00:00
|
|
|
b := make([]byte, 8)
|
|
|
|
binary.BigEndian.PutUint64(b, a.GetIncarnation())
|
|
|
|
seekBytes := append(account.Bytes(), b...)
|
|
|
|
|
|
|
|
c, err := tx.CursorDupSort(kv.PlainState)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer c.Close()
|
|
|
|
var v []byte
|
|
|
|
var seekVal []byte
|
|
|
|
if offset != nil {
|
|
|
|
seekVal = *offset
|
|
|
|
}
|
|
|
|
|
|
|
|
for v, err = c.SeekBothRange(seekBytes, seekVal); v != nil && len(keys) != quantity && err == nil; _, v, err = c.NextDup() {
|
2022-11-20 03:58:20 +00:00
|
|
|
if len(v) > length.Hash {
|
|
|
|
keys = append(keys, v[:length.Hash])
|
2022-01-13 11:05:30 +00:00
|
|
|
} else {
|
|
|
|
keys = append(keys, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return keys, nil
|
|
|
|
}
|
2022-02-08 13:02:18 +00:00
|
|
|
|
|
|
|
func (api *ParityAPIImpl) checkBlockNumber(blockNumber rpc.BlockNumberOrHash) error {
|
|
|
|
num, isNum := blockNumber.Number()
|
|
|
|
if isNum && rpc.LatestBlockNumber == num {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return ErrWrongTag
|
|
|
|
}
|