Merge pull request #23178 from karalabe/feeapi-fixes
eth/gasprice, internal/ethapi, miner: minor feehistory fixes
This commit is contained in:
commit
9624f92ede
@ -19,6 +19,7 @@ package gasprice
|
|||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
"math/big"
|
"math/big"
|
||||||
"sort"
|
"sort"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
@ -30,11 +31,19 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
errInvalidPercentiles = errors.New("Invalid reward percentiles")
|
errInvalidPercentile = errors.New("invalid reward percentile")
|
||||||
errRequestBeyondHead = errors.New("Request beyond head block")
|
errRequestBeyondHead = errors.New("request beyond head block")
|
||||||
)
|
)
|
||||||
|
|
||||||
const maxBlockCount = 1024 // number of blocks retrievable with a single query
|
const (
|
||||||
|
// maxFeeHistory is the maximum number of blocks that can be retrieved for a
|
||||||
|
// fee history request.
|
||||||
|
maxFeeHistory = 1024
|
||||||
|
|
||||||
|
// maxBlockFetchers is the max number of goroutines to spin up to pull blocks
|
||||||
|
// for the fee history calculation (mostly relevant for LES).
|
||||||
|
maxBlockFetchers = 4
|
||||||
|
)
|
||||||
|
|
||||||
// blockFees represents a single block for processing
|
// blockFees represents a single block for processing
|
||||||
type blockFees struct {
|
type blockFees struct {
|
||||||
@ -124,23 +133,22 @@ func (oracle *Oracle) processBlock(bf *blockFees, percentiles []float64) {
|
|||||||
// also returned if requested and available.
|
// also returned if requested and available.
|
||||||
// Note: an error is only returned if retrieving the head header has failed. If there are no
|
// Note: an error is only returned if retrieving the head header has failed. If there are no
|
||||||
// retrievable blocks in the specified range then zero block count is returned with no error.
|
// retrievable blocks in the specified range then zero block count is returned with no error.
|
||||||
func (oracle *Oracle) resolveBlockRange(ctx context.Context, lastBlockNumber rpc.BlockNumber, blockCount, maxHistory int) (*types.Block, types.Receipts, rpc.BlockNumber, int, error) {
|
func (oracle *Oracle) resolveBlockRange(ctx context.Context, lastBlock rpc.BlockNumber, blocks, maxHistory int) (*types.Block, []*types.Receipt, rpc.BlockNumber, int, error) {
|
||||||
var (
|
var (
|
||||||
headBlockNumber rpc.BlockNumber
|
headBlock rpc.BlockNumber
|
||||||
pendingBlock *types.Block
|
pendingBlock *types.Block
|
||||||
pendingReceipts types.Receipts
|
pendingReceipts types.Receipts
|
||||||
)
|
)
|
||||||
|
// query either pending block or head header and set headBlock
|
||||||
// query either pending block or head header and set headBlockNumber
|
if lastBlock == rpc.PendingBlockNumber {
|
||||||
if lastBlockNumber == rpc.PendingBlockNumber {
|
|
||||||
if pendingBlock, pendingReceipts = oracle.backend.PendingBlockAndReceipts(); pendingBlock != nil {
|
if pendingBlock, pendingReceipts = oracle.backend.PendingBlockAndReceipts(); pendingBlock != nil {
|
||||||
lastBlockNumber = rpc.BlockNumber(pendingBlock.NumberU64())
|
lastBlock = rpc.BlockNumber(pendingBlock.NumberU64())
|
||||||
headBlockNumber = lastBlockNumber - 1
|
headBlock = lastBlock - 1
|
||||||
} else {
|
} else {
|
||||||
// pending block not supported by backend, process until latest block
|
// pending block not supported by backend, process until latest block
|
||||||
lastBlockNumber = rpc.LatestBlockNumber
|
lastBlock = rpc.LatestBlockNumber
|
||||||
blockCount--
|
blocks--
|
||||||
if blockCount == 0 {
|
if blocks == 0 {
|
||||||
return nil, nil, 0, 0, nil
|
return nil, nil, 0, 0, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -148,32 +156,32 @@ func (oracle *Oracle) resolveBlockRange(ctx context.Context, lastBlockNumber rpc
|
|||||||
if pendingBlock == nil {
|
if pendingBlock == nil {
|
||||||
// if pending block is not fetched then we retrieve the head header to get the head block number
|
// if pending block is not fetched then we retrieve the head header to get the head block number
|
||||||
if latestHeader, err := oracle.backend.HeaderByNumber(ctx, rpc.LatestBlockNumber); err == nil {
|
if latestHeader, err := oracle.backend.HeaderByNumber(ctx, rpc.LatestBlockNumber); err == nil {
|
||||||
headBlockNumber = rpc.BlockNumber(latestHeader.Number.Uint64())
|
headBlock = rpc.BlockNumber(latestHeader.Number.Uint64())
|
||||||
} else {
|
} else {
|
||||||
return nil, nil, 0, 0, err
|
return nil, nil, 0, 0, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if lastBlockNumber == rpc.LatestBlockNumber {
|
if lastBlock == rpc.LatestBlockNumber {
|
||||||
lastBlockNumber = headBlockNumber
|
lastBlock = headBlock
|
||||||
} else if pendingBlock == nil && lastBlockNumber > headBlockNumber {
|
} else if pendingBlock == nil && lastBlock > headBlock {
|
||||||
return nil, nil, 0, 0, errRequestBeyondHead
|
return nil, nil, 0, 0, fmt.Errorf("%w: requested %d, head %d", errRequestBeyondHead, lastBlock, headBlock)
|
||||||
}
|
}
|
||||||
if maxHistory != 0 {
|
if maxHistory != 0 {
|
||||||
// limit retrieval to the given number of latest blocks
|
// limit retrieval to the given number of latest blocks
|
||||||
if tooOldCount := int64(headBlockNumber) - int64(maxHistory) - int64(lastBlockNumber) + int64(blockCount); tooOldCount > 0 {
|
if tooOldCount := int64(headBlock) - int64(maxHistory) - int64(lastBlock) + int64(blocks); tooOldCount > 0 {
|
||||||
// tooOldCount is the number of requested blocks that are too old to be served
|
// tooOldCount is the number of requested blocks that are too old to be served
|
||||||
if int64(blockCount) > tooOldCount {
|
if int64(blocks) > tooOldCount {
|
||||||
blockCount -= int(tooOldCount)
|
blocks -= int(tooOldCount)
|
||||||
} else {
|
} else {
|
||||||
return nil, nil, 0, 0, nil
|
return nil, nil, 0, 0, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// ensure not trying to retrieve before genesis
|
// ensure not trying to retrieve before genesis
|
||||||
if rpc.BlockNumber(blockCount) > lastBlockNumber+1 {
|
if rpc.BlockNumber(blocks) > lastBlock+1 {
|
||||||
blockCount = int(lastBlockNumber + 1)
|
blocks = int(lastBlock + 1)
|
||||||
}
|
}
|
||||||
return pendingBlock, pendingReceipts, lastBlockNumber, blockCount, nil
|
return pendingBlock, pendingReceipts, lastBlock, blocks, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// FeeHistory returns data relevant for fee estimation based on the specified range of blocks.
|
// FeeHistory returns data relevant for fee estimation based on the specified range of blocks.
|
||||||
@ -189,90 +197,89 @@ func (oracle *Oracle) resolveBlockRange(ctx context.Context, lastBlockNumber rpc
|
|||||||
// - gasUsedRatio: gasUsed/gasLimit in the given block
|
// - gasUsedRatio: gasUsed/gasLimit in the given block
|
||||||
// Note: baseFee includes the next block after the newest of the returned range, because this
|
// Note: baseFee includes the next block after the newest of the returned range, because this
|
||||||
// value can be derived from the newest block.
|
// value can be derived from the newest block.
|
||||||
func (oracle *Oracle) FeeHistory(ctx context.Context, blockCount int, lastBlockNumber rpc.BlockNumber, rewardPercentiles []float64) (firstBlockNumber rpc.BlockNumber, reward [][]*big.Int, baseFee []*big.Int, gasUsedRatio []float64, err error) {
|
func (oracle *Oracle) FeeHistory(ctx context.Context, blocks int, lastBlock rpc.BlockNumber, rewardPercentiles []float64) (rpc.BlockNumber, [][]*big.Int, []*big.Int, []float64, error) {
|
||||||
if blockCount < 1 {
|
if blocks < 1 {
|
||||||
// returning with no data and no error means there are no retrievable blocks
|
return 0, nil, nil, nil, nil // returning with no data and no error means there are no retrievable blocks
|
||||||
return
|
|
||||||
}
|
}
|
||||||
if blockCount > maxBlockCount {
|
if blocks > maxFeeHistory {
|
||||||
blockCount = maxBlockCount
|
log.Warn("Sanitizing fee history length", "requested", blocks, "truncated", maxFeeHistory)
|
||||||
|
blocks = maxFeeHistory
|
||||||
}
|
}
|
||||||
for i, p := range rewardPercentiles {
|
for i, p := range rewardPercentiles {
|
||||||
if p < 0 || p > 100 || (i > 0 && p < rewardPercentiles[i-1]) {
|
if p < 0 || p > 100 {
|
||||||
return 0, nil, nil, nil, errInvalidPercentiles
|
return 0, nil, nil, nil, fmt.Errorf("%w: %f", errInvalidPercentile, p)
|
||||||
|
}
|
||||||
|
if i > 0 && p < rewardPercentiles[i-1] {
|
||||||
|
return 0, nil, nil, nil, fmt.Errorf("%w: #%d:%f > #%d:%f", errInvalidPercentile, i-1, rewardPercentiles[i-1], i, p)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Only process blocks if reward percentiles were requested
|
||||||
processBlocks := len(rewardPercentiles) != 0
|
maxHistory := oracle.maxHeaderHistory
|
||||||
// limit retrieval to maxHistory if set
|
if len(rewardPercentiles) != 0 {
|
||||||
var maxHistory int
|
|
||||||
if processBlocks {
|
|
||||||
maxHistory = oracle.maxBlockHistory
|
maxHistory = oracle.maxBlockHistory
|
||||||
} else {
|
|
||||||
maxHistory = oracle.maxHeaderHistory
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
pendingBlock *types.Block
|
pendingBlock *types.Block
|
||||||
pendingReceipts types.Receipts
|
pendingReceipts []*types.Receipt
|
||||||
|
err error
|
||||||
)
|
)
|
||||||
if pendingBlock, pendingReceipts, lastBlockNumber, blockCount, err = oracle.resolveBlockRange(ctx, lastBlockNumber, blockCount, maxHistory); err != nil || blockCount == 0 {
|
pendingBlock, pendingReceipts, lastBlock, blocks, err = oracle.resolveBlockRange(ctx, lastBlock, blocks, maxHistory)
|
||||||
return
|
if err != nil || blocks == 0 {
|
||||||
|
return 0, nil, nil, nil, err
|
||||||
}
|
}
|
||||||
firstBlockNumber = lastBlockNumber + 1 - rpc.BlockNumber(blockCount)
|
oldestBlock := lastBlock + 1 - rpc.BlockNumber(blocks)
|
||||||
|
|
||||||
processNext := int64(firstBlockNumber)
|
var (
|
||||||
resultCh := make(chan *blockFees, blockCount)
|
next = int64(oldestBlock)
|
||||||
threadCount := 4
|
results = make(chan *blockFees, blocks)
|
||||||
if blockCount < threadCount {
|
)
|
||||||
threadCount = blockCount
|
for i := 0; i < maxBlockFetchers && i < blocks; i++ {
|
||||||
}
|
|
||||||
for i := 0; i < threadCount; i++ {
|
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
blockNumber := rpc.BlockNumber(atomic.AddInt64(&processNext, 1) - 1)
|
// Retrieve the next block number to fetch with this goroutine
|
||||||
if blockNumber > lastBlockNumber {
|
blockNumber := rpc.BlockNumber(atomic.AddInt64(&next, 1) - 1)
|
||||||
|
if blockNumber > lastBlock {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
bf := &blockFees{blockNumber: blockNumber}
|
fees := &blockFees{blockNumber: blockNumber}
|
||||||
if pendingBlock != nil && blockNumber >= rpc.BlockNumber(pendingBlock.NumberU64()) {
|
if pendingBlock != nil && blockNumber >= rpc.BlockNumber(pendingBlock.NumberU64()) {
|
||||||
bf.block, bf.receipts = pendingBlock, pendingReceipts
|
fees.block, fees.receipts = pendingBlock, pendingReceipts
|
||||||
} else {
|
} else {
|
||||||
if processBlocks {
|
if len(rewardPercentiles) != 0 {
|
||||||
bf.block, bf.err = oracle.backend.BlockByNumber(ctx, blockNumber)
|
fees.block, fees.err = oracle.backend.BlockByNumber(ctx, blockNumber)
|
||||||
if bf.block != nil {
|
if fees.block != nil && fees.err == nil {
|
||||||
bf.receipts, bf.err = oracle.backend.GetReceipts(ctx, bf.block.Hash())
|
fees.receipts, fees.err = oracle.backend.GetReceipts(ctx, fees.block.Hash())
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
bf.header, bf.err = oracle.backend.HeaderByNumber(ctx, blockNumber)
|
fees.header, fees.err = oracle.backend.HeaderByNumber(ctx, blockNumber)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if bf.block != nil {
|
if fees.block != nil {
|
||||||
bf.header = bf.block.Header()
|
fees.header = fees.block.Header()
|
||||||
}
|
}
|
||||||
if bf.header != nil {
|
if fees.header != nil {
|
||||||
oracle.processBlock(bf, rewardPercentiles)
|
oracle.processBlock(fees, rewardPercentiles)
|
||||||
}
|
}
|
||||||
// send to resultCh even if empty to guarantee that blockCount items are sent in total
|
// send to results even if empty to guarantee that blocks items are sent in total
|
||||||
resultCh <- bf
|
results <- fees
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
var (
|
||||||
reward = make([][]*big.Int, blockCount)
|
reward = make([][]*big.Int, blocks)
|
||||||
baseFee = make([]*big.Int, blockCount+1)
|
baseFee = make([]*big.Int, blocks+1)
|
||||||
gasUsedRatio = make([]float64, blockCount)
|
gasUsedRatio = make([]float64, blocks)
|
||||||
firstMissing := blockCount
|
firstMissing = blocks
|
||||||
|
)
|
||||||
for ; blockCount > 0; blockCount-- {
|
for ; blocks > 0; blocks-- {
|
||||||
bf := <-resultCh
|
fees := <-results
|
||||||
if bf.err != nil {
|
if fees.err != nil {
|
||||||
return 0, nil, nil, nil, bf.err
|
return 0, nil, nil, nil, fees.err
|
||||||
}
|
}
|
||||||
i := int(bf.blockNumber - firstBlockNumber)
|
i := int(fees.blockNumber - oldestBlock)
|
||||||
if bf.header != nil {
|
if fees.header != nil {
|
||||||
reward[i], baseFee[i], baseFee[i+1], gasUsedRatio[i] = bf.reward, bf.baseFee, bf.nextBaseFee, bf.gasUsedRatio
|
reward[i], baseFee[i], baseFee[i+1], gasUsedRatio[i] = fees.reward, fees.baseFee, fees.nextBaseFee, fees.gasUsedRatio
|
||||||
} else {
|
} else {
|
||||||
// getting no block and no error means we are requesting into the future (might happen because of a reorg)
|
// getting no block and no error means we are requesting into the future (might happen because of a reorg)
|
||||||
if i < firstMissing {
|
if i < firstMissing {
|
||||||
@ -283,11 +290,11 @@ func (oracle *Oracle) FeeHistory(ctx context.Context, blockCount int, lastBlockN
|
|||||||
if firstMissing == 0 {
|
if firstMissing == 0 {
|
||||||
return 0, nil, nil, nil, nil
|
return 0, nil, nil, nil, nil
|
||||||
}
|
}
|
||||||
if processBlocks {
|
if len(rewardPercentiles) != 0 {
|
||||||
reward = reward[:firstMissing]
|
reward = reward[:firstMissing]
|
||||||
} else {
|
} else {
|
||||||
reward = nil
|
reward = nil
|
||||||
}
|
}
|
||||||
baseFee, gasUsedRatio = baseFee[:firstMissing+1], gasUsedRatio[:firstMissing]
|
baseFee, gasUsedRatio = baseFee[:firstMissing+1], gasUsedRatio[:firstMissing]
|
||||||
return
|
return oldestBlock, reward, baseFee, gasUsedRatio, nil
|
||||||
}
|
}
|
||||||
|
@ -18,6 +18,7 @@ package gasprice
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"math/big"
|
"math/big"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
@ -37,7 +38,7 @@ func TestFeeHistory(t *testing.T) {
|
|||||||
}{
|
}{
|
||||||
{false, 0, 0, 10, 30, nil, 21, 10, nil},
|
{false, 0, 0, 10, 30, nil, 21, 10, nil},
|
||||||
{false, 0, 0, 10, 30, []float64{0, 10}, 21, 10, nil},
|
{false, 0, 0, 10, 30, []float64{0, 10}, 21, 10, nil},
|
||||||
{false, 0, 0, 10, 30, []float64{20, 10}, 0, 0, errInvalidPercentiles},
|
{false, 0, 0, 10, 30, []float64{20, 10}, 0, 0, errInvalidPercentile},
|
||||||
{false, 0, 0, 1000000000, 30, nil, 0, 31, nil},
|
{false, 0, 0, 1000000000, 30, nil, 0, 31, nil},
|
||||||
{false, 0, 0, 1000000000, rpc.LatestBlockNumber, nil, 0, 33, nil},
|
{false, 0, 0, 1000000000, rpc.LatestBlockNumber, nil, 0, 33, nil},
|
||||||
{false, 0, 0, 10, 40, nil, 0, 0, errRequestBeyondHead},
|
{false, 0, 0, 10, 40, nil, 0, 0, errRequestBeyondHead},
|
||||||
@ -81,7 +82,7 @@ func TestFeeHistory(t *testing.T) {
|
|||||||
if len(ratio) != c.expCount {
|
if len(ratio) != c.expCount {
|
||||||
t.Fatalf("Test case %d: gasUsedRatio array length mismatch, want %d, got %d", i, c.expCount, len(ratio))
|
t.Fatalf("Test case %d: gasUsedRatio array length mismatch, want %d, got %d", i, c.expCount, len(ratio))
|
||||||
}
|
}
|
||||||
if err != c.expErr {
|
if err != c.expErr && !errors.Is(err, c.expErr) {
|
||||||
t.Fatalf("Test case %d: error mismatch, want %v, got %v", i, c.expErr, err)
|
t.Fatalf("Test case %d: error mismatch, want %v, got %v", i, c.expErr, err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -80,28 +80,28 @@ func (s *PublicEthereumAPI) MaxPriorityFeePerGas(ctx context.Context) (*hexutil.
|
|||||||
return (*hexutil.Big)(tipcap), err
|
return (*hexutil.Big)(tipcap), err
|
||||||
}
|
}
|
||||||
|
|
||||||
type feeHistoryResults struct {
|
type feeHistoryResult struct {
|
||||||
FirstBlock rpc.BlockNumber
|
OldestBlock rpc.BlockNumber `json:"oldestBlock"`
|
||||||
Reward [][]*hexutil.Big
|
Reward [][]*hexutil.Big `json:"reward,omitempty"`
|
||||||
BaseFee []*hexutil.Big
|
BaseFee []*hexutil.Big `json:"baseFeePerGas,omitempty"`
|
||||||
GasUsedRatio []float64
|
GasUsedRatio []float64 `json:"gasUsedRatio"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *PublicEthereumAPI) FeeHistory(ctx context.Context, blockCount int, lastBlock rpc.BlockNumber, rewardPercentiles []float64) (feeHistoryResults, error) {
|
func (s *PublicEthereumAPI) FeeHistory(ctx context.Context, blockCount int, lastBlock rpc.BlockNumber, rewardPercentiles []float64) (*feeHistoryResult, error) {
|
||||||
firstBlock, reward, baseFee, gasUsedRatio, err := s.b.FeeHistory(ctx, blockCount, lastBlock, rewardPercentiles)
|
oldest, reward, baseFee, gasUsed, err := s.b.FeeHistory(ctx, blockCount, lastBlock, rewardPercentiles)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return feeHistoryResults{}, err
|
return nil, err
|
||||||
}
|
}
|
||||||
results := feeHistoryResults{
|
results := &feeHistoryResult{
|
||||||
FirstBlock: firstBlock,
|
OldestBlock: oldest,
|
||||||
GasUsedRatio: gasUsedRatio,
|
GasUsedRatio: gasUsed,
|
||||||
}
|
}
|
||||||
if reward != nil {
|
if reward != nil {
|
||||||
results.Reward = make([][]*hexutil.Big, len(reward))
|
results.Reward = make([][]*hexutil.Big, len(reward))
|
||||||
for j, w := range reward {
|
for i, w := range reward {
|
||||||
results.Reward[j] = make([]*hexutil.Big, len(w))
|
results.Reward[i] = make([]*hexutil.Big, len(w))
|
||||||
for i, v := range w {
|
for j, v := range w {
|
||||||
results.Reward[j][i] = (*hexutil.Big)(v)
|
results.Reward[i][j] = (*hexutil.Big)(v)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -162,7 +162,7 @@ type worker struct {
|
|||||||
pendingMu sync.RWMutex
|
pendingMu sync.RWMutex
|
||||||
pendingTasks map[common.Hash]*task
|
pendingTasks map[common.Hash]*task
|
||||||
|
|
||||||
snapshotMu sync.RWMutex // The lock used to protect the block snapshot and state snapshot
|
snapshotMu sync.RWMutex // The lock used to protect the snapshots below
|
||||||
snapshotBlock *types.Block
|
snapshotBlock *types.Block
|
||||||
snapshotReceipts types.Receipts
|
snapshotReceipts types.Receipts
|
||||||
snapshotState *state.StateDB
|
snapshotState *state.StateDB
|
||||||
@ -745,7 +745,7 @@ func (w *worker) updateSnapshot() {
|
|||||||
w.current.receipts,
|
w.current.receipts,
|
||||||
trie.NewStackTrie(nil),
|
trie.NewStackTrie(nil),
|
||||||
)
|
)
|
||||||
w.snapshotReceipts = w.current.receipts
|
w.snapshotReceipts = copyReceipts(w.current.receipts)
|
||||||
w.snapshotState = w.current.state.Copy()
|
w.snapshotState = w.current.state.Copy()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user