mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-19 09:04:13 +00:00
981 lines
31 KiB
Go
981 lines
31 KiB
Go
/*
|
|
Copyright 2021 Erigon contributors
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package txpool
|
|
|
|
import (
|
|
"container/heap"
|
|
"context"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"math"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/google/btree"
|
|
"github.com/hashicorp/golang-lru/simplelru"
|
|
"github.com/holiman/uint256"
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/log/v3"
|
|
"go.uber.org/atomic"
|
|
)
|
|
|
|
// Pool is interface for the transaction pool
|
|
// This interface exists for the convinience of testing, and not yet because
|
|
// there are multiple implementations
|
|
type Pool interface {
|
|
// IdHashKnown check whether transaction with given Id hash is known to the pool
|
|
IdHashKnown(hash []byte) bool
|
|
Started() bool
|
|
GetRlp(hash []byte) []byte
|
|
Add(db kv.Tx, newTxs TxSlots) error
|
|
OnNewBlock(db kv.Tx, stateChanges map[string]senderInfo, unwindTxs, minedTxs TxSlots, protocolBaseFee, pendingBaseFee, blockHeight uint64) error
|
|
|
|
AddNewGoodPeer(peerID PeerID)
|
|
}
|
|
|
|
var _ Pool = (*TxPool)(nil) // compile-time interface check
|
|
|
|
// SubPoolMarker ordered bitset responsible to sort transactions by sub-pools. Bits meaning:
|
|
// 1. Minimum fee requirement. Set to 1 if feeCap of the transaction is no less than in-protocol parameter of minimal base fee. Set to 0 if feeCap is less than minimum base fee, which means this transaction will never be included into this particular chain.
|
|
// 2. Absence of nonce gaps. Set to 1 for transactions whose nonce is N, state nonce for the sender is M, and there are transactions for all nonces between M and N from the same sender. Set to 0 is the transaction's nonce is divided from the state nonce by one or more nonce gaps.
|
|
// 3. Sufficient balance for gas. Set to 1 if the balance of sender's account in the state is B, nonce of the sender in the state is M, nonce of the transaction is N, and the sum of feeCap x gasLimit + transferred_value of all transactions from this sender with nonces N+1 ... M is no more than B. Set to 0 otherwise. In other words, this bit is set if there is currently a guarantee that the transaction and all its required prior transactions will be able to pay for gas.
|
|
// 4. Dynamic fee requirement. Set to 1 if feeCap of the transaction is no less than baseFee of the currently pending block. Set to 0 otherwise.
|
|
// 5. Local transaction. Set to 1 if transaction is local.
|
|
type SubPoolMarker uint8
|
|
|
|
const (
|
|
EnoughFeeCapProtocol = 0b10000
|
|
NoNonceGaps = 0b01000
|
|
EnoughBalance = 0b00100
|
|
EnoughFeeCapBlock = 0b00010
|
|
IsLocal = 0b00001
|
|
)
|
|
|
|
// metaTx holds transaction and some metadata
|
|
type metaTx struct {
|
|
subPool SubPoolMarker
|
|
effectiveTip uint64 // max(minTip, minFeeCap - baseFee)
|
|
Tx *TxSlot
|
|
bestIndex int
|
|
worstIndex int
|
|
currentSubPool SubPoolType
|
|
}
|
|
|
|
func newMetaTx(slot *TxSlot, isLocal bool) *metaTx {
|
|
mt := &metaTx{Tx: slot, worstIndex: -1, bestIndex: -1}
|
|
if isLocal {
|
|
mt.subPool = IsLocal
|
|
}
|
|
return mt
|
|
}
|
|
|
|
type SubPoolType uint8
|
|
|
|
const PendingSubPool SubPoolType = 1
|
|
const BaseFeeSubPool SubPoolType = 2
|
|
const QueuedSubPool SubPoolType = 3
|
|
|
|
const PendingSubPoolLimit = 1024
|
|
const BaseFeeSubPoolLimit = 1024
|
|
const QueuedSubPoolLimit = 1024
|
|
|
|
const MaxSendersInfoCache = 1024
|
|
|
|
type nonce2Tx struct{ *btree.BTree }
|
|
|
|
type senderInfo struct {
|
|
balance uint256.Int
|
|
nonce uint64
|
|
txNonce2Tx *nonce2Tx // sorted map of nonce => *metaTx
|
|
}
|
|
|
|
//nolint
|
|
func newSenderInfo(nonce uint64, balance uint256.Int) *senderInfo {
|
|
return &senderInfo{nonce: nonce, balance: balance, txNonce2Tx: &nonce2Tx{btree.New(32)}}
|
|
}
|
|
|
|
type nonce2TxItem struct{ *metaTx }
|
|
|
|
func (i *nonce2TxItem) Less(than btree.Item) bool {
|
|
return i.metaTx.Tx.nonce < than.(*nonce2TxItem).metaTx.Tx.nonce
|
|
}
|
|
|
|
// TxPool - holds all pool-related data structures and lock-based tiny methods
|
|
// most of logic implemented by pure tests-friendly functions
|
|
type TxPool struct {
|
|
lock *sync.RWMutex
|
|
|
|
blockHeight atomic.Uint64
|
|
protocolBaseFee atomic.Uint64
|
|
pendingBaseFee atomic.Uint64
|
|
|
|
senderID uint64
|
|
senderIDs map[string]uint64
|
|
senderInfo map[uint64]*senderInfo
|
|
byHash map[string]*metaTx // tx_hash => tx
|
|
pending, baseFee, queued *SubPool
|
|
|
|
// track isLocal flag of already mined transactions. used at unwind.
|
|
localsHistoryCommited time.Time
|
|
localsHistory *simplelru.LRU
|
|
db kv.RwDB
|
|
|
|
// fields for transaction propagation
|
|
recentlyConnectedPeers *recentlyConnectedPeers
|
|
newTxs chan Hashes
|
|
//lastTxPropagationTimestamp time.Time
|
|
}
|
|
|
|
func New(newTxs chan Hashes, db kv.RwDB) (*TxPool, error) {
|
|
localsHistory, err := simplelru.NewLRU(1024, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err = restoreIsLocalHistory(db, localsHistory); err != nil {
|
|
return nil, err
|
|
}
|
|
return &TxPool{
|
|
lock: &sync.RWMutex{},
|
|
senderIDs: map[string]uint64{},
|
|
senderInfo: map[uint64]*senderInfo{},
|
|
byHash: map[string]*metaTx{},
|
|
localsHistory: localsHistory,
|
|
recentlyConnectedPeers: &recentlyConnectedPeers{},
|
|
pending: NewSubPool(),
|
|
baseFee: NewSubPool(),
|
|
queued: NewSubPool(),
|
|
newTxs: newTxs,
|
|
db: db,
|
|
senderID: 1,
|
|
}, nil
|
|
}
|
|
|
|
func (p *TxPool) logStats() {
|
|
p.lock.RLock()
|
|
defer p.lock.RUnlock()
|
|
protocolBaseFee, pendingBaseFee := p.protocolBaseFee.Load(), p.pendingBaseFee.Load()
|
|
log.Info(fmt.Sprintf("[txpool] baseFee: protocol=%d,pending=%d; queues size: pending=%d/%d, baseFee=%d/%d, queued=%d/%d", protocolBaseFee, pendingBaseFee, p.pending.Len(), PendingSubPoolLimit, p.baseFee.Len(), BaseFeeSubPoolLimit, p.queued.Len(), QueuedSubPoolLimit))
|
|
}
|
|
func (p *TxPool) GetRlp(hash []byte) []byte {
|
|
p.lock.RLock()
|
|
defer p.lock.RUnlock()
|
|
|
|
txn, ok := p.byHash[string(hash)]
|
|
if !ok {
|
|
return nil
|
|
}
|
|
return txn.Tx.rlp
|
|
}
|
|
func (p *TxPool) AppendLocalHashes(buf []byte) {
|
|
p.lock.RLock()
|
|
defer p.lock.RUnlock()
|
|
for hash, txn := range p.byHash {
|
|
if txn.subPool&IsLocal == 0 {
|
|
continue
|
|
}
|
|
buf = append(buf, hash...)
|
|
}
|
|
}
|
|
func (p *TxPool) AppendRemoteHashes(buf []byte) {
|
|
p.lock.RLock()
|
|
defer p.lock.RUnlock()
|
|
|
|
for hash, txn := range p.byHash {
|
|
if txn.subPool&IsLocal != 0 {
|
|
continue
|
|
}
|
|
buf = append(buf, hash...)
|
|
}
|
|
}
|
|
func (p *TxPool) AppendAllHashes(buf []byte) {
|
|
p.AppendLocalHashes(buf)
|
|
p.AppendRemoteHashes(buf[len(buf):])
|
|
}
|
|
func (p *TxPool) IdHashKnown(hash []byte) bool {
|
|
p.lock.RLock()
|
|
defer p.lock.RUnlock()
|
|
|
|
_, ok := p.byHash[string(hash)]
|
|
return ok
|
|
}
|
|
func (p *TxPool) IdHashIsLocal(hash []byte) bool {
|
|
p.lock.RLock()
|
|
defer p.lock.RUnlock()
|
|
|
|
txn, ok := p.byHash[string(hash)]
|
|
if !ok {
|
|
return false
|
|
}
|
|
return txn.subPool&IsLocal != 0
|
|
}
|
|
func (p *TxPool) AddNewGoodPeer(peerID PeerID) { p.recentlyConnectedPeers.AddPeer(peerID) }
|
|
|
|
func (p *TxPool) Started() bool {
|
|
p.lock.Lock()
|
|
defer p.lock.Unlock()
|
|
|
|
protocolBaseFee := p.protocolBaseFee.Load()
|
|
return protocolBaseFee > 0
|
|
}
|
|
|
|
func (p *TxPool) Add(coreDB kv.Tx, newTxs TxSlots) error {
|
|
log.Debug("[txpool.add]", "txs", len(newTxs.txs))
|
|
p.lock.Lock()
|
|
defer p.lock.Unlock()
|
|
if err := newTxs.Valid(); err != nil {
|
|
return err
|
|
}
|
|
|
|
protocolBaseFee, pendingBaseFee := p.protocolBaseFee.Load(), p.pendingBaseFee.Load()
|
|
if protocolBaseFee == 0 || pendingBaseFee == 0 {
|
|
return fmt.Errorf("non-zero base fee: %d,%d", protocolBaseFee, pendingBaseFee)
|
|
}
|
|
|
|
if err := setTxSenderID(coreDB, &p.senderID, p.senderIDs, p.senderInfo, newTxs); err != nil {
|
|
return err
|
|
}
|
|
if err := onNewTxs(p.senderInfo, newTxs, protocolBaseFee, pendingBaseFee, p.pending, p.baseFee, p.queued, p.byHash, p.localsHistory); err != nil {
|
|
return err
|
|
}
|
|
|
|
notifyNewTxs := make(Hashes, 0, 32*len(newTxs.txs))
|
|
for i := range newTxs.txs {
|
|
_, ok := p.byHash[string(newTxs.txs[i].idHash[:])]
|
|
if !ok {
|
|
continue
|
|
}
|
|
notifyNewTxs = append(notifyNewTxs, newTxs.txs[i].idHash[:]...)
|
|
}
|
|
if len(notifyNewTxs) > 0 {
|
|
select {
|
|
case p.newTxs <- notifyNewTxs:
|
|
default:
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
func onNewTxs(senderInfo map[uint64]*senderInfo, newTxs TxSlots, protocolBaseFee, pendingBaseFee uint64, pending, baseFee, queued *SubPool, byHash map[string]*metaTx, localsHistory *simplelru.LRU) error {
|
|
for i := range newTxs.txs {
|
|
if newTxs.txs[i].senderID == 0 {
|
|
return fmt.Errorf("senderID can't be zero")
|
|
}
|
|
}
|
|
|
|
unsafeAddToPool(senderInfo, newTxs, queued, QueuedSubPool, func(i *metaTx) {
|
|
if _, ok := localsHistory.Get(i.Tx.idHash); ok {
|
|
//TODO: also check if sender is in list of local-senders
|
|
i.subPool |= IsLocal
|
|
}
|
|
byHash[string(i.Tx.idHash[:])] = i
|
|
replaced := senderInfo[i.Tx.senderID].txNonce2Tx.ReplaceOrInsert(&nonce2TxItem{i})
|
|
if replaced != nil {
|
|
replacedMT := replaced.(*nonce2TxItem).metaTx
|
|
delete(byHash, string(replacedMT.Tx.idHash[:]))
|
|
switch replacedMT.currentSubPool {
|
|
case PendingSubPool:
|
|
pending.UnsafeRemove(replacedMT)
|
|
case BaseFeeSubPool:
|
|
baseFee.UnsafeRemove(replacedMT)
|
|
case QueuedSubPool:
|
|
queued.UnsafeRemove(replacedMT)
|
|
default:
|
|
//already removed
|
|
}
|
|
}
|
|
})
|
|
|
|
for i := range senderInfo {
|
|
// TODO: aggregate changed senders before call this func
|
|
onSenderChange(senderInfo[i], protocolBaseFee, pendingBaseFee)
|
|
}
|
|
|
|
pending.EnforceInvariants()
|
|
baseFee.EnforceInvariants()
|
|
queued.EnforceInvariants()
|
|
|
|
promote(pending, baseFee, queued, func(i *metaTx) {
|
|
delete(byHash, string(i.Tx.idHash[:]))
|
|
senderInfo[i.Tx.senderID].txNonce2Tx.Delete(&nonce2TxItem{i})
|
|
if i.subPool&IsLocal != 0 {
|
|
//TODO: only add to history if sender is not in list of local-senders
|
|
localsHistory.Add(i.Tx.idHash, struct{}{})
|
|
}
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *TxPool) setBaseFee(protocolBaseFee, pendingBaseFee uint64) (uint64, uint64) {
|
|
p.protocolBaseFee.Store(protocolBaseFee)
|
|
hasNewVal := pendingBaseFee > 0
|
|
if pendingBaseFee < protocolBaseFee {
|
|
pendingBaseFee = protocolBaseFee
|
|
hasNewVal = true
|
|
}
|
|
if hasNewVal {
|
|
p.pendingBaseFee.Store(pendingBaseFee)
|
|
}
|
|
log.Debug("set base fee", "protocol", protocolBaseFee, "pending", pendingBaseFee)
|
|
return protocolBaseFee, p.pendingBaseFee.Load()
|
|
}
|
|
|
|
func (p *TxPool) OnNewBlock(coreDB kv.Tx, stateChanges map[string]senderInfo, unwindTxs, minedTxs TxSlots, protocolBaseFee, pendingBaseFee, blockHeight uint64) error {
|
|
log.Debug("[txpool.onNewBlock]", "unwinded", len(unwindTxs.txs), "mined", len(minedTxs.txs), "protocolBaseFee", protocolBaseFee, "blockHeight", blockHeight)
|
|
p.lock.Lock()
|
|
defer p.lock.Unlock()
|
|
p.blockHeight.Store(blockHeight)
|
|
protocolBaseFee, pendingBaseFee = p.setBaseFee(protocolBaseFee, pendingBaseFee)
|
|
if err := unwindTxs.Valid(); err != nil {
|
|
return err
|
|
}
|
|
if err := minedTxs.Valid(); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := setTxSenderID(coreDB, &p.senderID, p.senderIDs, p.senderInfo, unwindTxs); err != nil {
|
|
return err
|
|
}
|
|
if err := setTxSenderID(coreDB, &p.senderID, p.senderIDs, p.senderInfo, minedTxs); err != nil {
|
|
return err
|
|
}
|
|
for addr, id := range p.senderIDs { // merge state changes
|
|
if v, ok := stateChanges[addr]; ok {
|
|
p.senderInfo[id] = newSenderInfo(v.nonce, v.balance)
|
|
}
|
|
}
|
|
|
|
log.Debug("[txpool.onNewBlock]", "senderInfo", len(p.senderInfo))
|
|
|
|
if err := onNewBlock(p.senderInfo, unwindTxs, minedTxs.txs, protocolBaseFee, pendingBaseFee, p.pending, p.baseFee, p.queued, p.byHash, p.localsHistory); err != nil {
|
|
return err
|
|
}
|
|
|
|
notifyNewTxs := make(Hashes, 0, 32*len(unwindTxs.txs))
|
|
for i := range unwindTxs.txs {
|
|
_, ok := p.byHash[string(unwindTxs.txs[i].idHash[:])]
|
|
if !ok {
|
|
continue
|
|
}
|
|
notifyNewTxs = append(notifyNewTxs, unwindTxs.txs[i].idHash[:]...)
|
|
}
|
|
if len(notifyNewTxs) > 0 {
|
|
select {
|
|
case p.newTxs <- notifyNewTxs:
|
|
default:
|
|
}
|
|
}
|
|
|
|
/*
|
|
// evict sendersInfo without txs
|
|
if len(p.senderIDs) > MaxSendersInfoCache {
|
|
for i := range p.senderInfo {
|
|
if p.senderInfo[i].txNonce2Tx.Len() > 0 {
|
|
continue
|
|
}
|
|
for addr, id := range p.senderIDs {
|
|
if id == i {
|
|
delete(p.senderIDs, addr)
|
|
}
|
|
}
|
|
delete(p.senderInfo, i)
|
|
}
|
|
}
|
|
*/
|
|
|
|
if err := commitIsLocalHistory(p.db, p.localsHistoryCommited, p.localsHistory); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
func setTxSenderID(coreDB kv.Tx, senderIDSequence *uint64, senderIDs map[string]uint64, sendersInfo map[uint64]*senderInfo, txs TxSlots) error {
|
|
for i := range txs.txs {
|
|
addr := string(txs.senders.At(i))
|
|
|
|
// assign ID to each new sender
|
|
id, ok := senderIDs[addr]
|
|
if !ok {
|
|
*senderIDSequence++
|
|
id = *senderIDSequence
|
|
senderIDs[addr] = id
|
|
}
|
|
txs.txs[i].senderID = id
|
|
|
|
// load data from db if need
|
|
_, ok = sendersInfo[txs.txs[i].senderID]
|
|
if !ok {
|
|
encoded, err := coreDB.GetOne(kv.PlainState, txs.senders.At(i))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
nonce, balance, err := DecodeSender(encoded)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
sendersInfo[txs.txs[i].senderID] = newSenderInfo(nonce, balance)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func onNewBlock(senderInfo map[uint64]*senderInfo, unwindTxs TxSlots, minedTxs []*TxSlot, protocolBaseFee, pendingBaseFee uint64, pending, baseFee, queued *SubPool, byHash map[string]*metaTx, localsHistory *simplelru.LRU) error {
|
|
for i := range unwindTxs.txs {
|
|
if unwindTxs.txs[i].senderID == 0 {
|
|
return fmt.Errorf("onNewBlock.unwindTxs: senderID can't be zero")
|
|
}
|
|
}
|
|
for i := range minedTxs {
|
|
if minedTxs[i].senderID == 0 {
|
|
return fmt.Errorf("onNewBlock.minedTxs: senderID can't be zero")
|
|
}
|
|
}
|
|
|
|
removeMined(senderInfo, minedTxs, pending, baseFee, queued, func(i *metaTx) {
|
|
delete(byHash, string(i.Tx.idHash[:]))
|
|
senderInfo[i.Tx.senderID].txNonce2Tx.Delete(&nonce2TxItem{i})
|
|
if i.subPool&IsLocal != 0 {
|
|
//TODO: only add to history if sender is not in list of local-senders
|
|
localsHistory.Add(i.Tx.idHash, struct{}{})
|
|
}
|
|
})
|
|
|
|
// This can be thought of a reverse operation from the one described before.
|
|
// When a block that was deemed "the best" of its height, is no longer deemed "the best", the
|
|
// transactions contained in it, are now viable for inclusion in other blocks, and therefore should
|
|
// be returned into the transaction pool.
|
|
// An interesting note here is that if the block contained any transactions local to the node,
|
|
// by being first removed from the pool (from the "local" part of it), and then re-injected,
|
|
// they effective lose their priority over the "remote" transactions. In order to prevent that,
|
|
// somehow the fact that certain transactions were local, needs to be remembered for some
|
|
// time (up to some "immutability threshold").
|
|
unsafeAddToPool(senderInfo, unwindTxs, pending, PendingSubPool, func(i *metaTx) {
|
|
//fmt.Printf("add: %d,%d\n", i.Tx.senderID, i.Tx.nonce)
|
|
if _, ok := localsHistory.Get(i.Tx.idHash); ok {
|
|
//TODO: also check if sender is in list of local-senders
|
|
i.subPool |= IsLocal
|
|
}
|
|
byHash[string(i.Tx.idHash[:])] = i
|
|
replaced := senderInfo[i.Tx.senderID].txNonce2Tx.ReplaceOrInsert(&nonce2TxItem{i})
|
|
if replaced != nil {
|
|
replacedMT := replaced.(*nonce2TxItem).metaTx
|
|
delete(byHash, string(replacedMT.Tx.idHash[:]))
|
|
switch replacedMT.currentSubPool {
|
|
case PendingSubPool:
|
|
pending.UnsafeRemove(replacedMT)
|
|
case BaseFeeSubPool:
|
|
baseFee.UnsafeRemove(replacedMT)
|
|
case QueuedSubPool:
|
|
queued.UnsafeRemove(replacedMT)
|
|
default:
|
|
//already removed
|
|
}
|
|
}
|
|
})
|
|
|
|
for i := range senderInfo {
|
|
// TODO: aggregate changed senders before call this func
|
|
onSenderChange(senderInfo[i], protocolBaseFee, pendingBaseFee)
|
|
}
|
|
|
|
pending.EnforceInvariants()
|
|
baseFee.EnforceInvariants()
|
|
queued.EnforceInvariants()
|
|
|
|
promote(pending, baseFee, queued, func(i *metaTx) {
|
|
//fmt.Printf("del1 nonce: %d, %d,%d\n", i.Tx.senderID, senderInfo[i.Tx.senderID].nonce, i.Tx.nonce)
|
|
//fmt.Printf("del2 balance: %d,%d,%d\n", i.Tx.value.Uint64(), i.Tx.tip, senderInfo[i.Tx.senderID].balance.Uint64())
|
|
delete(byHash, string(i.Tx.idHash[:]))
|
|
senderInfo[i.Tx.senderID].txNonce2Tx.Delete(&nonce2TxItem{i})
|
|
if i.subPool&IsLocal != 0 {
|
|
//TODO: only add to history if sender is not in list of local-senders
|
|
localsHistory.Add(i.Tx.idHash, struct{}{})
|
|
}
|
|
})
|
|
|
|
return nil
|
|
}
|
|
|
|
// removeMined - apply new highest block (or batch of blocks)
|
|
//
|
|
// 1. New best block arrives, which potentially changes the balance and the nonce of some senders.
|
|
// We use senderIds data structure to find relevant senderId values, and then use senders data structure to
|
|
// modify state_balance and state_nonce, potentially remove some elements (if transaction with some nonce is
|
|
// included into a block), and finally, walk over the transaction records and update SubPool fields depending on
|
|
// the actual presence of nonce gaps and what the balance is.
|
|
func removeMined(senderInfo map[uint64]*senderInfo, minedTxs []*TxSlot, pending, baseFee, queued *SubPool, discard func(tx *metaTx)) {
|
|
for _, tx := range minedTxs {
|
|
sender, ok := senderInfo[tx.senderID]
|
|
if !ok {
|
|
panic("not implemented yet")
|
|
}
|
|
// delete mined transactions from everywhere
|
|
sender.txNonce2Tx.Ascend(func(i btree.Item) bool {
|
|
it := i.(*nonce2TxItem)
|
|
if it.metaTx.Tx.nonce > sender.nonce {
|
|
return false
|
|
}
|
|
// TODO: save local transactions to cache with TTL, in case of re-org - to restore isLocal flag of re-injected transactions
|
|
|
|
// del from nonce2tx mapping
|
|
sender.txNonce2Tx.Delete(i)
|
|
// del from sub-pool
|
|
switch it.metaTx.currentSubPool {
|
|
case PendingSubPool:
|
|
pending.UnsafeRemove(it.metaTx)
|
|
discard(it.metaTx)
|
|
case BaseFeeSubPool:
|
|
baseFee.UnsafeRemove(it.metaTx)
|
|
discard(it.metaTx)
|
|
case QueuedSubPool:
|
|
queued.UnsafeRemove(it.metaTx)
|
|
discard(it.metaTx)
|
|
default:
|
|
//already removed
|
|
}
|
|
return true
|
|
})
|
|
}
|
|
}
|
|
|
|
// unwind
|
|
func unsafeAddToPool(senderInfo map[uint64]*senderInfo, unwindTxs TxSlots, to *SubPool, subPoolType SubPoolType, beforeAdd func(tx *metaTx)) {
|
|
for i, tx := range unwindTxs.txs {
|
|
sender, ok := senderInfo[tx.senderID]
|
|
if !ok {
|
|
panic("not implemented yet")
|
|
}
|
|
|
|
mt := newMetaTx(tx, unwindTxs.isLocal[i])
|
|
// Insert to pending pool, if pool doesn't have tx with same Nonce and bigger Tip
|
|
if found := sender.txNonce2Tx.Get(&nonce2TxItem{mt}); found != nil {
|
|
if tx.tip <= found.(*nonce2TxItem).metaTx.Tx.tip {
|
|
continue
|
|
}
|
|
}
|
|
beforeAdd(mt)
|
|
to.UnsafeAdd(mt, subPoolType)
|
|
}
|
|
}
|
|
|
|
func onSenderChange(sender *senderInfo, protocolBaseFee, pendingBaseFee uint64) {
|
|
prevNonce := sender.nonce
|
|
cumulativeRequiredBalance := uint256.NewInt(0)
|
|
minFeeCap := uint64(math.MaxUint64)
|
|
minTip := uint64(math.MaxUint64)
|
|
sender.txNonce2Tx.Ascend(func(i btree.Item) bool {
|
|
it := i.(*nonce2TxItem)
|
|
|
|
// Sender has enough balance for: gasLimit x feeCap + transferred_value
|
|
needBalance := uint256.NewInt(it.metaTx.Tx.gas)
|
|
needBalance.Mul(needBalance, uint256.NewInt(it.metaTx.Tx.feeCap))
|
|
needBalance.Add(needBalance, &it.metaTx.Tx.value)
|
|
minFeeCap = min(minFeeCap, it.metaTx.Tx.feeCap)
|
|
minTip = min(minTip, it.metaTx.Tx.tip)
|
|
if pendingBaseFee >= minFeeCap {
|
|
it.metaTx.effectiveTip = minTip
|
|
} else {
|
|
it.metaTx.effectiveTip = minFeeCap - pendingBaseFee
|
|
}
|
|
// 1. Minimum fee requirement. Set to 1 if feeCap of the transaction is no less than in-protocol
|
|
// parameter of minimal base fee. Set to 0 if feeCap is less than minimum base fee, which means
|
|
// this transaction will never be included into this particular chain.
|
|
it.metaTx.subPool &^= EnoughFeeCapProtocol
|
|
if it.metaTx.Tx.feeCap >= protocolBaseFee {
|
|
//fmt.Printf("alex1: %d,%d,%d,%d\n", it.metaTx.NeedBalance.Uint64(), it.metaTx.Tx.gas, it.metaTx.Tx.feeCap, it.metaTx.Tx.value.Uint64())
|
|
//fmt.Printf("alex2: %d,%t\n", sender.balance.Uint64(), it.metaTx.SenderHasEnoughBalance)
|
|
it.metaTx.subPool |= EnoughFeeCapProtocol
|
|
}
|
|
|
|
// 2. Absence of nonce gaps. Set to 1 for transactions whose nonce is N, state nonce for
|
|
// the sender is M, and there are transactions for all nonces between M and N from the same
|
|
// sender. Set to 0 is the transaction's nonce is divided from the state nonce by one or more nonce gaps.
|
|
it.metaTx.subPool &^= NoNonceGaps
|
|
if uint64(prevNonce)+1 == it.metaTx.Tx.nonce {
|
|
it.metaTx.subPool |= NoNonceGaps
|
|
prevNonce = it.Tx.nonce
|
|
}
|
|
|
|
// 3. Sufficient balance for gas. Set to 1 if the balance of sender's account in the
|
|
// state is B, nonce of the sender in the state is M, nonce of the transaction is N, and the
|
|
// sum of feeCap x gasLimit + transferred_value of all transactions from this sender with
|
|
// nonces N+1 ... M is no more than B. Set to 0 otherwise. In other words, this bit is
|
|
// set if there is currently a guarantee that the transaction and all its required prior
|
|
// transactions will be able to pay for gas.
|
|
cumulativeRequiredBalance = cumulativeRequiredBalance.Add(cumulativeRequiredBalance, needBalance) // already deleted all transactions with nonce <= sender.nonce
|
|
it.metaTx.subPool &^= EnoughBalance
|
|
if sender.balance.Gt(cumulativeRequiredBalance) || sender.balance.Eq(cumulativeRequiredBalance) {
|
|
it.metaTx.subPool |= EnoughBalance
|
|
}
|
|
|
|
// 4. Dynamic fee requirement. Set to 1 if feeCap of the transaction is no less than
|
|
// baseFee of the currently pending block. Set to 0 otherwise.
|
|
it.metaTx.subPool &^= EnoughFeeCapBlock
|
|
if it.metaTx.Tx.feeCap >= pendingBaseFee {
|
|
it.metaTx.subPool |= EnoughFeeCapBlock
|
|
}
|
|
|
|
// 5. Local transaction. Set to 1 if transaction is local.
|
|
// can't change
|
|
|
|
return true
|
|
})
|
|
}
|
|
|
|
func promote(pending, baseFee, queued *SubPool, discard func(tx *metaTx)) {
|
|
//1. If top element in the worst green queue has subPool != 0b1111 (binary), it needs to be removed from the green pool.
|
|
// If subPool < 0b1000 (not satisfying minimum fee), discard.
|
|
// If subPool == 0b1110, demote to the yellow pool, otherwise demote to the red pool.
|
|
for worst := pending.Worst(); pending.Len() > 0; worst = pending.Worst() {
|
|
if worst.subPool >= 0b11110 {
|
|
break
|
|
}
|
|
if worst.subPool >= 0b11100 {
|
|
baseFee.Add(pending.PopWorst(), BaseFeeSubPool)
|
|
continue
|
|
}
|
|
if worst.subPool >= 0b10000 {
|
|
queued.Add(pending.PopWorst(), QueuedSubPool)
|
|
continue
|
|
}
|
|
discard(pending.PopWorst())
|
|
}
|
|
|
|
//2. If top element in the worst green queue has subPool == 0b1111, but there is not enough room in the pool, discard.
|
|
for worst := pending.Worst(); pending.Len() > PendingSubPoolLimit; worst = pending.Worst() {
|
|
if worst.subPool >= 0b11111 { // TODO: here must 'subPool == 0b1111' or 'subPool <= 0b1111' ?
|
|
break
|
|
}
|
|
pending.PopWorst()
|
|
}
|
|
|
|
//3. If the top element in the best yellow queue has subPool == 0b1111, promote to the green pool.
|
|
for best := baseFee.Best(); baseFee.Len() > 0; best = baseFee.Best() {
|
|
if best.subPool < 0b11110 {
|
|
break
|
|
}
|
|
pending.Add(baseFee.PopBest(), PendingSubPool)
|
|
}
|
|
|
|
//4. If the top element in the worst yellow queue has subPool != 0x1110, it needs to be removed from the yellow pool.
|
|
// If subPool < 0b1000 (not satisfying minimum fee), discard. Otherwise, demote to the red pool.
|
|
for worst := baseFee.Worst(); baseFee.Len() > 0; worst = baseFee.Worst() {
|
|
if worst.subPool >= 0b11100 {
|
|
break
|
|
}
|
|
if worst.subPool >= 0b10000 {
|
|
queued.Add(baseFee.PopWorst(), QueuedSubPool)
|
|
continue
|
|
}
|
|
discard(baseFee.PopWorst())
|
|
}
|
|
|
|
//5. If the top element in the worst yellow queue has subPool == 0x1110, but there is not enough room in the pool, discard.
|
|
for worst := baseFee.Worst(); baseFee.Len() > BaseFeeSubPoolLimit; worst = baseFee.Worst() {
|
|
if worst.subPool >= 0b11110 {
|
|
break
|
|
}
|
|
discard(baseFee.PopWorst())
|
|
}
|
|
|
|
//6. If the top element in the best red queue has subPool == 0x1110, promote to the yellow pool. If subPool == 0x1111, promote to the green pool.
|
|
for best := queued.Best(); queued.Len() > 0; best = queued.Best() {
|
|
if best.subPool < 0b11100 {
|
|
break
|
|
}
|
|
if best.subPool < 0b11110 {
|
|
baseFee.Add(queued.PopBest(), BaseFeeSubPool)
|
|
continue
|
|
}
|
|
|
|
pending.Add(queued.PopBest(), PendingSubPool)
|
|
}
|
|
|
|
//7. If the top element in the worst red queue has subPool < 0b1000 (not satisfying minimum fee), discard.
|
|
for worst := queued.Worst(); queued.Len() > 0; worst = queued.Worst() {
|
|
if worst.subPool >= 0b10000 {
|
|
break
|
|
}
|
|
|
|
discard(queued.PopWorst())
|
|
}
|
|
|
|
//8. If the top element in the worst red queue has subPool >= 0b100, but there is not enough room in the pool, discard.
|
|
for _ = queued.Worst(); queued.Len() > QueuedSubPoolLimit; _ = queued.Worst() {
|
|
discard(queued.PopWorst())
|
|
}
|
|
}
|
|
|
|
type SubPool struct {
|
|
best *BestQueue
|
|
worst *WorstQueue
|
|
}
|
|
|
|
func NewSubPool() *SubPool {
|
|
return &SubPool{best: &BestQueue{}, worst: &WorstQueue{}}
|
|
}
|
|
|
|
func (p *SubPool) EnforceInvariants() {
|
|
heap.Init(p.worst)
|
|
heap.Init(p.best)
|
|
}
|
|
func (p *SubPool) Best() *metaTx {
|
|
if len(*p.best) == 0 {
|
|
return nil
|
|
}
|
|
return (*p.best)[0]
|
|
}
|
|
func (p *SubPool) Worst() *metaTx {
|
|
if len(*p.worst) == 0 {
|
|
return nil
|
|
}
|
|
return (*p.worst)[0]
|
|
}
|
|
func (p *SubPool) PopBest() *metaTx {
|
|
i := heap.Pop(p.best).(*metaTx)
|
|
heap.Remove(p.worst, i.worstIndex)
|
|
return i
|
|
}
|
|
func (p *SubPool) PopWorst() *metaTx {
|
|
i := heap.Pop(p.worst).(*metaTx)
|
|
heap.Remove(p.best, i.bestIndex)
|
|
return i
|
|
}
|
|
func (p *SubPool) Len() int { return p.best.Len() }
|
|
func (p *SubPool) Add(i *metaTx, subPoolType SubPoolType) {
|
|
i.currentSubPool = subPoolType
|
|
heap.Push(p.best, i)
|
|
heap.Push(p.worst, i)
|
|
}
|
|
|
|
// UnsafeRemove - does break Heap invariants, but it has O(1) instead of O(log(n)) complexity.
|
|
// Must manually call heap.Init after such changes.
|
|
// Make sense to batch unsafe changes
|
|
func (p *SubPool) UnsafeRemove(i *metaTx) {
|
|
if p.Len() == 0 {
|
|
return
|
|
}
|
|
if p.Len() == 1 && i.bestIndex == 0 {
|
|
p.worst.Pop()
|
|
p.best.Pop()
|
|
return
|
|
}
|
|
// manually call funcs instead of heap.Pop
|
|
p.worst.Swap(i.worstIndex, p.worst.Len()-1)
|
|
p.worst.Pop()
|
|
p.best.Swap(i.bestIndex, p.best.Len()-1)
|
|
p.best.Pop()
|
|
}
|
|
func (p *SubPool) UnsafeAdd(i *metaTx, subPoolType SubPoolType) {
|
|
i.currentSubPool = subPoolType
|
|
p.worst.Push(i)
|
|
p.best.Push(i)
|
|
}
|
|
func (p *SubPool) DebugPrint() {
|
|
for i, it := range *p.best {
|
|
fmt.Printf("best: %d, %d, %d\n", i, it.subPool, it.bestIndex)
|
|
}
|
|
for i, it := range *p.worst {
|
|
fmt.Printf("worst: %d, %d, %d\n", i, it.subPool, it.worstIndex)
|
|
}
|
|
}
|
|
|
|
type BestQueue []*metaTx
|
|
|
|
func (mt *metaTx) Less(than *metaTx) bool {
|
|
if mt.subPool != than.subPool {
|
|
return mt.subPool < than.subPool
|
|
}
|
|
|
|
if mt.effectiveTip != than.effectiveTip {
|
|
return mt.effectiveTip < than.effectiveTip
|
|
}
|
|
|
|
if mt.Tx.nonce != than.Tx.nonce {
|
|
return mt.Tx.nonce < than.Tx.nonce
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (p BestQueue) Len() int { return len(p) }
|
|
func (p BestQueue) Less(i, j int) bool { return !p[i].Less(p[j]) } // We want Pop to give us the highest, not lowest, priority so we use !less here.
|
|
func (p BestQueue) Swap(i, j int) {
|
|
p[i], p[j] = p[j], p[i]
|
|
p[i].bestIndex = i
|
|
p[j].bestIndex = j
|
|
}
|
|
func (p *BestQueue) Push(x interface{}) {
|
|
n := len(*p)
|
|
item := x.(*metaTx)
|
|
item.bestIndex = n
|
|
*p = append(*p, item)
|
|
}
|
|
|
|
func (p *BestQueue) Pop() interface{} {
|
|
old := *p
|
|
n := len(old)
|
|
item := old[n-1]
|
|
old[n-1] = nil // avoid memory leak
|
|
item.bestIndex = -1 // for safety
|
|
item.currentSubPool = 0 // for safety
|
|
*p = old[0 : n-1]
|
|
return item
|
|
}
|
|
|
|
type WorstQueue []*metaTx
|
|
|
|
func (p WorstQueue) Len() int { return len(p) }
|
|
func (p WorstQueue) Less(i, j int) bool { return p[i].Less(p[j]) }
|
|
func (p WorstQueue) Swap(i, j int) {
|
|
p[i], p[j] = p[j], p[i]
|
|
p[i].worstIndex = i
|
|
p[j].worstIndex = j
|
|
}
|
|
func (p *WorstQueue) Push(x interface{}) {
|
|
n := len(*p)
|
|
item := x.(*metaTx)
|
|
item.worstIndex = n
|
|
*p = append(*p, x.(*metaTx))
|
|
}
|
|
func (p *WorstQueue) Pop() interface{} {
|
|
old := *p
|
|
n := len(old)
|
|
item := old[n-1]
|
|
old[n-1] = nil // avoid memory leak
|
|
item.worstIndex = -1 // for safety
|
|
item.currentSubPool = 0 // for safety
|
|
*p = old[0 : n-1]
|
|
return item
|
|
}
|
|
|
|
// BroadcastLoop - does:
|
|
// send pending byHash to p2p:
|
|
// - new byHash
|
|
// - all pooled byHash to recently connected peers
|
|
// - all local pooled byHash to random peers periodically
|
|
// promote/demote transactions
|
|
// reorgs
|
|
func BroadcastLoop(ctx context.Context, p *TxPool, newTxs chan Hashes, send *Send, timings Timings) {
|
|
logEvery := time.NewTicker(timings.logEvery)
|
|
defer logEvery.Stop()
|
|
|
|
syncToNewPeersEvery := time.NewTicker(timings.syncToNewPeersEvery)
|
|
defer syncToNewPeersEvery.Stop()
|
|
|
|
localTxHashes := make([]byte, 0, 128)
|
|
remoteTxHashes := make([]byte, 0, 128)
|
|
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
case <-logEvery.C:
|
|
p.logStats()
|
|
case h := <-newTxs:
|
|
// first broadcast all local txs to all peers, then non-local to random sqrt(peersAmount) peers
|
|
localTxHashes = localTxHashes[:0]
|
|
remoteTxHashes = remoteTxHashes[:0]
|
|
|
|
for i := 0; i < h.Len(); i++ {
|
|
if p.IdHashIsLocal(h.At(i)) {
|
|
localTxHashes = append(localTxHashes, h.At(i)...)
|
|
} else {
|
|
remoteTxHashes = append(localTxHashes, h.At(i)...)
|
|
}
|
|
}
|
|
|
|
send.BroadcastLocalPooledTxs(localTxHashes)
|
|
send.BroadcastRemotePooledTxs(remoteTxHashes)
|
|
case <-syncToNewPeersEvery.C: // new peer
|
|
newPeers := p.recentlyConnectedPeers.GetAndClean()
|
|
if len(newPeers) == 0 {
|
|
continue
|
|
}
|
|
p.AppendAllHashes(remoteTxHashes[:0])
|
|
send.PropagatePooledTxsToPeersList(newPeers, remoteTxHashes)
|
|
}
|
|
}
|
|
}
|
|
|
|
// commitIsLocalHistory - use u64 sequence as keys to preserve order
|
|
func commitIsLocalHistory(db kv.RwDB, commited time.Time, localsHistory *simplelru.LRU) error {
|
|
if db == nil || time.Since(commited) < 30*time.Second {
|
|
return nil
|
|
}
|
|
txHashes := localsHistory.Keys()
|
|
key := make([]byte, 8)
|
|
return db.Update(context.Background(), func(tx kv.RwTx) error {
|
|
if err := tx.ClearBucket(kv.RecentLocalTransactions); err != nil {
|
|
return err
|
|
}
|
|
for i := range txHashes {
|
|
binary.BigEndian.PutUint64(key, uint64(i))
|
|
if err := tx.Append(kv.RecentLocalTransactions, key, txHashes[i].([]byte)); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
}
|
|
|
|
func restoreIsLocalHistory(db kv.RwDB, localsHistory *simplelru.LRU) error {
|
|
if db == nil {
|
|
return nil
|
|
}
|
|
return db.View(context.Background(), func(tx kv.Tx) error {
|
|
return tx.ForPrefix(kv.RecentLocalTransactions, nil, func(k, v []byte) error {
|
|
localsHistory.Add(copyBytes(v), struct{}{})
|
|
return nil
|
|
})
|
|
})
|
|
}
|
|
|
|
func copyBytes(b []byte) (copiedBytes []byte) {
|
|
copiedBytes = make([]byte, len(b))
|
|
copy(copiedBytes, b)
|
|
return
|
|
}
|
|
|
|
// recentlyConnectedPeers does buffer IDs of recently connected good peers
|
|
// then sync of pooled Transaction can happen to all of then at once
|
|
// DoS protection and performance saving
|
|
// it doesn't track if peer disconnected, it's fine
|
|
type recentlyConnectedPeers struct {
|
|
lock sync.RWMutex
|
|
peers []PeerID
|
|
}
|
|
|
|
func (l *recentlyConnectedPeers) AddPeer(p PeerID) {
|
|
l.lock.Lock()
|
|
defer l.lock.Unlock()
|
|
l.peers = append(l.peers, p)
|
|
}
|
|
|
|
func (l *recentlyConnectedPeers) GetAndClean() []PeerID {
|
|
l.lock.Lock()
|
|
defer l.lock.Unlock()
|
|
peers := l.peers
|
|
l.peers = nil
|
|
return peers
|
|
}
|
|
|
|
func min(a, b uint64) uint64 {
|
|
if a <= b {
|
|
return a
|
|
}
|
|
return b
|
|
}
|