mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-17 16:18:47 +00:00
273 lines
8.4 KiB
Go
273 lines
8.4 KiB
Go
// Copyright 2015 The go-ethereum Authors
|
|
// This file is part of the go-ethereum library.
|
|
//
|
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// The go-ethereum library 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 Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package gasprice
|
|
|
|
import (
|
|
"container/heap"
|
|
"context"
|
|
"errors"
|
|
"math/big"
|
|
"sync"
|
|
|
|
"github.com/holiman/uint256"
|
|
"github.com/ledgerwatch/erigon/common"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/params"
|
|
"github.com/ledgerwatch/erigon/rpc"
|
|
"github.com/ledgerwatch/log/v3"
|
|
)
|
|
|
|
const sampleNumber = 3 // Number of transactions sampled in a block
|
|
|
|
var (
|
|
DefaultMaxPrice = big.NewInt(500 * params.GWei)
|
|
DefaultIgnorePrice = big.NewInt(2 * params.Wei)
|
|
)
|
|
|
|
type Config struct {
|
|
Blocks int
|
|
Percentile int
|
|
Default *big.Int `toml:",omitempty"`
|
|
MaxPrice *big.Int `toml:",omitempty"`
|
|
IgnorePrice *big.Int `toml:",omitempty"`
|
|
}
|
|
|
|
// OracleBackend includes all necessary background APIs for oracle.
|
|
type OracleBackend interface {
|
|
HeaderByNumber(ctx context.Context, number rpc.BlockNumber) (*types.Header, error)
|
|
BlockByNumber(ctx context.Context, number rpc.BlockNumber) (*types.Block, error)
|
|
ChainConfig() *params.ChainConfig
|
|
}
|
|
|
|
// Oracle recommends gas prices based on the content of recent
|
|
// blocks. Suitable for both light and full clients.
|
|
type Oracle struct {
|
|
backend OracleBackend
|
|
lastHead common.Hash
|
|
lastPrice *big.Int
|
|
maxPrice *big.Int
|
|
ignorePrice *big.Int
|
|
cacheLock sync.RWMutex
|
|
|
|
checkBlocks int
|
|
percentile int
|
|
}
|
|
|
|
// NewOracle returns a new gasprice oracle which can recommend suitable
|
|
// gasprice for newly created transaction.
|
|
func NewOracle(backend OracleBackend, params Config) *Oracle {
|
|
blocks := params.Blocks
|
|
if blocks < 1 {
|
|
blocks = 1
|
|
log.Warn("Sanitizing invalid gasprice oracle sample blocks", "provided", params.Blocks, "updated", blocks)
|
|
}
|
|
percent := params.Percentile
|
|
if percent < 0 {
|
|
percent = 0
|
|
log.Warn("Sanitizing invalid gasprice oracle sample percentile", "provided", params.Percentile, "updated", percent)
|
|
}
|
|
if percent > 100 {
|
|
percent = 100
|
|
log.Warn("Sanitizing invalid gasprice oracle sample percentile", "provided", params.Percentile, "updated", percent)
|
|
}
|
|
maxPrice := params.MaxPrice
|
|
if maxPrice == nil || maxPrice.Int64() <= 0 {
|
|
maxPrice = DefaultMaxPrice
|
|
log.Warn("Sanitizing invalid gasprice oracle price cap", "provided", params.MaxPrice, "updated", maxPrice)
|
|
}
|
|
ignorePrice := params.IgnorePrice
|
|
if ignorePrice == nil || ignorePrice.Int64() < 0 {
|
|
ignorePrice = DefaultIgnorePrice
|
|
log.Warn("Sanitizing invalid gasprice oracle ignore price", "provided", params.IgnorePrice, "updated", ignorePrice)
|
|
}
|
|
return &Oracle{
|
|
backend: backend,
|
|
lastPrice: params.Default,
|
|
maxPrice: maxPrice,
|
|
ignorePrice: ignorePrice,
|
|
checkBlocks: blocks,
|
|
percentile: percent,
|
|
}
|
|
}
|
|
|
|
// SuggestPrice returns a TipCap so that newly created transaction can
|
|
// have a very high chance to be included in the following blocks.
|
|
// NODE: if caller wants legacy tx SuggestedPrice, we need to add
|
|
// baseFee to the returned bigInt
|
|
func (gpo *Oracle) SuggestPrice(ctx context.Context) (*big.Int, error) {
|
|
head, _ := gpo.backend.HeaderByNumber(ctx, rpc.LatestBlockNumber)
|
|
headHash := head.Hash()
|
|
|
|
// If the latest gasprice is still available, return it.
|
|
gpo.cacheLock.RLock()
|
|
lastHead, lastPrice := gpo.lastHead, gpo.lastPrice
|
|
gpo.cacheLock.RUnlock()
|
|
if headHash == lastHead {
|
|
return lastPrice, nil
|
|
}
|
|
|
|
// Try checking the cache again, maybe the last fetch fetched what we need
|
|
gpo.cacheLock.RLock()
|
|
lastHead, lastPrice = gpo.lastHead, gpo.lastPrice
|
|
gpo.cacheLock.RUnlock()
|
|
if headHash == lastHead {
|
|
return lastPrice, nil
|
|
}
|
|
number := head.Number.Uint64()
|
|
txPrices := make(sortingHeap, 0, sampleNumber*gpo.checkBlocks)
|
|
for txPrices.Len() < sampleNumber*gpo.checkBlocks && number > 0 {
|
|
err := gpo.getBlockPrices(ctx, number, sampleNumber, gpo.ignorePrice, &txPrices)
|
|
if err != nil {
|
|
return lastPrice, err
|
|
}
|
|
number--
|
|
}
|
|
price := lastPrice
|
|
if txPrices.Len() > 0 {
|
|
// Item with this position needs to be extracted from the sorting heap
|
|
// so we pop all the items before it
|
|
percentilePosition := (txPrices.Len() - 1) * gpo.percentile / 100
|
|
for i := 0; i < percentilePosition; i++ {
|
|
heap.Pop(&txPrices)
|
|
}
|
|
}
|
|
if txPrices.Len() > 0 {
|
|
// Don't need to pop it, just take from the top of the heap
|
|
price = txPrices[0].ToBig()
|
|
}
|
|
if price.Cmp(gpo.maxPrice) > 0 {
|
|
price = new(big.Int).Set(gpo.maxPrice)
|
|
}
|
|
gpo.cacheLock.Lock()
|
|
gpo.lastHead = headHash
|
|
gpo.lastPrice = price
|
|
gpo.cacheLock.Unlock()
|
|
return price, nil
|
|
}
|
|
|
|
type transactionsByGasPrice struct {
|
|
txs []types.Transaction
|
|
baseFee *uint256.Int
|
|
}
|
|
|
|
func newTransactionsByGasPrice(txs []types.Transaction,
|
|
baseFee *uint256.Int) transactionsByGasPrice {
|
|
return transactionsByGasPrice{
|
|
txs: txs,
|
|
baseFee: baseFee,
|
|
}
|
|
}
|
|
|
|
func (t transactionsByGasPrice) Len() int { return len(t.txs) }
|
|
func (t transactionsByGasPrice) Swap(i, j int) { t.txs[i], t.txs[j] = t.txs[j], t.txs[i] }
|
|
func (t transactionsByGasPrice) Less(i, j int) bool {
|
|
tip1 := t.txs[i].GetEffectiveGasTip(t.baseFee)
|
|
tip2 := t.txs[j].GetEffectiveGasTip(t.baseFee)
|
|
return tip1.Lt(tip2)
|
|
}
|
|
|
|
// Push (part of heap.Interface) places a new link onto the end of queue
|
|
func (t *transactionsByGasPrice) Push(x interface{}) {
|
|
// Push and Pop use pointer receivers because they modify the slice's length,
|
|
// not just its contents.
|
|
l := x.(types.Transaction)
|
|
t.txs = append(t.txs, l)
|
|
}
|
|
|
|
// Pop (part of heap.Interface) removes the first link from the queue
|
|
func (t *transactionsByGasPrice) Pop() interface{} {
|
|
old := t.txs
|
|
n := len(old)
|
|
x := old[n-1]
|
|
t.txs = old[0 : n-1]
|
|
return x
|
|
}
|
|
|
|
// getBlockPrices calculates the lowest transaction gas price in a given block.
|
|
// the block is empty or all transactions are sent by the miner
|
|
// itself(it doesn't make any sense to include this kind of transaction prices for sampling),
|
|
// nil gasprice is returned.
|
|
func (gpo *Oracle) getBlockPrices(ctx context.Context, blockNum uint64, limit int,
|
|
ingoreUnderBig *big.Int, s *sortingHeap) error {
|
|
ignoreUnder, overflow := uint256.FromBig(ingoreUnderBig)
|
|
if overflow {
|
|
err := errors.New("overflow in getBlockPrices, gasprice.go: ignoreUnder too large")
|
|
log.Error("gasprice.go: getBlockPrices", "error", err)
|
|
return err
|
|
}
|
|
block, err := gpo.backend.BlockByNumber(ctx, rpc.BlockNumber(blockNum))
|
|
if block == nil {
|
|
log.Error("gasprice.go: getBlockPrices", "error", err)
|
|
return err
|
|
}
|
|
blockTxs := block.Transactions()
|
|
plainTxs := make([]types.Transaction, len(blockTxs))
|
|
copy(plainTxs, blockTxs)
|
|
var baseFee *uint256.Int
|
|
if block.BaseFee() == nil {
|
|
baseFee = nil
|
|
} else {
|
|
baseFee, overflow = uint256.FromBig(block.BaseFee())
|
|
if overflow {
|
|
err := errors.New("overflow in getBlockPrices, gasprice.go: baseFee > 2^256-1")
|
|
log.Error("gasprice.go: getBlockPrices", "error", err)
|
|
return err
|
|
}
|
|
}
|
|
txs := newTransactionsByGasPrice(plainTxs, baseFee)
|
|
heap.Init(&txs)
|
|
|
|
for txs.Len() > 0 {
|
|
tx := heap.Pop(&txs).(types.Transaction)
|
|
tip := tx.GetEffectiveGasTip(baseFee)
|
|
if ignoreUnder != nil && tip.Lt(ignoreUnder) {
|
|
continue
|
|
}
|
|
sender, _ := tx.GetSender()
|
|
if err == nil && sender != block.Coinbase() {
|
|
heap.Push(s, tip)
|
|
if s.Len() >= limit {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type sortingHeap []*uint256.Int
|
|
|
|
func (s sortingHeap) Len() int { return len(s) }
|
|
func (s sortingHeap) Less(i, j int) bool { return s[i].Lt(s[j]) }
|
|
func (s sortingHeap) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
|
|
|
|
// Push (part of heap.Interface) places a new link onto the end of queue
|
|
func (s *sortingHeap) Push(x interface{}) {
|
|
// Push and Pop use pointer receivers because they modify the slice's length,
|
|
// not just its contents.
|
|
l := x.(*uint256.Int)
|
|
*s = append(*s, l)
|
|
}
|
|
|
|
// Pop (part of heap.Interface) removes the first link from the queue
|
|
func (s *sortingHeap) Pop() interface{} {
|
|
old := *s
|
|
n := len(old)
|
|
x := old[n-1]
|
|
*s = old[0 : n-1]
|
|
return x
|
|
}
|