2020-08-19 11:46:20 +00:00
|
|
|
package rpchelper
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2020-08-29 21:28:09 +00:00
|
|
|
|
2020-08-19 11:46:20 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core/rawdb"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core/types/accounts"
|
2020-08-29 21:28:09 +00:00
|
|
|
"github.com/ledgerwatch/turbo-geth/eth/stagedsync/stages"
|
2020-08-19 11:46:20 +00:00
|
|
|
"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 {
|
2020-08-29 21:28:09 +00:00
|
|
|
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)
|
|
|
|
}
|
2020-09-12 05:33:07 +00:00
|
|
|
|
|
|
|
} else if number == rpc.EarliestBlockNumber {
|
|
|
|
blockNumber = 0
|
|
|
|
|
|
|
|
} else if number == rpc.PendingBlockNumber {
|
|
|
|
return 0, common.Hash{}, fmt.Errorf("pending blocks are not supported")
|
|
|
|
|
2020-08-29 21:28:09 +00:00
|
|
|
} else {
|
|
|
|
blockNumber = uint64(number.Int64())
|
|
|
|
}
|
2020-08-19 11:46:20 +00:00
|
|
|
hash, err = GetHashByNumber(blockNumber, blockNrOrHash.RequireCanonical, dbReader)
|
|
|
|
if err != nil {
|
|
|
|
return 0, common.Hash{}, err
|
|
|
|
}
|
|
|
|
} else {
|
2020-10-24 06:57:09 +00:00
|
|
|
block, err := rawdb.ReadBlockByHash(dbReader, hash)
|
|
|
|
if err != nil {
|
|
|
|
return 0, common.Hash{}, err
|
|
|
|
}
|
2020-08-19 11:46:20 +00:00
|
|
|
if block == nil {
|
|
|
|
return 0, common.Hash{}, fmt.Errorf("block %x not found", hash)
|
|
|
|
}
|
|
|
|
blockNumber = block.NumberU64()
|
|
|
|
|
2020-10-10 06:05:56 +00:00
|
|
|
ch, err := rawdb.ReadCanonicalHash(dbReader, blockNumber)
|
|
|
|
if err != nil {
|
|
|
|
return 0, common.Hash{}, err
|
|
|
|
}
|
|
|
|
if blockNrOrHash.RequireCanonical && ch != hash {
|
2020-08-19 11:46:20 +00:00
|
|
|
return 0, common.Hash{}, fmt.Errorf("hash %q is not currently canonical", hash.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return blockNumber, hash, nil
|
|
|
|
}
|
|
|
|
|
2020-10-12 08:39:04 +00:00
|
|
|
func GetAccount(tx ethdb.Tx, blockNumber uint64, address common.Address) (*accounts.Account, error) {
|
|
|
|
reader := adapter.NewStateReader(tx, blockNumber)
|
2020-08-19 11:46:20 +00:00
|
|
|
return reader.ReadAccountData(address)
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetHashByNumber(blockNumber uint64, requireCanonical bool, dbReader rawdb.DatabaseReader) (common.Hash, error) {
|
|
|
|
if requireCanonical {
|
2020-10-10 06:05:56 +00:00
|
|
|
return rawdb.ReadCanonicalHash(dbReader, blockNumber)
|
2020-08-19 11:46:20 +00:00
|
|
|
}
|
|
|
|
|
2020-10-24 06:57:09 +00:00
|
|
|
block, err := rawdb.ReadBlockByNumber(dbReader, blockNumber)
|
|
|
|
if err != nil {
|
|
|
|
return common.Hash{}, fmt.Errorf("block read fail: %w", err)
|
|
|
|
}
|
2020-08-19 11:46:20 +00:00
|
|
|
if block == nil {
|
|
|
|
return common.Hash{}, fmt.Errorf("block %d not found", blockNumber)
|
|
|
|
}
|
|
|
|
return block.Hash(), nil
|
|
|
|
}
|