564 lines
18 KiB
Go
564 lines
18 KiB
Go
// VulcanizeDB
|
|
// Copyright © 2019 Vulcanize
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Affero General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package eth
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"math/big"
|
|
|
|
"github.com/ethereum/go-ethereum/core/state"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/common/hexutil"
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
"github.com/ethereum/go-ethereum/ethdb"
|
|
"github.com/ethereum/go-ethereum/rlp"
|
|
"github.com/ethereum/go-ethereum/rpc"
|
|
ipfsethdb "github.com/vulcanize/pg-ipfs-ethdb"
|
|
|
|
"github.com/vulcanize/ipld-eth-indexer/pkg/ipfs"
|
|
"github.com/vulcanize/ipld-eth-indexer/pkg/postgres"
|
|
"github.com/vulcanize/ipld-eth-server/pkg/shared"
|
|
)
|
|
|
|
var (
|
|
errPendingBlockNumber = errors.New("pending block number not supported")
|
|
)
|
|
|
|
type Backend struct {
|
|
Retriever *CIDRetriever
|
|
Fetcher *IPLDFetcher
|
|
DB *postgres.DB
|
|
EthDB ethdb.Database
|
|
}
|
|
|
|
func NewEthBackend(db *postgres.DB) (*Backend, error) {
|
|
r := NewCIDRetriever(db)
|
|
return &Backend{
|
|
Retriever: r,
|
|
Fetcher: NewIPLDFetcher(db),
|
|
DB: db,
|
|
EthDB: ipfsethdb.NewDatabase(db.DB),
|
|
}, nil
|
|
}
|
|
|
|
func (b *Backend) HeaderByNumber(ctx context.Context, blockNumber rpc.BlockNumber) (*types.Header, error) {
|
|
var err error
|
|
number := blockNumber.Int64()
|
|
if blockNumber == rpc.LatestBlockNumber {
|
|
number, err = b.Retriever.RetrieveLastBlockNumber()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
if blockNumber == rpc.PendingBlockNumber {
|
|
return nil, errPendingBlockNumber
|
|
}
|
|
|
|
var headerBytes []byte
|
|
pgStr := `SELECT DISTINCT ON (block_number) data
|
|
FROM public.blocks INNER JOIN eth.header_cids ON (header_cids.mh_key = blocks.key)
|
|
WHERE block_number = $1
|
|
ORDER BY block_number, times_validated DESC`
|
|
if err := b.DB.Get(&headerBytes, pgStr, number); err != nil {
|
|
return nil, fmt.Errorf("header at block %d is not available; err %s", number, err.Error())
|
|
}
|
|
header := new(types.Header)
|
|
return header, rlp.DecodeBytes(headerBytes, header)
|
|
}
|
|
|
|
// GetTd retrieves and returns the total difficulty at the given block hash
|
|
func (b *Backend) GetTd(blockHash common.Hash) (*big.Int, error) {
|
|
pgStr := `SELECT td FROM eth.header_cids
|
|
WHERE header_cids.block_hash = $1`
|
|
var tdStr string
|
|
err := b.DB.Get(&tdStr, pgStr, blockHash.String())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
td, ok := new(big.Int).SetString(tdStr, 10)
|
|
if !ok {
|
|
return nil, errors.New("total difficulty retrieved from Postgres cannot be converted to an integer")
|
|
}
|
|
return td, nil
|
|
}
|
|
|
|
// GetLogs returns all the logs for the given block hash
|
|
func (b *Backend) GetLogs(ctx context.Context, hash common.Hash) ([][]*types.Log, error) {
|
|
pgStr := `SELECT blocks.data FROM public.blocks, eth.receipt_cids, eth.transaction_cids, eth.header_cids
|
|
WHERE header_cids.block_hash = $1
|
|
AND receipt_cids.tx_id = transaction_cids.id
|
|
AND transaction_cids.header_id = header_cids.id
|
|
ORDER BY transaction_cids.index ASC`
|
|
var receiptBytes [][]byte
|
|
if err := b.DB.Select(&receiptBytes, pgStr, hash.Hex()); err != nil {
|
|
return nil, err
|
|
}
|
|
logs := make([][]*types.Log, len(receiptBytes))
|
|
for i, rctBytes := range receiptBytes {
|
|
var rct types.Receipt
|
|
if err := rlp.DecodeBytes(rctBytes, &rct); err != nil {
|
|
return nil, err
|
|
}
|
|
logs[i] = rct.Logs
|
|
}
|
|
return logs, nil
|
|
}
|
|
|
|
// BlockByNumber returns the requested canonical block.
|
|
// Since the ipld-eth-server database can contain forked blocks, it is recommended to fetch BlockByHash as
|
|
// fetching by number can return non-deterministic results (returns the first block found at that height)
|
|
func (b *Backend) BlockByNumber(ctx context.Context, blockNumber rpc.BlockNumber) (*types.Block, error) {
|
|
var err error
|
|
number := blockNumber.Int64()
|
|
if blockNumber == rpc.LatestBlockNumber {
|
|
number, err = b.Retriever.RetrieveLastBlockNumber()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
if blockNumber == rpc.PendingBlockNumber {
|
|
return nil, errPendingBlockNumber
|
|
}
|
|
// Retrieve all the CIDs for the block
|
|
headerCID, uncleCIDs, txCIDs, rctCIDs, err := b.Retriever.RetrieveBlockByNumber(number)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Begin tx
|
|
tx, err := b.DB.Beginx()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer func() {
|
|
if p := recover(); p != nil {
|
|
shared.Rollback(tx)
|
|
panic(p)
|
|
} else if err != nil {
|
|
shared.Rollback(tx)
|
|
} else {
|
|
err = tx.Commit()
|
|
}
|
|
}()
|
|
|
|
// Fetch and decode the header IPLD
|
|
headerIPLD, err := b.Fetcher.FetchHeader(tx, headerCID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var header types.Header
|
|
if err := rlp.DecodeBytes(headerIPLD.Data, &header); err != nil {
|
|
return nil, err
|
|
}
|
|
// Fetch and decode the uncle IPLDs
|
|
uncleIPLDs, err := b.Fetcher.FetchUncles(tx, uncleCIDs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var uncles []*types.Header
|
|
for _, uncleIPLD := range uncleIPLDs {
|
|
var uncle types.Header
|
|
if err := rlp.DecodeBytes(uncleIPLD.Data, &uncle); err != nil {
|
|
return nil, err
|
|
}
|
|
uncles = append(uncles, &uncle)
|
|
}
|
|
// Fetch and decode the transaction IPLDs
|
|
txIPLDs, err := b.Fetcher.FetchTrxs(tx, txCIDs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var transactions []*types.Transaction
|
|
for _, txIPLD := range txIPLDs {
|
|
var transaction types.Transaction
|
|
if err := rlp.DecodeBytes(txIPLD.Data, &transaction); err != nil {
|
|
return nil, err
|
|
}
|
|
transactions = append(transactions, &transaction)
|
|
}
|
|
// Fetch and decode the receipt IPLDs
|
|
rctIPLDs, err := b.Fetcher.FetchRcts(tx, rctCIDs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var receipts []*types.Receipt
|
|
for _, rctIPLD := range rctIPLDs {
|
|
var receipt types.Receipt
|
|
if err := rlp.DecodeBytes(rctIPLD.Data, &receipt); err != nil {
|
|
return nil, err
|
|
}
|
|
receipts = append(receipts, &receipt)
|
|
}
|
|
// Compose everything together into a complete block
|
|
return types.NewBlock(&header, transactions, uncles, receipts), err
|
|
}
|
|
|
|
// BlockByHash returns the requested block. When fullTx is true all transactions in the block are returned in full
|
|
// detail, otherwise only the transaction hash is returned.
|
|
func (b *Backend) BlockByHash(ctx context.Context, hash common.Hash) (*types.Block, error) {
|
|
// Retrieve all the CIDs for the block
|
|
headerCID, uncleCIDs, txCIDs, rctCIDs, err := b.Retriever.RetrieveBlockByHash(hash)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Begin tx
|
|
tx, err := b.DB.Beginx()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer func() {
|
|
if p := recover(); p != nil {
|
|
shared.Rollback(tx)
|
|
panic(p)
|
|
} else if err != nil {
|
|
shared.Rollback(tx)
|
|
} else {
|
|
err = tx.Commit()
|
|
}
|
|
}()
|
|
|
|
// Fetch and decode the header IPLD
|
|
headerIPLD, err := b.Fetcher.FetchHeader(tx, headerCID)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var header types.Header
|
|
if err := rlp.DecodeBytes(headerIPLD.Data, &header); err != nil {
|
|
return nil, err
|
|
}
|
|
// Fetch and decode the uncle IPLDs
|
|
uncleIPLDs, err := b.Fetcher.FetchUncles(tx, uncleCIDs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var uncles []*types.Header
|
|
for _, uncleIPLD := range uncleIPLDs {
|
|
var uncle types.Header
|
|
if err := rlp.DecodeBytes(uncleIPLD.Data, &uncle); err != nil {
|
|
return nil, err
|
|
}
|
|
uncles = append(uncles, &uncle)
|
|
}
|
|
// Fetch and decode the transaction IPLDs
|
|
txIPLDs, err := b.Fetcher.FetchTrxs(tx, txCIDs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var transactions []*types.Transaction
|
|
for _, txIPLD := range txIPLDs {
|
|
var transaction types.Transaction
|
|
if err := rlp.DecodeBytes(txIPLD.Data, &transaction); err != nil {
|
|
return nil, err
|
|
}
|
|
transactions = append(transactions, &transaction)
|
|
}
|
|
// Fetch and decode the receipt IPLDs
|
|
rctIPLDs, err := b.Fetcher.FetchRcts(tx, rctCIDs)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var receipts []*types.Receipt
|
|
for _, rctIPLD := range rctIPLDs {
|
|
var receipt types.Receipt
|
|
if err := rlp.DecodeBytes(rctIPLD.Data, &receipt); err != nil {
|
|
return nil, err
|
|
}
|
|
receipts = append(receipts, &receipt)
|
|
}
|
|
// Compose everything together into a complete block
|
|
return types.NewBlock(&header, transactions, uncles, receipts), err
|
|
}
|
|
|
|
// GetTransaction retrieves a tx by hash
|
|
// It also returns the blockhash, blocknumber, and tx index associated with the transaction
|
|
func (b *Backend) GetTransaction(ctx context.Context, txHash common.Hash) (*types.Transaction, common.Hash, uint64, uint64, error) {
|
|
var tempTxStruct struct {
|
|
Data []byte `db:"data"`
|
|
BlockHash string `db:"block_hash"`
|
|
BlockNumber uint64 `db:"block_number"`
|
|
Index uint64 `db:"index"`
|
|
}
|
|
pgStr := `SELECT blocks.data, block_hash, block_number, index FROM public.blocks, eth.transaction_cids, eth.header_cids
|
|
WHERE blocks.key = transaction_cids.mh_key
|
|
AND transaction_cids.header_id = header_cids.id
|
|
AND transaction_cids.tx_hash = $1`
|
|
if err := b.DB.Get(&tempTxStruct, pgStr, txHash.String()); err != nil {
|
|
return nil, common.Hash{}, 0, 0, err
|
|
}
|
|
var transaction types.Transaction
|
|
if err := rlp.DecodeBytes(tempTxStruct.Data, &transaction); err != nil {
|
|
return nil, common.Hash{}, 0, 0, err
|
|
}
|
|
return &transaction, common.HexToHash(tempTxStruct.BlockHash), tempTxStruct.BlockNumber, tempTxStruct.Index, nil
|
|
}
|
|
|
|
// extractLogsOfInterest returns logs from the receipt IPLD
|
|
func extractLogsOfInterest(rctIPLDs []ipfs.BlockModel, wantedTopics [][]string) ([]*types.Log, error) {
|
|
var logs []*types.Log
|
|
for _, rctIPLD := range rctIPLDs {
|
|
rctRLP := rctIPLD
|
|
var rct types.Receipt
|
|
if err := rlp.DecodeBytes(rctRLP.Data, &rct); err != nil {
|
|
return nil, err
|
|
}
|
|
for _, log := range rct.Logs {
|
|
if wanted := wantedLog(wantedTopics, log.Topics); wanted == true {
|
|
logs = append(logs, log)
|
|
}
|
|
}
|
|
}
|
|
return logs, nil
|
|
}
|
|
|
|
// returns true if the log matches on the filter
|
|
func wantedLog(wantedTopics [][]string, actualTopics []common.Hash) bool {
|
|
// actualTopics will always have length <= 4
|
|
// wantedTopics will always have length 4
|
|
matches := 0
|
|
for i, actualTopic := range actualTopics {
|
|
// If we have topics in this filter slot, count as a match if the actualTopic matches one of the ones in this filter slot
|
|
if len(wantedTopics[i]) > 0 {
|
|
matches += sliceContainsHash(wantedTopics[i], actualTopic)
|
|
} else {
|
|
// Filter slot is empty, not matching any topics at this slot => counts as a match
|
|
matches++
|
|
}
|
|
}
|
|
if matches == len(actualTopics) {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
// returns 1 if the slice contains the hash, 0 if it does not
|
|
func sliceContainsHash(slice []string, hash common.Hash) int {
|
|
for _, str := range slice {
|
|
if str == hash.String() {
|
|
return 1
|
|
}
|
|
}
|
|
return 0
|
|
}
|
|
|
|
// rpcMarshalHeader uses the generalized output filler, then adds the total difficulty field, which requires
|
|
// a `PublicEthAPI`.
|
|
func (pea *PublicEthAPI) rpcMarshalHeader(header *types.Header) (map[string]interface{}, error) {
|
|
fields := RPCMarshalHeader(header)
|
|
td, err := pea.B.GetTd(header.Hash())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
fields["totalDifficulty"] = (*hexutil.Big)(td)
|
|
return fields, nil
|
|
}
|
|
|
|
// RPCMarshalHeader converts the given header to the RPC output.
|
|
// This function is eth/internal so we have to make our own version here...
|
|
func RPCMarshalHeader(head *types.Header) map[string]interface{} {
|
|
return map[string]interface{}{
|
|
"number": (*hexutil.Big)(head.Number),
|
|
"hash": head.Hash(),
|
|
"parentHash": head.ParentHash,
|
|
"nonce": head.Nonce,
|
|
"mixHash": head.MixDigest,
|
|
"sha3Uncles": head.UncleHash,
|
|
"logsBloom": head.Bloom,
|
|
"stateRoot": head.Root,
|
|
"miner": head.Coinbase,
|
|
"difficulty": (*hexutil.Big)(head.Difficulty),
|
|
"extraData": hexutil.Bytes(head.Extra),
|
|
"size": hexutil.Uint64(head.Size()),
|
|
"gasLimit": hexutil.Uint64(head.GasLimit),
|
|
"gasUsed": hexutil.Uint64(head.GasUsed),
|
|
"timestamp": hexutil.Uint64(head.Time),
|
|
"transactionsRoot": head.TxHash,
|
|
"receiptsRoot": head.ReceiptHash,
|
|
}
|
|
}
|
|
|
|
// rpcMarshalBlock uses the generalized output filler, then adds the total difficulty field, which requires
|
|
// a `PublicBlockchainAPI`.
|
|
func (pea *PublicEthAPI) rpcMarshalBlock(b *types.Block, inclTx bool, fullTx bool) (map[string]interface{}, error) {
|
|
fields, err := RPCMarshalBlock(b, inclTx, fullTx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
td, err := pea.B.GetTd(b.Hash())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
fields["totalDifficulty"] = (*hexutil.Big)(td)
|
|
return fields, err
|
|
}
|
|
|
|
// RPCMarshalBlock converts the given block to the RPC output which depends on fullTx. If inclTx is true transactions are
|
|
// returned. When fullTx is true the returned block contains full transaction details, otherwise it will only contain
|
|
// transaction hashes.
|
|
func RPCMarshalBlock(block *types.Block, inclTx bool, fullTx bool) (map[string]interface{}, error) {
|
|
fields := RPCMarshalHeader(block.Header())
|
|
fields["size"] = hexutil.Uint64(block.Size())
|
|
|
|
if inclTx {
|
|
formatTx := func(tx *types.Transaction) (interface{}, error) {
|
|
return tx.Hash(), nil
|
|
}
|
|
if fullTx {
|
|
formatTx = func(tx *types.Transaction) (interface{}, error) {
|
|
return NewRPCTransactionFromBlockHash(block, tx.Hash()), nil
|
|
}
|
|
}
|
|
txs := block.Transactions()
|
|
transactions := make([]interface{}, len(txs))
|
|
var err error
|
|
for i, tx := range txs {
|
|
if transactions[i], err = formatTx(tx); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
fields["transactions"] = transactions
|
|
}
|
|
uncles := block.Uncles()
|
|
uncleHashes := make([]common.Hash, len(uncles))
|
|
for i, uncle := range uncles {
|
|
uncleHashes[i] = uncle.Hash()
|
|
}
|
|
fields["uncles"] = uncleHashes
|
|
|
|
return fields, nil
|
|
}
|
|
|
|
// NewRPCTransactionFromBlockHash returns a transaction that will serialize to the RPC representation.
|
|
func NewRPCTransactionFromBlockHash(b *types.Block, hash common.Hash) *RPCTransaction {
|
|
for idx, tx := range b.Transactions() {
|
|
if tx.Hash() == hash {
|
|
return newRPCTransactionFromBlockIndex(b, uint64(idx))
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// newRPCTransactionFromBlockIndex returns a transaction that will serialize to the RPC representation.
|
|
func newRPCTransactionFromBlockIndex(b *types.Block, index uint64) *RPCTransaction {
|
|
txs := b.Transactions()
|
|
if index >= uint64(len(txs)) {
|
|
return nil
|
|
}
|
|
return NewRPCTransaction(txs[index], b.Hash(), b.NumberU64(), index)
|
|
}
|
|
|
|
// RPCTransaction represents a transaction that will serialize to the RPC representation of a transaction
|
|
type RPCTransaction struct {
|
|
BlockHash *common.Hash `json:"blockHash"`
|
|
BlockNumber *hexutil.Big `json:"blockNumber"`
|
|
From common.Address `json:"from"`
|
|
Gas hexutil.Uint64 `json:"gas"`
|
|
GasPrice *hexutil.Big `json:"gasPrice"`
|
|
Hash common.Hash `json:"hash"`
|
|
Input hexutil.Bytes `json:"input"`
|
|
Nonce hexutil.Uint64 `json:"nonce"`
|
|
To *common.Address `json:"to"`
|
|
TransactionIndex *hexutil.Uint64 `json:"transactionIndex"`
|
|
Value *hexutil.Big `json:"value"`
|
|
V *hexutil.Big `json:"v"`
|
|
R *hexutil.Big `json:"r"`
|
|
S *hexutil.Big `json:"s"`
|
|
}
|
|
|
|
// NewRPCTransaction returns a transaction that will serialize to the RPC
|
|
// representation, with the given location metadata set (if available).
|
|
func NewRPCTransaction(tx *types.Transaction, blockHash common.Hash, blockNumber uint64, index uint64) *RPCTransaction {
|
|
var signer types.Signer = types.FrontierSigner{}
|
|
if tx.Protected() {
|
|
signer = types.NewEIP155Signer(tx.ChainId())
|
|
}
|
|
from, _ := types.Sender(signer, tx)
|
|
v, r, s := tx.RawSignatureValues()
|
|
|
|
result := &RPCTransaction{
|
|
From: from,
|
|
Gas: hexutil.Uint64(tx.Gas()),
|
|
GasPrice: (*hexutil.Big)(tx.GasPrice()),
|
|
Hash: tx.Hash(),
|
|
Input: hexutil.Bytes(tx.Data()), // somehow this is ending up `nil`
|
|
Nonce: hexutil.Uint64(tx.Nonce()),
|
|
To: tx.To(),
|
|
Value: (*hexutil.Big)(tx.Value()),
|
|
V: (*hexutil.Big)(v),
|
|
R: (*hexutil.Big)(r),
|
|
S: (*hexutil.Big)(s),
|
|
}
|
|
if blockHash != (common.Hash{}) {
|
|
result.BlockHash = &blockHash
|
|
result.BlockNumber = (*hexutil.Big)(new(big.Int).SetUint64(blockNumber))
|
|
result.TransactionIndex = (*hexutil.Uint64)(&index)
|
|
}
|
|
return result
|
|
}
|
|
|
|
// StateAndHeaderByNumberOrHash returns the statedb and header for the provided block number or hash
|
|
func (b *Backend) StateAndHeaderByNumberOrHash(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (*state.StateDB, *types.Header, error) {
|
|
if blockNr, ok := blockNrOrHash.Number(); ok {
|
|
return b.StateAndHeaderByNumber(ctx, blockNr)
|
|
}
|
|
if hash, ok := blockNrOrHash.Hash(); ok {
|
|
header, err := b.HeaderByHash(ctx, hash)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
if header == nil {
|
|
return nil, nil, errors.New("header for hash not found")
|
|
}
|
|
if blockNrOrHash.RequireCanonical && b.eth.blockchain.GetCanonicalHash(header.Number.Uint64()) != hash {
|
|
return nil, nil, errors.New("hash is not currently canonical")
|
|
}
|
|
stateDb, err := b.eth.BlockChain().StateAt(header.Root)
|
|
return stateDb, header, err
|
|
}
|
|
return nil, nil, errors.New("invalid arguments; neither block nor hash specified")
|
|
}
|
|
|
|
func (b *Backend) HeaderByHash(ctx context.Context, hash common.Hash) (*types.Header, error) {
|
|
return b.GetHeaderByHash(hash), nil
|
|
}
|
|
|
|
func (b *Backend) GetHeaderByHash(hash common.Hash) *types.Header {
|
|
|
|
}
|
|
|
|
// StateAndHeaderByNumber returns the statedb and header for a provided block number
|
|
func (b *Backend) StateAndHeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*state.StateDB, *types.Header, error) {
|
|
// Pending state is only known by the miner
|
|
if number == rpc.PendingBlockNumber {
|
|
block, state := b.eth.miner.Pending()
|
|
return state, block.Header(), nil
|
|
}
|
|
// Otherwise resolve the block number and return its state
|
|
header, err := b.HeaderByNumber(ctx, number)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
if header == nil {
|
|
return nil, nil, errors.New("header not found")
|
|
}
|
|
stateDb, err := b.eth.BlockChain().StateAt(header.Root)
|
|
return stateDb, header, err
|
|
}
|