mirror of
https://github.com/ethereum/go-ethereum.git
synced 2026-05-24 08:49:29 +00:00
order = order[1:] reslices without releasing the backing array. After N total insertions the array retains N hashes (32 bytes each) but only the last maxTracked are live. On a long-running node processing ~100 txs/s this leaks ~275 MB/day. Compact by copying to a fresh array when capacity exceeds 2×maxTracked.
246 lines
7 KiB
Go
246 lines
7 KiB
Go
// Copyright 2026 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 txtracker provides minimal per-peer transaction inclusion tracking.
|
|
//
|
|
// It records which peer delivered each accepted transaction (via NotifyAccepted)
|
|
// and monitors the chain for inclusion and finalization events. When a
|
|
// delivered transaction is finalized on chain, the delivering peer is
|
|
// credited. A per-block exponential moving average (EMA) of inclusions
|
|
// tracks recent peer productivity.
|
|
//
|
|
// The primary consumer is the peer dropper (eth/dropper.go), which uses
|
|
// these stats to protect high-value peers from random disconnection.
|
|
package txtracker
|
|
|
|
import (
|
|
"sync"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/core"
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
"github.com/ethereum/go-ethereum/event"
|
|
"github.com/ethereum/go-ethereum/log"
|
|
)
|
|
|
|
const (
|
|
// Maximum number of tx→deliverer mappings to retain.
|
|
maxTracked = 262144
|
|
// EMA smoothing factor for per-block inclusion rate.
|
|
emaAlpha = 0.05
|
|
)
|
|
|
|
// PeerStats holds the per-peer inclusion data.
|
|
type PeerStats struct {
|
|
Finalized int64 // Cumulative finalized inclusions attributed to this peer
|
|
RecentIncluded float64 // EMA of per-block inclusions (at chain head time)
|
|
}
|
|
|
|
// Chain is the blockchain interface needed by the tracker.
|
|
type Chain interface {
|
|
SubscribeChainHeadEvent(ch chan<- core.ChainHeadEvent) event.Subscription
|
|
GetBlockByNumber(number uint64) *types.Block
|
|
GetBlock(hash common.Hash, number uint64) *types.Block
|
|
CurrentFinalBlock() *types.Header
|
|
}
|
|
|
|
type peerStats struct {
|
|
finalized int64
|
|
recentIncluded float64
|
|
}
|
|
|
|
// Tracker records which peer delivered each transaction and credits peers
|
|
// when their transactions appear on chain.
|
|
type Tracker struct {
|
|
mu sync.Mutex
|
|
txs map[common.Hash]string // hash → deliverer peer ID
|
|
peers map[string]*peerStats
|
|
order []common.Hash // insertion order for LRU eviction
|
|
|
|
chain Chain
|
|
lastFinalNum uint64 // last finalized block number processed
|
|
headCh chan core.ChainHeadEvent
|
|
sub event.Subscription
|
|
|
|
quit chan struct{}
|
|
wg sync.WaitGroup
|
|
}
|
|
|
|
// New creates a new tracker.
|
|
func New() *Tracker {
|
|
return &Tracker{
|
|
txs: make(map[common.Hash]string),
|
|
peers: make(map[string]*peerStats),
|
|
quit: make(chan struct{}),
|
|
}
|
|
}
|
|
|
|
// Start begins listening for chain head events.
|
|
func (t *Tracker) Start(chain Chain) {
|
|
t.chain = chain
|
|
// Seed lastFinalNum so checkFinalization doesn't backfill from genesis.
|
|
if fh := chain.CurrentFinalBlock(); fh != nil {
|
|
t.lastFinalNum = fh.Number.Uint64()
|
|
}
|
|
t.headCh = make(chan core.ChainHeadEvent, 128)
|
|
t.sub = chain.SubscribeChainHeadEvent(t.headCh)
|
|
t.wg.Add(1)
|
|
go t.loop()
|
|
}
|
|
|
|
// NotifyPeerDrop removes a disconnected peer's stats to prevent unbounded
|
|
// growth. Safe to call from any goroutine.
|
|
func (t *Tracker) NotifyPeerDrop(peer string) {
|
|
t.mu.Lock()
|
|
defer t.mu.Unlock()
|
|
delete(t.peers, peer)
|
|
}
|
|
|
|
// Stop shuts down the tracker.
|
|
func (t *Tracker) Stop() {
|
|
t.sub.Unsubscribe()
|
|
close(t.quit)
|
|
t.wg.Wait()
|
|
}
|
|
|
|
// NotifyAccepted records that a peer delivered transactions that were accepted
|
|
// by the pool. Only accepted (not rejected/duplicate) txs should be recorded
|
|
// to prevent attribution poisoning from replayed or invalid txs.
|
|
// Safe to call from any goroutine.
|
|
func (t *Tracker) NotifyAccepted(peer string, hashes []common.Hash) {
|
|
t.mu.Lock()
|
|
defer t.mu.Unlock()
|
|
|
|
for _, hash := range hashes {
|
|
if _, ok := t.txs[hash]; ok {
|
|
continue // already tracked, keep first deliverer
|
|
}
|
|
t.txs[hash] = peer
|
|
t.order = append(t.order, hash)
|
|
}
|
|
// Evict oldest entries if over capacity.
|
|
for len(t.txs) > maxTracked {
|
|
oldest := t.order[0]
|
|
t.order = t.order[1:]
|
|
delete(t.txs, oldest)
|
|
}
|
|
// Compact the backing array when it grows too large. Reslicing
|
|
// with order[1:] doesn't free earlier slots in the array.
|
|
if cap(t.order) > 2*maxTracked {
|
|
t.order = append([]common.Hash(nil), t.order...)
|
|
}
|
|
}
|
|
|
|
// GetAllPeerStats returns a snapshot of per-peer inclusion statistics.
|
|
// Safe to call from any goroutine.
|
|
func (t *Tracker) GetAllPeerStats() map[string]PeerStats {
|
|
t.mu.Lock()
|
|
defer t.mu.Unlock()
|
|
|
|
result := make(map[string]PeerStats, len(t.peers))
|
|
for id, ps := range t.peers {
|
|
result[id] = PeerStats{
|
|
Finalized: ps.finalized,
|
|
RecentIncluded: ps.recentIncluded,
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
func (t *Tracker) loop() {
|
|
defer t.wg.Done()
|
|
|
|
for {
|
|
select {
|
|
case ev := <-t.headCh:
|
|
t.handleChainHead(ev)
|
|
case <-t.sub.Err():
|
|
return
|
|
case <-t.quit:
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (t *Tracker) handleChainHead(ev core.ChainHeadEvent) {
|
|
// Fetch the head block by hash (not just number) to avoid using a
|
|
// reorged block if the tracker goroutine lags behind the chain.
|
|
block := t.chain.GetBlock(ev.Header.Hash(), ev.Header.Number.Uint64())
|
|
if block == nil {
|
|
return
|
|
}
|
|
t.mu.Lock()
|
|
defer t.mu.Unlock()
|
|
|
|
// Count per-peer inclusions in this block for the EMA.
|
|
blockIncl := make(map[string]int)
|
|
for _, tx := range block.Transactions() {
|
|
if peer := t.txs[tx.Hash()]; peer != "" {
|
|
blockIncl[peer]++
|
|
}
|
|
}
|
|
// Ensure peers with inclusions in this block have entries.
|
|
for peer := range blockIncl {
|
|
if t.peers[peer] == nil {
|
|
t.peers[peer] = &peerStats{}
|
|
}
|
|
}
|
|
// Update EMA for all tracked peers (decay inactive ones).
|
|
for peer, ps := range t.peers {
|
|
ps.recentIncluded = (1-emaAlpha)*ps.recentIncluded + emaAlpha*float64(blockIncl[peer])
|
|
}
|
|
// Check if the finalized block has advanced.
|
|
t.checkFinalization()
|
|
}
|
|
|
|
// checkFinalization credits peers for transactions in newly finalized blocks.
|
|
// Must be called with t.mu held.
|
|
func (t *Tracker) checkFinalization() {
|
|
finalHeader := t.chain.CurrentFinalBlock()
|
|
if finalHeader == nil {
|
|
return
|
|
}
|
|
finalNum := finalHeader.Number.Uint64()
|
|
if finalNum <= t.lastFinalNum {
|
|
return
|
|
}
|
|
// Credit peers for all blocks from lastFinalNum+1 to finalNum.
|
|
var credited int
|
|
for num := t.lastFinalNum + 1; num <= finalNum; num++ {
|
|
block := t.chain.GetBlockByNumber(num)
|
|
if block == nil {
|
|
continue
|
|
}
|
|
for _, tx := range block.Transactions() {
|
|
peer := t.txs[tx.Hash()]
|
|
if peer == "" {
|
|
continue
|
|
}
|
|
ps := t.peers[peer]
|
|
if ps == nil {
|
|
ps = &peerStats{}
|
|
t.peers[peer] = ps
|
|
}
|
|
ps.finalized++
|
|
credited++
|
|
}
|
|
}
|
|
if credited > 0 {
|
|
log.Trace("Credited peers for finalized inclusions",
|
|
"from", t.lastFinalNum+1, "to", finalNum, "txs", credited)
|
|
}
|
|
t.lastFinalNum = finalNum
|
|
}
|