github.1485827954.workers.dev/ethereum/go-ethereum@v1.14.3/eth/gasprice/feehistory.go (about) 1 // Copyright 2021 The go-ethereum Authors 2 // This file is part of the go-ethereum library. 3 // 4 // The go-ethereum library is free software: you can redistribute it and/or modify 5 // it under the terms of the GNU Lesser General Public License as published by 6 // the Free Software Foundation, either version 3 of the License, or 7 // (at your option) any later version. 8 // 9 // The go-ethereum library is distributed in the hope that it will be useful, 10 // but WITHOUT ANY WARRANTY; without even the implied warranty of 11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 // GNU Lesser General Public License for more details. 13 // 14 // You should have received a copy of the GNU Lesser General Public License 15 // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. 16 17 package gasprice 18 19 import ( 20 "context" 21 "encoding/binary" 22 "errors" 23 "fmt" 24 "math" 25 "math/big" 26 "slices" 27 "sync/atomic" 28 29 "github.com/ethereum/go-ethereum/common" 30 "github.com/ethereum/go-ethereum/consensus/misc/eip1559" 31 "github.com/ethereum/go-ethereum/consensus/misc/eip4844" 32 "github.com/ethereum/go-ethereum/core/types" 33 "github.com/ethereum/go-ethereum/log" 34 "github.com/ethereum/go-ethereum/params" 35 "github.com/ethereum/go-ethereum/rpc" 36 ) 37 38 var ( 39 errInvalidPercentile = errors.New("invalid reward percentile") 40 errRequestBeyondHead = errors.New("request beyond head block") 41 ) 42 43 const ( 44 // maxBlockFetchers is the max number of goroutines to spin up to pull blocks 45 // for the fee history calculation (mostly relevant for LES). 46 maxBlockFetchers = 4 47 maxQueryLimit = 100 48 ) 49 50 // blockFees represents a single block for processing 51 type blockFees struct { 52 // set by the caller 53 blockNumber uint64 54 header *types.Header 55 block *types.Block // only set if reward percentiles are requested 56 receipts types.Receipts 57 // filled by processBlock 58 results processedFees 59 err error 60 } 61 62 type cacheKey struct { 63 number uint64 64 percentiles string 65 } 66 67 // processedFees contains the results of a processed block. 68 type processedFees struct { 69 reward []*big.Int 70 baseFee, nextBaseFee *big.Int 71 gasUsedRatio float64 72 blobGasUsedRatio float64 73 blobBaseFee, nextBlobBaseFee *big.Int 74 } 75 76 // txGasAndReward is sorted in ascending order based on reward 77 type txGasAndReward struct { 78 gasUsed uint64 79 reward *big.Int 80 } 81 82 // processBlock takes a blockFees structure with the blockNumber, the header and optionally 83 // the block field filled in, retrieves the block from the backend if not present yet and 84 // fills in the rest of the fields. 85 func (oracle *Oracle) processBlock(bf *blockFees, percentiles []float64) { 86 config := oracle.backend.ChainConfig() 87 88 // Fill in base fee and next base fee. 89 if bf.results.baseFee = bf.header.BaseFee; bf.results.baseFee == nil { 90 bf.results.baseFee = new(big.Int) 91 } 92 if config.IsLondon(big.NewInt(int64(bf.blockNumber + 1))) { 93 bf.results.nextBaseFee = eip1559.CalcBaseFee(config, bf.header) 94 } else { 95 bf.results.nextBaseFee = new(big.Int) 96 } 97 // Fill in blob base fee and next blob base fee. 98 if excessBlobGas := bf.header.ExcessBlobGas; excessBlobGas != nil { 99 bf.results.blobBaseFee = eip4844.CalcBlobFee(*excessBlobGas) 100 bf.results.nextBlobBaseFee = eip4844.CalcBlobFee(eip4844.CalcExcessBlobGas(*excessBlobGas, *bf.header.BlobGasUsed)) 101 } else { 102 bf.results.blobBaseFee = new(big.Int) 103 bf.results.nextBlobBaseFee = new(big.Int) 104 } 105 // Compute gas used ratio for normal and blob gas. 106 bf.results.gasUsedRatio = float64(bf.header.GasUsed) / float64(bf.header.GasLimit) 107 if blobGasUsed := bf.header.BlobGasUsed; blobGasUsed != nil { 108 bf.results.blobGasUsedRatio = float64(*blobGasUsed) / params.MaxBlobGasPerBlock 109 } 110 111 if len(percentiles) == 0 { 112 // rewards were not requested, return null 113 return 114 } 115 if bf.block == nil || (bf.receipts == nil && len(bf.block.Transactions()) != 0) { 116 log.Error("Block or receipts are missing while reward percentiles are requested") 117 return 118 } 119 120 bf.results.reward = make([]*big.Int, len(percentiles)) 121 if len(bf.block.Transactions()) == 0 { 122 // return an all zero row if there are no transactions to gather data from 123 for i := range bf.results.reward { 124 bf.results.reward[i] = new(big.Int) 125 } 126 return 127 } 128 129 sorter := make([]txGasAndReward, len(bf.block.Transactions())) 130 for i, tx := range bf.block.Transactions() { 131 reward, _ := tx.EffectiveGasTip(bf.block.BaseFee()) 132 sorter[i] = txGasAndReward{gasUsed: bf.receipts[i].GasUsed, reward: reward} 133 } 134 slices.SortStableFunc(sorter, func(a, b txGasAndReward) int { 135 return a.reward.Cmp(b.reward) 136 }) 137 138 var txIndex int 139 sumGasUsed := sorter[0].gasUsed 140 141 for i, p := range percentiles { 142 thresholdGasUsed := uint64(float64(bf.block.GasUsed()) * p / 100) 143 for sumGasUsed < thresholdGasUsed && txIndex < len(bf.block.Transactions())-1 { 144 txIndex++ 145 sumGasUsed += sorter[txIndex].gasUsed 146 } 147 bf.results.reward[i] = sorter[txIndex].reward 148 } 149 } 150 151 // resolveBlockRange resolves the specified block range to absolute block numbers while also 152 // enforcing backend specific limitations. The pending block and corresponding receipts are 153 // also returned if requested and available. 154 // Note: an error is only returned if retrieving the head header has failed. If there are no 155 // retrievable blocks in the specified range then zero block count is returned with no error. 156 func (oracle *Oracle) resolveBlockRange(ctx context.Context, reqEnd rpc.BlockNumber, blocks uint64) (*types.Block, []*types.Receipt, uint64, uint64, error) { 157 var ( 158 headBlock *types.Header 159 pendingBlock *types.Block 160 pendingReceipts types.Receipts 161 err error 162 ) 163 164 // Get the chain's current head. 165 if headBlock, err = oracle.backend.HeaderByNumber(ctx, rpc.LatestBlockNumber); err != nil { 166 return nil, nil, 0, 0, err 167 } 168 head := rpc.BlockNumber(headBlock.Number.Uint64()) 169 170 // Fail if request block is beyond the chain's current head. 171 if head < reqEnd { 172 return nil, nil, 0, 0, fmt.Errorf("%w: requested %d, head %d", errRequestBeyondHead, reqEnd, head) 173 } 174 175 // Resolve block tag. 176 if reqEnd < 0 { 177 var ( 178 resolved *types.Header 179 err error 180 ) 181 switch reqEnd { 182 case rpc.PendingBlockNumber: 183 if pendingBlock, pendingReceipts, _ = oracle.backend.Pending(); pendingBlock != nil { 184 resolved = pendingBlock.Header() 185 } else { 186 // Pending block not supported by backend, process only until latest block. 187 resolved = headBlock 188 189 // Update total blocks to return to account for this. 190 blocks-- 191 } 192 case rpc.LatestBlockNumber: 193 // Retrieved above. 194 resolved = headBlock 195 case rpc.SafeBlockNumber: 196 resolved, err = oracle.backend.HeaderByNumber(ctx, rpc.SafeBlockNumber) 197 case rpc.FinalizedBlockNumber: 198 resolved, err = oracle.backend.HeaderByNumber(ctx, rpc.FinalizedBlockNumber) 199 case rpc.EarliestBlockNumber: 200 resolved, err = oracle.backend.HeaderByNumber(ctx, rpc.EarliestBlockNumber) 201 } 202 if resolved == nil || err != nil { 203 return nil, nil, 0, 0, err 204 } 205 // Absolute number resolved. 206 reqEnd = rpc.BlockNumber(resolved.Number.Uint64()) 207 } 208 209 // If there are no blocks to return, short circuit. 210 if blocks == 0 { 211 return nil, nil, 0, 0, nil 212 } 213 // Ensure not trying to retrieve before genesis. 214 if uint64(reqEnd+1) < blocks { 215 blocks = uint64(reqEnd + 1) 216 } 217 return pendingBlock, pendingReceipts, uint64(reqEnd), blocks, nil 218 } 219 220 // FeeHistory returns data relevant for fee estimation based on the specified range of blocks. 221 // The range can be specified either with absolute block numbers or ending with the latest 222 // or pending block. Backends may or may not support gathering data from the pending block 223 // or blocks older than a certain age (specified in maxHistory). The first block of the 224 // actually processed range is returned to avoid ambiguity when parts of the requested range 225 // are not available or when the head has changed during processing this request. 226 // Five arrays are returned based on the processed blocks: 227 // - reward: the requested percentiles of effective priority fees per gas of transactions in each 228 // block, sorted in ascending order and weighted by gas used. 229 // - baseFee: base fee per gas in the given block 230 // - gasUsedRatio: gasUsed/gasLimit in the given block 231 // - blobBaseFee: the blob base fee per gas in the given block 232 // - blobGasUsedRatio: blobGasUsed/blobGasLimit in the given block 233 // 234 // Note: baseFee and blobBaseFee both include the next block after the newest of the returned range, 235 // because this value can be derived from the newest block. 236 func (oracle *Oracle) FeeHistory(ctx context.Context, blocks uint64, unresolvedLastBlock rpc.BlockNumber, rewardPercentiles []float64) (*big.Int, [][]*big.Int, []*big.Int, []float64, []*big.Int, []float64, error) { 237 if blocks < 1 { 238 return common.Big0, nil, nil, nil, nil, nil, nil // returning with no data and no error means there are no retrievable blocks 239 } 240 maxFeeHistory := oracle.maxHeaderHistory 241 if len(rewardPercentiles) != 0 { 242 maxFeeHistory = oracle.maxBlockHistory 243 } 244 if len(rewardPercentiles) > maxQueryLimit { 245 return common.Big0, nil, nil, nil, nil, nil, fmt.Errorf("%w: over the query limit %d", errInvalidPercentile, maxQueryLimit) 246 } 247 if blocks > maxFeeHistory { 248 log.Warn("Sanitizing fee history length", "requested", blocks, "truncated", maxFeeHistory) 249 blocks = maxFeeHistory 250 } 251 for i, p := range rewardPercentiles { 252 if p < 0 || p > 100 { 253 return common.Big0, nil, nil, nil, nil, nil, fmt.Errorf("%w: %f", errInvalidPercentile, p) 254 } 255 if i > 0 && p <= rewardPercentiles[i-1] { 256 return common.Big0, nil, nil, nil, nil, nil, fmt.Errorf("%w: #%d:%f >= #%d:%f", errInvalidPercentile, i-1, rewardPercentiles[i-1], i, p) 257 } 258 } 259 var ( 260 pendingBlock *types.Block 261 pendingReceipts []*types.Receipt 262 err error 263 ) 264 pendingBlock, pendingReceipts, lastBlock, blocks, err := oracle.resolveBlockRange(ctx, unresolvedLastBlock, blocks) 265 if err != nil || blocks == 0 { 266 return common.Big0, nil, nil, nil, nil, nil, err 267 } 268 oldestBlock := lastBlock + 1 - blocks 269 270 var next atomic.Uint64 271 next.Store(oldestBlock) 272 results := make(chan *blockFees, blocks) 273 274 percentileKey := make([]byte, 8*len(rewardPercentiles)) 275 for i, p := range rewardPercentiles { 276 binary.LittleEndian.PutUint64(percentileKey[i*8:(i+1)*8], math.Float64bits(p)) 277 } 278 for i := 0; i < maxBlockFetchers && i < int(blocks); i++ { 279 go func() { 280 for { 281 // Retrieve the next block number to fetch with this goroutine 282 blockNumber := next.Add(1) - 1 283 if blockNumber > lastBlock { 284 return 285 } 286 287 fees := &blockFees{blockNumber: blockNumber} 288 if pendingBlock != nil && blockNumber >= pendingBlock.NumberU64() { 289 fees.block, fees.receipts = pendingBlock, pendingReceipts 290 fees.header = fees.block.Header() 291 oracle.processBlock(fees, rewardPercentiles) 292 results <- fees 293 } else { 294 cacheKey := cacheKey{number: blockNumber, percentiles: string(percentileKey)} 295 296 if p, ok := oracle.historyCache.Get(cacheKey); ok { 297 fees.results = p 298 results <- fees 299 } else { 300 if len(rewardPercentiles) != 0 { 301 fees.block, fees.err = oracle.backend.BlockByNumber(ctx, rpc.BlockNumber(blockNumber)) 302 if fees.block != nil && fees.err == nil { 303 fees.receipts, fees.err = oracle.backend.GetReceipts(ctx, fees.block.Hash()) 304 fees.header = fees.block.Header() 305 } 306 } else { 307 fees.header, fees.err = oracle.backend.HeaderByNumber(ctx, rpc.BlockNumber(blockNumber)) 308 } 309 if fees.header != nil && fees.err == nil { 310 oracle.processBlock(fees, rewardPercentiles) 311 if fees.err == nil { 312 oracle.historyCache.Add(cacheKey, fees.results) 313 } 314 } 315 // send to results even if empty to guarantee that blocks items are sent in total 316 results <- fees 317 } 318 } 319 } 320 }() 321 } 322 var ( 323 reward = make([][]*big.Int, blocks) 324 baseFee = make([]*big.Int, blocks+1) 325 gasUsedRatio = make([]float64, blocks) 326 blobGasUsedRatio = make([]float64, blocks) 327 blobBaseFee = make([]*big.Int, blocks+1) 328 firstMissing = blocks 329 ) 330 for ; blocks > 0; blocks-- { 331 fees := <-results 332 if fees.err != nil { 333 return common.Big0, nil, nil, nil, nil, nil, fees.err 334 } 335 i := fees.blockNumber - oldestBlock 336 if fees.results.baseFee != nil { 337 reward[i], baseFee[i], baseFee[i+1], gasUsedRatio[i] = fees.results.reward, fees.results.baseFee, fees.results.nextBaseFee, fees.results.gasUsedRatio 338 blobGasUsedRatio[i], blobBaseFee[i], blobBaseFee[i+1] = fees.results.blobGasUsedRatio, fees.results.blobBaseFee, fees.results.nextBlobBaseFee 339 } else { 340 // getting no block and no error means we are requesting into the future (might happen because of a reorg) 341 if i < firstMissing { 342 firstMissing = i 343 } 344 } 345 } 346 if firstMissing == 0 { 347 return common.Big0, nil, nil, nil, nil, nil, nil 348 } 349 if len(rewardPercentiles) != 0 { 350 reward = reward[:firstMissing] 351 } else { 352 reward = nil 353 } 354 baseFee, gasUsedRatio = baseFee[:firstMissing+1], gasUsedRatio[:firstMissing] 355 blobBaseFee, blobGasUsedRatio = blobBaseFee[:firstMissing+1], blobGasUsedRatio[:firstMissing] 356 return new(big.Int).SetUint64(oldestBlock), reward, baseFee, gasUsedRatio, blobBaseFee, blobGasUsedRatio, nil 357 }