mirror of https://github.com/ethereum/go-ethereum
core/txpool/legacypool: ensure pending nonces are reset by SubPool.Clear (#31020)
closes https://github.com/ethereum/go-ethereum/issues/30842pull/31037/head
parent
9e4f08c25d
commit
47d17acdc9
@ -0,0 +1,102 @@ |
||||
package simulated |
||||
|
||||
import ( |
||||
"context" |
||||
"crypto/ecdsa" |
||||
"math/big" |
||||
"testing" |
||||
"time" |
||||
|
||||
"github.com/ethereum/go-ethereum/core/types" |
||||
) |
||||
|
||||
// TestTransactionRollbackBehavior tests that calling Rollback on the simulated backend doesn't prevent subsequent
|
||||
// addition of new transactions
|
||||
func TestTransactionRollbackBehavior(t *testing.T) { |
||||
sim := NewBackend( |
||||
types.GenesisAlloc{ |
||||
testAddr: {Balance: big.NewInt(10000000000000000)}, |
||||
testAddr2: {Balance: big.NewInt(10000000000000000)}, |
||||
}, |
||||
) |
||||
defer sim.Close() |
||||
client := sim.Client() |
||||
|
||||
btx0 := testSendSignedTx(t, testKey, sim, true) |
||||
tx0 := testSendSignedTx(t, testKey2, sim, false) |
||||
tx1 := testSendSignedTx(t, testKey2, sim, false) |
||||
|
||||
sim.Rollback() |
||||
|
||||
if pendingStateHasTx(client, btx0) || pendingStateHasTx(client, tx0) || pendingStateHasTx(client, tx1) { |
||||
t.Fatalf("all transactions were not rolled back") |
||||
} |
||||
|
||||
btx2 := testSendSignedTx(t, testKey, sim, true) |
||||
tx2 := testSendSignedTx(t, testKey2, sim, false) |
||||
tx3 := testSendSignedTx(t, testKey2, sim, false) |
||||
|
||||
sim.Commit() |
||||
|
||||
if !pendingStateHasTx(client, btx2) || !pendingStateHasTx(client, tx2) || !pendingStateHasTx(client, tx3) { |
||||
t.Fatalf("all post-rollback transactions were not included") |
||||
} |
||||
} |
||||
|
||||
// testSendSignedTx sends a signed transaction to the simulated backend.
|
||||
// It does not commit the block.
|
||||
func testSendSignedTx(t *testing.T, key *ecdsa.PrivateKey, sim *Backend, isBlobTx bool) *types.Transaction { |
||||
t.Helper() |
||||
client := sim.Client() |
||||
ctx := context.Background() |
||||
|
||||
var ( |
||||
err error |
||||
signedTx *types.Transaction |
||||
) |
||||
if isBlobTx { |
||||
signedTx, err = newBlobTx(sim, key) |
||||
} else { |
||||
signedTx, err = newTx(sim, key) |
||||
} |
||||
if err != nil { |
||||
t.Fatalf("failed to create transaction: %v", err) |
||||
} |
||||
|
||||
if err = client.SendTransaction(ctx, signedTx); err != nil { |
||||
t.Fatalf("failed to send transaction: %v", err) |
||||
} |
||||
|
||||
return signedTx |
||||
} |
||||
|
||||
// pendingStateHasTx returns true if a given transaction was successfully included as of the latest pending state.
|
||||
func pendingStateHasTx(client Client, tx *types.Transaction) bool { |
||||
ctx := context.Background() |
||||
|
||||
var ( |
||||
receipt *types.Receipt |
||||
err error |
||||
) |
||||
|
||||
// Poll for receipt with timeout
|
||||
deadline := time.Now().Add(2 * time.Second) |
||||
for time.Now().Before(deadline) { |
||||
receipt, err = client.TransactionReceipt(ctx, tx.Hash()) |
||||
if err == nil && receipt != nil { |
||||
break |
||||
} |
||||
time.Sleep(100 * time.Millisecond) |
||||
} |
||||
|
||||
if err != nil { |
||||
return false |
||||
} |
||||
if receipt == nil { |
||||
return false |
||||
} |
||||
if receipt.Status != types.ReceiptStatusSuccessful { |
||||
return false |
||||
} |
||||
return true |
||||
} |
Loading…
Reference in new issue