package rpchelper import ( "fmt" "github.com/ledgerwatch/turbo-geth/common" "github.com/ledgerwatch/turbo-geth/core/rawdb" "github.com/ledgerwatch/turbo-geth/core/types/accounts" "github.com/ledgerwatch/turbo-geth/eth/stagedsync/stages" "github.com/ledgerwatch/turbo-geth/ethdb" "github.com/ledgerwatch/turbo-geth/rpc" "github.com/ledgerwatch/turbo-geth/turbo/adapter" ) func GetBlockNumber(blockNrOrHash rpc.BlockNumberOrHash, dbReader rawdb.DatabaseReader) (uint64, common.Hash, error) { var blockNumber uint64 var err error hash, ok := blockNrOrHash.Hash() if !ok { number := *blockNrOrHash.BlockNumber if number == rpc.LatestBlockNumber { blockNumber, _, err = stages.GetStageProgress(dbReader, stages.Execution) if err != nil { return 0, common.Hash{}, fmt.Errorf("getting latest block number: %v", err) } } else if number == rpc.PendingBlockNumber || number == rpc.EarliestBlockNumber { return 0, common.Hash{}, fmt.Errorf("pending and earliest blocks are not supported") } else { blockNumber = uint64(number.Int64()) } hash, err = GetHashByNumber(blockNumber, blockNrOrHash.RequireCanonical, dbReader) if err != nil { return 0, common.Hash{}, err } } else { block := rawdb.ReadBlockByHash(dbReader, hash) if block == nil { return 0, common.Hash{}, fmt.Errorf("block %x not found", hash) } blockNumber = block.NumberU64() if blockNrOrHash.RequireCanonical && rawdb.ReadCanonicalHash(dbReader, blockNumber) != hash { return 0, common.Hash{}, fmt.Errorf("hash %q is not currently canonical", hash.String()) } } return blockNumber, hash, nil } func GetAccount(chainKV ethdb.KV, blockNumber uint64, address common.Address) (*accounts.Account, error) { reader := adapter.NewStateReader(chainKV, blockNumber) return reader.ReadAccountData(address) } func GetHashByNumber(blockNumber uint64, requireCanonical bool, dbReader rawdb.DatabaseReader) (common.Hash, error) { if requireCanonical { return rawdb.ReadCanonicalHash(dbReader, blockNumber), nil } block := rawdb.ReadBlockByNumber(dbReader, blockNumber) if block == nil { return common.Hash{}, fmt.Errorf("block %d not found", blockNumber) } return block.Hash(), nil }