From 632a47987cdce87d9879a37862210b7293b9f859 Mon Sep 17 00:00:00 2001 From: emailtovamos Date: Wed, 13 Nov 2024 14:04:31 +0000 Subject: [PATCH 1/5] core: handle edge case of tx size mismatch --- core/txpool/legacypool/legacypool.go | 25 ++---- core/txpool/legacypool/tx_overflowpool.go | 35 ++++++-- .../txpool/legacypool/tx_overflowpool_test.go | 82 +++++++++++++++++++ 3 files changed, 116 insertions(+), 26 deletions(-) diff --git a/core/txpool/legacypool/legacypool.go b/core/txpool/legacypool/legacypool.go index 0d5a1fb183..a1e75bc376 100644 --- a/core/txpool/legacypool/legacypool.go +++ b/core/txpool/legacypool/legacypool.go @@ -907,25 +907,14 @@ func (pool *LegacyPool) add(tx *types.Transaction, local bool) (replaced bool, e } func (pool *LegacyPool) addToOverflowPool(drop types.Transactions, isLocal bool) { - // calculate total number of slots in drop. Accordingly add them to OverflowPool (if there is space) - availableSlotsOverflowPool := pool.availableSlotsOverflowPool() - if availableSlotsOverflowPool > 0 { - // transfer availableSlotsOverflowPool number of transactions slots from drop to OverflowPool - currentSlotsUsed := 0 - for i, tx := range drop { - txSlots := numSlots(tx) - if currentSlotsUsed+txSlots <= availableSlotsOverflowPool { - from, _ := types.Sender(pool.signer, tx) - pool.localBufferPool.Add(tx) - log.Debug("adding to OverflowPool", "transaction", tx.Hash().String(), "from", from.String()) - currentSlotsUsed += txSlots - } else { - log.Debug("not all got added to OverflowPool", "totalAdded", i+1) - return - } + for _, tx := range drop { + added := pool.localBufferPool.Add(tx) + if added { + from, _ := types.Sender(pool.signer, tx) + log.Debug("Added to OverflowPool", "transaction", tx.Hash().String(), "from", from.String()) + } else { + log.Debug("Failed to add transaction to OverflowPool", "transaction", tx.Hash().String()) } - } else { - log.Debug("adding to OverflowPool unsuccessful", "availableSlotsOverflowPool", availableSlotsOverflowPool) } } diff --git a/core/txpool/legacypool/tx_overflowpool.go b/core/txpool/legacypool/tx_overflowpool.go index 4bfd4b6f5a..290cb2220c 100644 --- a/core/txpool/legacypool/tx_overflowpool.go +++ b/core/txpool/legacypool/tx_overflowpool.go @@ -8,6 +8,7 @@ import ( "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/core/types" + "github.com/ethereum/go-ethereum/log" ) // txHeapItem implements the Interface interface (https://pkg.go.dev/container/heap#Interface) of heap so that it can be heapified @@ -65,8 +66,8 @@ type TxOverflowPool struct { txHeap txHeap index map[common.Hash]*txHeapItem mu sync.RWMutex - maxSize uint64 - totalSize int + maxSize uint64 // Maximum slots + totalSize int // Total number of slots currently } func NewTxOverflowPoolHeap(estimatedMaxSize uint64) *TxOverflowPool { @@ -77,34 +78,52 @@ func NewTxOverflowPoolHeap(estimatedMaxSize uint64) *TxOverflowPool { } } -func (tp *TxOverflowPool) Add(tx *types.Transaction) { +func (tp *TxOverflowPool) Add(tx *types.Transaction) bool { tp.mu.Lock() defer tp.mu.Unlock() if _, exists := tp.index[tx.Hash()]; exists { // Transaction already in pool, ignore - return + return false + } + + txSlots := numSlots(tx) + + // If the transaction is too big to ever fit (and the pool isn't empty right now), reject it + if uint64(txSlots) >= tp.maxSize && tp.totalSize != 0 { + log.Warn("Transaction too large to fit in OverflowPool", "transaction", tx.Hash().String(), "requiredSlots", txSlots, "maxSlots", tp.maxSize) + return false } - if uint64(len(tp.txHeap)) >= tp.maxSize { - // Remove the oldest transaction to make space + // Remove transactions until there is room for the new transaction + for uint64(tp.totalSize+txSlots) > tp.maxSize { + if tp.txHeap.Len() == 0 { + // No transactions left to remove, cannot make room + log.Warn("Not enough space in OverflowPool even after clearing", "transaction", tx.Hash().String()) + return false + } + // Remove the oldest transaction oldestItem, ok := heap.Pop(&tp.txHeap).(*txHeapItem) if !ok || oldestItem == nil { - return + log.Error("Failed to pop from txHeap during Add") + return false } delete(tp.index, oldestItem.tx.Hash()) tp.totalSize -= numSlots(oldestItem.tx) OverflowPoolGauge.Dec(1) } + // Add the new transaction item := &txHeapItem{ tx: tx, timestamp: time.Now().UnixNano(), } heap.Push(&tp.txHeap, item) tp.index[tx.Hash()] = item - tp.totalSize += numSlots(tx) + tp.totalSize += txSlots OverflowPoolGauge.Inc(1) + + return true } func (tp *TxOverflowPool) Get(hash common.Hash) (*types.Transaction, bool) { diff --git a/core/txpool/legacypool/tx_overflowpool_test.go b/core/txpool/legacypool/tx_overflowpool_test.go index 9a4aee5008..e0976eca2f 100644 --- a/core/txpool/legacypool/tx_overflowpool_test.go +++ b/core/txpool/legacypool/tx_overflowpool_test.go @@ -1,6 +1,7 @@ package legacypool import ( + rand3 "crypto/rand" "math/big" rand2 "math/rand" "testing" @@ -9,6 +10,7 @@ import ( "github.com/cometbft/cometbft/libs/rand" "github.com/ethereum/go-ethereum/common" "github.com/ethereum/go-ethereum/core/types" + "github.com/stretchr/testify/assert" ) // Helper function to create a test transaction @@ -157,6 +159,64 @@ func TestTxOverflowPoolHeapLen(t *testing.T) { } } +func TestTxOverflowPoolSlotCalculation(t *testing.T) { + // Initialize the pool with a maximum size of 2 + pool := NewTxOverflowPoolHeap(2) + + // Create two transactions with different slot requirements + tx1 := createTestTx(1, big.NewInt(1000)) // tx1 takes 1 slot + tx2 := createTestTx(2, big.NewInt(2000)) // tx2 takes 1 slot + + // Add both transactions to fill the pool + pool.Add(tx1) + pool.Add(tx2) + + if pool.Len() != 2 { + t.Fatalf("Expected pool size 2, but got %d", pool.Len()) + } + + // Capture the initial total size + initialTotalSize := pool.totalSize + + dataSize := 40000 + tx3 := createLargeTestTx( + 3, // nonce + big.NewInt(100000000000), // gasPrice: 100 Gwei + dataSize, + ) // takes 3 slots + + // Create a third transaction with more slots than tx1 + tx3Added := pool.Add(tx3) + assert.Equal(t, false, tx3Added) + assert.Equal(t, 2, pool.totalSize) + + // Verify that the pool length remains at 2 after popping the oldest transaction + if pool.Len() != 2 { + t.Errorf("Expected pool size 2 after overflow, but got %d", pool.Len()) + } + + // Attempt to add a duplicate transaction (tx3) to see if it increases currentSlotsUsed erroneously + initialTotalSize = pool.totalSize + pool.Add(tx3) + + // The total size should not change since tx3 is already in the pool + if pool.totalSize != initialTotalSize { + t.Errorf("Duplicate transaction incorrectly updated totalSize; expected %d but got %d", initialTotalSize, pool.totalSize) + } +} + +func TestBiggerTx(t *testing.T) { + // Create a transaction with 40KB of data (which should take 2 slots) + dataSize := 40000 + tx := createLargeTestTx( + 0, // nonce + big.NewInt(100000000000), // gasPrice: 100 Gwei + dataSize, + ) + numberOfSlots := numSlots(tx) + assert.Equal(t, 2, numberOfSlots) +} + // Helper function to create a random test transaction func createRandomTestTx() *types.Transaction { nonce := uint64(rand.Intn(1000000)) @@ -176,6 +236,28 @@ func createRandomTestTxs(n int) []*types.Transaction { return txs } +// createLargeTestTx creates a transaction with a large data payload +func createLargeTestTx(nonce uint64, gasPrice *big.Int, dataSize int) *types.Transaction { + // Generate random data of specified size + data := make([]byte, dataSize) + rand3.Read(data) + + to := common.HexToAddress("0x1234567890123456789012345678901234567890") + + // Calculate gas needed for the data + // Gas costs: 21000 (base) + 16 (per non-zero byte) or 4 (per zero byte) + gasLimit := uint64(21000 + (16 * len(data))) + + return types.NewTransaction( + nonce, + to, + big.NewInt(1000), + gasLimit, + gasPrice, + data, + ) +} + // goos: darwin // goarch: arm64 // pkg: github.com/ethereum/go-ethereum/core/txpool/legacypool From 60454d3c3d2e678de623e1039dcfd3ce9e0ed1fa Mon Sep 17 00:00:00 2001 From: emailtovamos Date: Thu, 14 Nov 2024 07:16:30 +0000 Subject: [PATCH 2/5] core: lint and if condition improvement --- core/txpool/legacypool/legacypool.go | 9 --------- core/txpool/legacypool/tx_overflowpool.go | 2 +- core/txpool/legacypool/tx_overflowpool_test.go | 15 ++++----------- 3 files changed, 5 insertions(+), 21 deletions(-) diff --git a/core/txpool/legacypool/legacypool.go b/core/txpool/legacypool/legacypool.go index a1e75bc376..c4ede9c3bd 100644 --- a/core/txpool/legacypool/legacypool.go +++ b/core/txpool/legacypool/legacypool.go @@ -2097,15 +2097,6 @@ func (pool *LegacyPool) transferTransactions() { pool.Add(txs, true, false) } -func (pool *LegacyPool) availableSlotsOverflowPool() int { - maxOverflowPoolSize := int(pool.config.OverflowPoolSlots) - availableSlots := maxOverflowPoolSize - pool.localBufferPool.Size() - if availableSlots > 0 { - return availableSlots - } - return 0 -} - func (pool *LegacyPool) PrintTxStats() { for _, l := range pool.pending { for _, transaction := range l.txs.items { diff --git a/core/txpool/legacypool/tx_overflowpool.go b/core/txpool/legacypool/tx_overflowpool.go index 290cb2220c..69d1a7af19 100644 --- a/core/txpool/legacypool/tx_overflowpool.go +++ b/core/txpool/legacypool/tx_overflowpool.go @@ -90,7 +90,7 @@ func (tp *TxOverflowPool) Add(tx *types.Transaction) bool { txSlots := numSlots(tx) // If the transaction is too big to ever fit (and the pool isn't empty right now), reject it - if uint64(txSlots) >= tp.maxSize && tp.totalSize != 0 { + if (uint64(txSlots) > tp.maxSize) || (uint64(txSlots) == tp.maxSize && tp.totalSize != 0) { log.Warn("Transaction too large to fit in OverflowPool", "transaction", tx.Hash().String(), "requiredSlots", txSlots, "maxSlots", tp.maxSize) return false } diff --git a/core/txpool/legacypool/tx_overflowpool_test.go b/core/txpool/legacypool/tx_overflowpool_test.go index e0976eca2f..366aa6b8d1 100644 --- a/core/txpool/legacypool/tx_overflowpool_test.go +++ b/core/txpool/legacypool/tx_overflowpool_test.go @@ -175,15 +175,12 @@ func TestTxOverflowPoolSlotCalculation(t *testing.T) { t.Fatalf("Expected pool size 2, but got %d", pool.Len()) } - // Capture the initial total size - initialTotalSize := pool.totalSize - dataSize := 40000 tx3 := createLargeTestTx( 3, // nonce big.NewInt(100000000000), // gasPrice: 100 Gwei dataSize, - ) // takes 3 slots + ) // takes 2 slots // Create a third transaction with more slots than tx1 tx3Added := pool.Add(tx3) @@ -191,18 +188,14 @@ func TestTxOverflowPoolSlotCalculation(t *testing.T) { assert.Equal(t, 2, pool.totalSize) // Verify that the pool length remains at 2 after popping the oldest transaction - if pool.Len() != 2 { - t.Errorf("Expected pool size 2 after overflow, but got %d", pool.Len()) - } + assert.Equal(t, 2, pool.Len(), "Expected pool size 2 after overflow") // Attempt to add a duplicate transaction (tx3) to see if it increases currentSlotsUsed erroneously - initialTotalSize = pool.totalSize + initialTotalSize := pool.totalSize pool.Add(tx3) // The total size should not change since tx3 is already in the pool - if pool.totalSize != initialTotalSize { - t.Errorf("Duplicate transaction incorrectly updated totalSize; expected %d but got %d", initialTotalSize, pool.totalSize) - } + assert.Equal(t, initialTotalSize, pool.totalSize, "Duplicate transaction incorrectly updated totalSize") } func TestBiggerTx(t *testing.T) { From 6ea3562d81b4131c5e6fd8499e97ca1fc6e5a900 Mon Sep 17 00:00:00 2001 From: emailtovamos Date: Tue, 19 Nov 2024 11:35:50 +0000 Subject: [PATCH 3/5] pool: update the test for pool slots --- core/txpool/legacypool/tx_overflowpool_test.go | 15 +++++++++------ 1 file changed, 9 insertions(+), 6 deletions(-) diff --git a/core/txpool/legacypool/tx_overflowpool_test.go b/core/txpool/legacypool/tx_overflowpool_test.go index 366aa6b8d1..b44841b541 100644 --- a/core/txpool/legacypool/tx_overflowpool_test.go +++ b/core/txpool/legacypool/tx_overflowpool_test.go @@ -187,15 +187,18 @@ func TestTxOverflowPoolSlotCalculation(t *testing.T) { assert.Equal(t, false, tx3Added) assert.Equal(t, 2, pool.totalSize) - // Verify that the pool length remains at 2 after popping the oldest transaction + // Verify that the pool length remains at 2 assert.Equal(t, 2, pool.Len(), "Expected pool size 2 after overflow") - // Attempt to add a duplicate transaction (tx3) to see if it increases currentSlotsUsed erroneously - initialTotalSize := pool.totalSize - pool.Add(tx3) + tx4 := createTestTx(4, big.NewInt(3000)) // tx4 takes 1 slot + // Add tx4 to the pool + assert.True(t, pool.Add(tx4), "Failed to add tx4") + + // The pool should evict the oldest transaction (tx1) to make room for tx4 + // Verify that tx1 is no longer in the pool + _, exists := pool.Get(tx1.Hash()) + assert.False(t, exists, "Expected tx1 to be evicted from the pool") - // The total size should not change since tx3 is already in the pool - assert.Equal(t, initialTotalSize, pool.totalSize, "Duplicate transaction incorrectly updated totalSize") } func TestBiggerTx(t *testing.T) { From 86484a133b95d47a499586d26f92f0cd33f2418c Mon Sep 17 00:00:00 2001 From: emailtovamos Date: Tue, 19 Nov 2024 11:54:44 +0000 Subject: [PATCH 4/5] pool: lint --- core/txpool/legacypool/tx_overflowpool_test.go | 1 - 1 file changed, 1 deletion(-) diff --git a/core/txpool/legacypool/tx_overflowpool_test.go b/core/txpool/legacypool/tx_overflowpool_test.go index b44841b541..98801e02e8 100644 --- a/core/txpool/legacypool/tx_overflowpool_test.go +++ b/core/txpool/legacypool/tx_overflowpool_test.go @@ -198,7 +198,6 @@ func TestTxOverflowPoolSlotCalculation(t *testing.T) { // Verify that tx1 is no longer in the pool _, exists := pool.Get(tx1.Hash()) assert.False(t, exists, "Expected tx1 to be evicted from the pool") - } func TestBiggerTx(t *testing.T) { From b00a6c397f07ff85a43595ce7c7a2a042443c7a3 Mon Sep 17 00:00:00 2001 From: emailtovamos Date: Thu, 21 Nov 2024 11:39:06 +0000 Subject: [PATCH 5/5] pool: make maxSize & totalSize same type --- core/txpool/legacypool/legacypool.go | 2 +- core/txpool/legacypool/legacypool_test.go | 8 ++++---- core/txpool/legacypool/tx_overflowpool.go | 16 ++++++++-------- core/txpool/legacypool/tx_overflowpool_test.go | 2 +- 4 files changed, 14 insertions(+), 14 deletions(-) diff --git a/core/txpool/legacypool/legacypool.go b/core/txpool/legacypool/legacypool.go index c4ede9c3bd..9c9098c529 100644 --- a/core/txpool/legacypool/legacypool.go +++ b/core/txpool/legacypool/legacypool.go @@ -523,7 +523,7 @@ func (pool *LegacyPool) Stats() (int, int) { return pool.stats() } -func (pool *LegacyPool) statsOverflowPool() int { +func (pool *LegacyPool) statsOverflowPool() uint64 { pool.mu.RLock() defer pool.mu.RUnlock() diff --git a/core/txpool/legacypool/legacypool_test.go b/core/txpool/legacypool/legacypool_test.go index 53c62b9bd3..5bcc125f25 100644 --- a/core/txpool/legacypool/legacypool_test.go +++ b/core/txpool/legacypool/legacypool_test.go @@ -2268,17 +2268,17 @@ func TestTransferTransactions(t *testing.T) { assert.Equal(t, 0, pending, "pending transactions mismatched") assert.Equal(t, 0, queue, "queued transactions mismatched") - assert.Equal(t, 1, pool.statsOverflowPool(), "OverflowPool size unexpected") + assert.Equal(t, uint64(1), pool.statsOverflowPool(), "OverflowPool size unexpected") tx2 := dynamicFeeTx(0, 100000, big.NewInt(3), big.NewInt(2), keys[1]) pool.addToOverflowPool([]*types.Transaction{tx2}, true) - assert.Equal(t, 1, pool.statsOverflowPool(), "OverflowPool size unexpected") + assert.Equal(t, uint64(1), pool.statsOverflowPool(), "OverflowPool size unexpected") <-pool.requestPromoteExecutables(newAccountSet(pool.signer, from)) pending, queue = pool.Stats() assert.Equal(t, 0, pending, "pending transactions mismatched") assert.Equal(t, 1, queue, "queued transactions mismatched") - assert.Equal(t, 0, pool.statsOverflowPool(), "OverflowPool size unexpected") + assert.Equal(t, uint64(0), pool.statsOverflowPool(), "OverflowPool size unexpected") tx3 := dynamicFeeTx(0, 100000, big.NewInt(3), big.NewInt(2), keys[2]) pool.addToOverflowPool([]*types.Transaction{tx3}, true) @@ -2286,7 +2286,7 @@ func TestTransferTransactions(t *testing.T) { assert.Equal(t, 1, pending, "pending transactions mismatched") assert.Equal(t, 0, queue, "queued transactions mismatched") - assert.Equal(t, 1, pool.statsOverflowPool(), "OverflowPool size unexpected") + assert.Equal(t, uint64(1), pool.statsOverflowPool(), "OverflowPool size unexpected") } // Tests that the pool rejects replacement dynamic fee transactions that don't diff --git a/core/txpool/legacypool/tx_overflowpool.go b/core/txpool/legacypool/tx_overflowpool.go index 69d1a7af19..7d1c56bd1f 100644 --- a/core/txpool/legacypool/tx_overflowpool.go +++ b/core/txpool/legacypool/tx_overflowpool.go @@ -67,7 +67,7 @@ type TxOverflowPool struct { index map[common.Hash]*txHeapItem mu sync.RWMutex maxSize uint64 // Maximum slots - totalSize int // Total number of slots currently + totalSize uint64 // Total number of slots currently } func NewTxOverflowPoolHeap(estimatedMaxSize uint64) *TxOverflowPool { @@ -87,16 +87,16 @@ func (tp *TxOverflowPool) Add(tx *types.Transaction) bool { return false } - txSlots := numSlots(tx) + txSlots := uint64(numSlots(tx)) // If the transaction is too big to ever fit (and the pool isn't empty right now), reject it - if (uint64(txSlots) > tp.maxSize) || (uint64(txSlots) == tp.maxSize && tp.totalSize != 0) { + if (txSlots > tp.maxSize) || (txSlots == tp.maxSize && tp.totalSize != 0) { log.Warn("Transaction too large to fit in OverflowPool", "transaction", tx.Hash().String(), "requiredSlots", txSlots, "maxSlots", tp.maxSize) return false } // Remove transactions until there is room for the new transaction - for uint64(tp.totalSize+txSlots) > tp.maxSize { + for tp.totalSize+txSlots > tp.maxSize { if tp.txHeap.Len() == 0 { // No transactions left to remove, cannot make room log.Warn("Not enough space in OverflowPool even after clearing", "transaction", tx.Hash().String()) @@ -109,7 +109,7 @@ func (tp *TxOverflowPool) Add(tx *types.Transaction) bool { return false } delete(tp.index, oldestItem.tx.Hash()) - tp.totalSize -= numSlots(oldestItem.tx) + tp.totalSize -= uint64(numSlots(oldestItem.tx)) OverflowPoolGauge.Dec(1) } @@ -141,7 +141,7 @@ func (tp *TxOverflowPool) Remove(hash common.Hash) { if item, ok := tp.index[hash]; ok { heap.Remove(&tp.txHeap, item.index) delete(tp.index, hash) - tp.totalSize -= numSlots(item.tx) + tp.totalSize -= uint64(numSlots(item.tx)) OverflowPoolGauge.Dec(1) } } @@ -160,7 +160,7 @@ func (tp *TxOverflowPool) Flush(n int) []*types.Transaction { } txs[i] = item.tx delete(tp.index, item.tx.Hash()) - tp.totalSize -= numSlots(item.tx) + tp.totalSize -= uint64(numSlots(item.tx)) } OverflowPoolGauge.Dec(int64(n)) @@ -173,7 +173,7 @@ func (tp *TxOverflowPool) Len() int { return tp.txHeap.Len() } -func (tp *TxOverflowPool) Size() int { +func (tp *TxOverflowPool) Size() uint64 { tp.mu.RLock() defer tp.mu.RUnlock() return tp.totalSize diff --git a/core/txpool/legacypool/tx_overflowpool_test.go b/core/txpool/legacypool/tx_overflowpool_test.go index 98801e02e8..ca8b6635f4 100644 --- a/core/txpool/legacypool/tx_overflowpool_test.go +++ b/core/txpool/legacypool/tx_overflowpool_test.go @@ -185,7 +185,7 @@ func TestTxOverflowPoolSlotCalculation(t *testing.T) { // Create a third transaction with more slots than tx1 tx3Added := pool.Add(tx3) assert.Equal(t, false, tx3Added) - assert.Equal(t, 2, pool.totalSize) + assert.Equal(t, uint64(2), pool.totalSize) // Verify that the pool length remains at 2 assert.Equal(t, 2, pool.Len(), "Expected pool size 2 after overflow")