@aztec/p2p 0.0.1-commit.88c5703d4 → 0.0.1-commit.88e6f9396

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (89) hide show
  1. package/dest/client/p2p_client.d.ts +1 -1
  2. package/dest/client/p2p_client.d.ts.map +1 -1
  3. package/dest/client/p2p_client.js +6 -4
  4. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.js +6 -5
  5. package/dest/config.d.ts +7 -1
  6. package/dest/config.d.ts.map +1 -1
  7. package/dest/config.js +10 -0
  8. package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +3 -3
  9. package/dest/mem_pools/attestation_pool/attestation_pool.js +3 -3
  10. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.js +6 -6
  11. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts +4 -4
  12. package/dest/msg_validators/attestation_validator/attestation_validator.d.ts +1 -1
  13. package/dest/msg_validators/attestation_validator/attestation_validator.d.ts.map +1 -1
  14. package/dest/msg_validators/attestation_validator/attestation_validator.js +5 -4
  15. package/dest/msg_validators/clock_tolerance.d.ts +1 -1
  16. package/dest/msg_validators/clock_tolerance.d.ts.map +1 -1
  17. package/dest/msg_validators/clock_tolerance.js +4 -3
  18. package/dest/msg_validators/proposal_validator/proposal_validator.d.ts +1 -1
  19. package/dest/msg_validators/proposal_validator/proposal_validator.d.ts.map +1 -1
  20. package/dest/msg_validators/proposal_validator/proposal_validator.js +5 -5
  21. package/dest/services/encoding.d.ts +5 -1
  22. package/dest/services/encoding.d.ts.map +1 -1
  23. package/dest/services/encoding.js +7 -1
  24. package/dest/services/libp2p/libp2p_service.d.ts +2 -9
  25. package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
  26. package/dest/services/libp2p/libp2p_service.js +7 -24
  27. package/dest/services/peer-manager/peer_manager.d.ts +1 -1
  28. package/dest/services/peer-manager/peer_manager.d.ts.map +1 -1
  29. package/dest/services/peer-manager/peer_manager.js +4 -2
  30. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts +4 -7
  31. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts.map +1 -1
  32. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.js +40 -56
  33. package/dest/services/reqresp/batch-tx-requester/interface.d.ts +1 -2
  34. package/dest/services/reqresp/batch-tx-requester/interface.d.ts.map +1 -1
  35. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts +4 -4
  36. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts.map +1 -1
  37. package/dest/services/reqresp/batch-tx-requester/missing_txs.js +7 -7
  38. package/dest/services/reqresp/reqresp.d.ts +1 -1
  39. package/dest/services/reqresp/reqresp.d.ts.map +1 -1
  40. package/dest/services/reqresp/reqresp.js +16 -8
  41. package/dest/services/tx_collection/fast_tx_collection.d.ts +1 -4
  42. package/dest/services/tx_collection/fast_tx_collection.d.ts.map +1 -1
  43. package/dest/services/tx_collection/fast_tx_collection.js +57 -73
  44. package/dest/services/tx_collection/proposal_tx_collector.d.ts +6 -7
  45. package/dest/services/tx_collection/proposal_tx_collector.d.ts.map +1 -1
  46. package/dest/services/tx_collection/proposal_tx_collector.js +4 -4
  47. package/dest/services/tx_collection/request_tracker.d.ts +53 -0
  48. package/dest/services/tx_collection/request_tracker.d.ts.map +1 -0
  49. package/dest/services/tx_collection/request_tracker.js +84 -0
  50. package/dest/services/tx_collection/slow_tx_collection.js +1 -1
  51. package/dest/services/tx_collection/tx_collection.d.ts +3 -6
  52. package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
  53. package/dest/test-helpers/testbench-utils.d.ts +1 -1
  54. package/dest/test-helpers/testbench-utils.d.ts.map +1 -1
  55. package/dest/test-helpers/testbench-utils.js +20 -2
  56. package/dest/testbench/p2p_client_testbench_worker.d.ts +1 -1
  57. package/dest/testbench/p2p_client_testbench_worker.d.ts.map +1 -1
  58. package/dest/testbench/p2p_client_testbench_worker.js +6 -5
  59. package/package.json +14 -14
  60. package/src/client/p2p_client.ts +6 -4
  61. package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker.ts +6 -7
  62. package/src/config.ts +17 -0
  63. package/src/mem_pools/attestation_pool/attestation_pool.ts +3 -3
  64. package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +6 -6
  65. package/src/mem_pools/tx_pool_v2/interfaces.ts +4 -4
  66. package/src/msg_validators/attestation_validator/README.md +1 -1
  67. package/src/msg_validators/attestation_validator/attestation_validator.ts +5 -4
  68. package/src/msg_validators/clock_tolerance.ts +4 -3
  69. package/src/msg_validators/proposal_validator/README.md +3 -3
  70. package/src/msg_validators/proposal_validator/proposal_validator.ts +6 -5
  71. package/src/services/encoding.ts +9 -1
  72. package/src/services/libp2p/libp2p_service.ts +6 -27
  73. package/src/services/peer-manager/peer_manager.ts +5 -2
  74. package/src/services/reqresp/batch-tx-requester/README.md +46 -7
  75. package/src/services/reqresp/batch-tx-requester/batch_tx_requester.ts +32 -60
  76. package/src/services/reqresp/batch-tx-requester/interface.ts +0 -1
  77. package/src/services/reqresp/batch-tx-requester/missing_txs.ts +6 -6
  78. package/src/services/reqresp/reqresp.ts +18 -10
  79. package/src/services/tx_collection/fast_tx_collection.ts +57 -83
  80. package/src/services/tx_collection/proposal_tx_collector.ts +8 -13
  81. package/src/services/tx_collection/request_tracker.ts +127 -0
  82. package/src/services/tx_collection/slow_tx_collection.ts +1 -1
  83. package/src/services/tx_collection/tx_collection.ts +3 -5
  84. package/src/test-helpers/testbench-utils.ts +28 -3
  85. package/src/testbench/p2p_client_testbench_worker.ts +6 -8
  86. package/dest/services/tx_collection/missing_txs_tracker.d.ts +0 -32
  87. package/dest/services/tx_collection/missing_txs_tracker.d.ts.map +0 -1
  88. package/dest/services/tx_collection/missing_txs_tracker.js +0 -27
  89. package/src/services/tx_collection/missing_txs_tracker.ts +0 -52
@@ -18,7 +18,6 @@ import {
18
18
  type CheckpointProposalCore,
19
19
  type Gossipable,
20
20
  P2PMessage,
21
- type ValidationResult as P2PValidationResult,
22
21
  PeerErrorSeverity,
23
22
  PeerErrorSeverityByHarshness,
24
23
  TopicType,
@@ -226,7 +225,7 @@ export class LibP2PService extends WithTracer implements P2PService {
226
225
 
227
226
  const proposalValidatorOpts = {
228
227
  txsPermitted: !config.disableTransactions,
229
- maxTxsPerBlock: config.validateMaxTxsPerBlock,
228
+ maxTxsPerBlock: config.validateMaxTxsPerBlock ?? config.validateMaxTxsPerCheckpoint,
230
229
  };
231
230
  this.blockProposalValidator = new BlockProposalValidator(epochCache, proposalValidatorOpts);
232
231
  this.checkpointProposalValidator = new CheckpointProposalValidator(epochCache, proposalValidatorOpts);
@@ -976,6 +975,11 @@ export class LibP2PService extends WithTracer implements P2PService {
976
975
  } else if (wasIgnored) {
977
976
  return { result: TopicValidatorResult.Ignore, obj: tx };
978
977
  } else {
978
+ this.logger.warn(`Gossiped tx ${txHash.toString()} unexpectedly rejected by pool`, {
979
+ source: source.toString(),
980
+ txHash: txHash.toString(),
981
+ });
982
+ this.peerManager.penalizePeer(source, PeerErrorSeverity.HighToleranceError);
979
983
  return { result: TopicValidatorResult.Reject };
980
984
  }
981
985
  };
@@ -1742,31 +1746,6 @@ export class LibP2PService extends WithTracer implements P2PService {
1742
1746
  return PeerErrorSeverity.HighToleranceError;
1743
1747
  }
1744
1748
 
1745
- /**
1746
- * Validate a checkpoint attestation.
1747
- *
1748
- * @param attestation - The checkpoint attestation to validate.
1749
- * @returns True if the checkpoint attestation is valid, false otherwise.
1750
- */
1751
- @trackSpan('Libp2pService.validateCheckpointAttestation', async (_, attestation) => ({
1752
- [Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber,
1753
- [Attributes.BLOCK_ARCHIVE]: attestation.archive.toString(),
1754
- [Attributes.P2P_ID]: await attestation.p2pMessageLoggingIdentifier().then(i => i.toString()),
1755
- }))
1756
- public async validateCheckpointAttestation(
1757
- peerId: PeerId,
1758
- attestation: CheckpointAttestation,
1759
- ): Promise<P2PValidationResult> {
1760
- const result = await this.checkpointAttestationValidator.validate(attestation);
1761
-
1762
- if (result.result === 'reject') {
1763
- this.logger.warn(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
1764
- this.peerManager.penalizePeer(peerId, result.severity);
1765
- }
1766
-
1767
- return result;
1768
- }
1769
-
1770
1749
  public getPeerScore(peerId: PeerId): number {
1771
1750
  return this.node.services.pubsub.score.score(peerId.toString());
1772
1751
  }
@@ -32,7 +32,7 @@ import { PeerScoreState, type PeerScoring } from './peer_scoring.js';
32
32
  const MAX_DIAL_ATTEMPTS = 3;
33
33
  const MAX_CACHED_PEERS = 100;
34
34
  const MAX_CACHED_PEER_AGE_MS = 5 * 60 * 1000; // 5 minutes
35
- const FAILED_PEER_BAN_TIME_MS = 5 * 60 * 1000; // 5 minutes timeout after failing MAX_DIAL_ATTEMPTS
35
+ const DEFAULT_FAILED_PEER_BAN_TIME_MS = 5 * 60 * 1000; // 5 minutes timeout after failing MAX_DIAL_ATTEMPTS
36
36
  const GOODBYE_DIAL_TIMEOUT_MS = 1000;
37
37
  const FAILED_AUTH_HANDSHAKE_EXPIRY_MS = 60 * 60 * 1000; // 1 hour
38
38
 
@@ -776,7 +776,8 @@ export class PeerManager implements PeerManagerInterface {
776
776
  // Add to timed out peers
777
777
  this.timedOutPeers.set(id, {
778
778
  peerId: id,
779
- timeoutUntilMs: this.dateProvider.now() + FAILED_PEER_BAN_TIME_MS,
779
+ timeoutUntilMs:
780
+ this.dateProvider.now() + (this.config.peerFailedBanTimeMs ?? DEFAULT_FAILED_PEER_BAN_TIME_MS),
780
781
  });
781
782
  }
782
783
  }
@@ -938,6 +939,8 @@ export class PeerManager implements PeerManagerInterface {
938
939
  `Received auth for validator ${sender.toString()} from peer ${peerIdString}, but this validator is already authenticated to peer ${peerForAddress.toString()}`,
939
940
  { ...logData, address: sender.toString() },
940
941
  );
942
+ this.markAuthHandshakeFailed(peerId);
943
+ this.markPeerForDisconnect(peerId);
941
944
  return;
942
945
  }
943
946
 
@@ -170,6 +170,37 @@ class BlockTxsResponse {
170
170
 
171
171
  The `BitVector` is a compact representation where each bit corresponds to a transaction index in the block proposal. This allows efficient capability advertisement without repeating full hashes.
172
172
 
173
+ ## Cancellation
174
+
175
+ All cancellation is managed by a single `RequestTracker` instance, shared across the entire collection
176
+ flow. The `RequestTracker` owns the deadline, tracks which txs are still missing, and exposes a
177
+ `cancellationToken` promise that resolves when the request should stop (deadline hit, all txs fetched,
178
+ or external `cancel()` call).
179
+
180
+ Cancellation propagates from the deepest stack level upward:
181
+
182
+ ```
183
+ RequestTracker.finish()
184
+ ├── resolves cancellationToken promise
185
+
186
+ ├── BatchTxRequester workers (deepest)
187
+ │ ├── shouldStop() checks requestTracker.cancelled → exit loop
188
+ │ ├── sleepClampedToDeadline races sleep vs cancellationToken → wakes
189
+ │ └── semaphore.acquire races vs cancellationToken → wakes
190
+ │ │
191
+ │ ▼ workers settle → txQueue.end() → generator returns
192
+
193
+ ├── Node collection loops
194
+ │ ├── notFinished() checks requestTracker.cancelled → exit loop
195
+ │ └── inter-retry sleep races vs cancellationToken → wakes
196
+ │ │
197
+ │ ▼ all node loops settle
198
+
199
+ └── collectFast (outermost)
200
+ awaits Promise.allSettled([reqresp, nodes]) → settles after inner tasks
201
+ finally: requestTracker.cancel() (idempotent), cleanup
202
+ ```
203
+
173
204
  ## Key Files
174
205
 
175
206
  | File | Description |
@@ -179,15 +210,16 @@ The `BitVector` is a compact representation where each bit corresponds to a tran
179
210
  | `peer_collection.ts` | Manages peer classification (dumb/smart/bad) and rate limiting |
180
211
  | `interface.ts` | Type definitions for dependencies |
181
212
  | `../protocols/block_txs/` | Wire protocol definitions (`BlockTxsRequest`, `BlockTxsResponse`, `BitVector`) |
213
+ | `../../tx_collection/request_tracker.ts` | Centralized deadline, missing tx tracking, and cancellation signal |
182
214
 
183
215
  ## Stopping Conditions
184
216
 
185
- The `BatchTxRequester` stops when any of these conditions are met:
217
+ The `BatchTxRequester` stops when any of these conditions are met, all managed by the `RequestTracker`:
186
218
 
187
- 1. **All transactions fetched** - Success!
188
- 2. **Deadline exceeded** - Timeout configured by caller
189
- 3. **Abort signal** - External cancellation
190
- 4. **No transactions to fetch** - Nothing was missing
219
+ 1. **All transactions fetched** - `markFetched()` removes the last missing tx, triggering `finish()`
220
+ 2. **Deadline exceeded** - `setTimeout` in `RequestTracker` fires, triggering `finish()`
221
+ 3. **External cancellation** - `RequestTracker.cancel()` called (e.g., from `stop()`, `stopCollectingForBlocksUpTo`)
222
+ 4. **No transactions to fetch** - Empty hash set at construction, `RequestTracker` finishes immediately
191
223
 
192
224
  ## Configuration
193
225
 
@@ -228,11 +260,15 @@ Request to peer fails
228
260
  ## Usage Example
229
261
 
230
262
  ```typescript
263
+ const requestTracker = RequestTracker.create(
264
+ missingTxHashes, // TxHash[] - what we need
265
+ new Date(Date.now() + 5_000), // deadline
266
+ );
267
+
231
268
  const requester = new BatchTxRequester(
232
- missingTxHashes, // TxHash[] - what we need
269
+ requestTracker, // IRequestTracker - tracks missing txs, deadline, and cancellation
233
270
  blockTxsSource, // BlockTxsSource - the proposal or block we need txs for
234
271
  pinnedPeer, // PeerId | undefined - peer expected to have the txs
235
- timeoutMs, // number - how long to try
236
272
  p2pService, // BatchTxRequesterLibP2PService
237
273
  );
238
274
 
@@ -273,6 +309,8 @@ const txs = await BatchTxRequester.collectAllTxs(requester.run());
273
309
  │ 1. Try RPC nodes first (fast) │ │ Periodic polling of RPC nodes │
274
310
  │ 2. Fall back to BatchTxRequester │ │ and peers for missing txs │
275
311
  │ │ │ │
312
+ │ Creates RequestTracker per │ │ │
313
+ │ request with deadline │ │ │
276
314
  └───────────────────┬───────────────┘ └─────────────────────────────────────┘
277
315
 
278
316
  │ For 'proposal' and 'block' requests
@@ -281,6 +319,7 @@ const txs = await BatchTxRequester.collectAllTxs(requester.run());
281
319
  │ BatchTxRequester │
282
320
  │ │
283
321
  │ Aggressive parallel fetching from multiple peers │
322
+ │ Shares RequestTracker with FastTxCollection for unified cancellation │
284
323
  │ Uses BLOCK_TXS sub-protocol for efficient batching │
285
324
  └───────────────────┬─────────────────────────────────────────────────────────┘
286
325
 
@@ -1,15 +1,14 @@
1
1
  import { chunkWrapAround } from '@aztec/foundation/collection';
2
- import { TimeoutError } from '@aztec/foundation/error';
3
2
  import { type Logger, createLogger } from '@aztec/foundation/log';
4
3
  import { FifoMemoryQueue, type ISemaphore, Semaphore } from '@aztec/foundation/queue';
5
4
  import { sleep } from '@aztec/foundation/sleep';
6
- import { DateProvider, executeTimeout } from '@aztec/foundation/timer';
5
+ import { DateProvider } from '@aztec/foundation/timer';
7
6
  import { PeerErrorSeverity } from '@aztec/stdlib/p2p';
8
7
  import { Tx, TxArray, TxHash } from '@aztec/stdlib/tx';
9
8
 
10
9
  import type { PeerId } from '@libp2p/interface';
11
10
 
12
- import type { IMissingTxsTracker } from '../../tx_collection/missing_txs_tracker.js';
11
+ import type { IRequestTracker } from '../../tx_collection/request_tracker.js';
13
12
  import { ReqRespSubProtocol } from '.././interface.js';
14
13
  import { BlockTxsRequest, BlockTxsResponse, type BlockTxsSource } from '.././protocols/index.js';
15
14
  import { ReqRespStatus } from '.././status.js';
@@ -42,16 +41,14 @@ import { BatchRequestTxValidator, type IBatchRequestTxValidator } from './tx_val
42
41
  * - Is the peer which was unable to send us successful response N times in a row
43
42
  * */
44
43
  export class BatchTxRequester {
44
+ private readonly requestTracker: IRequestTracker;
45
45
  private readonly blockTxsSource: BlockTxsSource;
46
46
  private readonly pinnedPeer: PeerId | undefined;
47
- private readonly timeoutMs: number;
48
47
  private readonly p2pService: BatchTxRequesterLibP2PService;
49
48
  private readonly logger: Logger;
50
- private readonly dateProvider: DateProvider;
51
49
  private readonly opts: BatchTxRequesterOptions;
52
50
  private readonly peers: IPeerCollection;
53
51
  private readonly txsMetadata: ITxMetadataCollection;
54
- private readonly deadline: number;
55
52
  private readonly smartRequesterSemaphore: ISemaphore;
56
53
  private readonly txQueue: FifoMemoryQueue<Tx>;
57
54
  private readonly txValidator: IBatchRequestTxValidator;
@@ -60,21 +57,19 @@ export class BatchTxRequester {
60
57
  private readonly txBatchSize: number;
61
58
 
62
59
  constructor(
63
- missingTxsTracker: IMissingTxsTracker,
60
+ requestTracker: IRequestTracker,
64
61
  blockTxsSource: BlockTxsSource,
65
62
  pinnedPeer: PeerId | undefined,
66
- timeoutMs: number,
67
63
  p2pService: BatchTxRequesterLibP2PService,
68
64
  logger?: Logger,
69
65
  dateProvider?: DateProvider,
70
66
  opts?: BatchTxRequesterOptions,
71
67
  ) {
68
+ this.requestTracker = requestTracker;
72
69
  this.blockTxsSource = blockTxsSource;
73
70
  this.pinnedPeer = pinnedPeer;
74
- this.timeoutMs = timeoutMs;
75
71
  this.p2pService = p2pService;
76
72
  this.logger = logger ?? createLogger('p2p:reqresp_batch');
77
- this.dateProvider = dateProvider ?? new DateProvider();
78
73
  this.opts = opts ?? {};
79
74
 
80
75
  this.smartParallelWorkerCount =
@@ -82,7 +77,6 @@ export class BatchTxRequester {
82
77
  this.dumbParallelWorkerCount =
83
78
  this.opts.dumbParallelWorkerCount ?? DEFAULT_BATCH_TX_REQUESTER_DUMB_PARALLEL_WORKER_COUNT;
84
79
  this.txBatchSize = this.opts.txBatchSize ?? DEFAULT_BATCH_TX_REQUESTER_TX_BATCH_SIZE;
85
- this.deadline = this.dateProvider.now() + this.timeoutMs;
86
80
  this.txQueue = new FifoMemoryQueue(this.logger);
87
81
  this.txValidator = this.opts.txValidator ?? new BatchRequestTxValidator(this.p2pService.txValidatorConfig);
88
82
 
@@ -93,12 +87,12 @@ export class BatchTxRequester {
93
87
  this.peers = new PeerCollection(
94
88
  this.p2pService.connectionSampler,
95
89
  this.pinnedPeer,
96
- this.dateProvider,
90
+ dateProvider ?? new DateProvider(),
97
91
  badPeerThreshold,
98
92
  this.p2pService.peerScoring,
99
93
  );
100
94
  }
101
- this.txsMetadata = new MissingTxMetadataCollection(missingTxsTracker, this.txBatchSize);
95
+ this.txsMetadata = new MissingTxMetadataCollection(requestTracker, this.txBatchSize);
102
96
  this.smartRequesterSemaphore = this.opts.semaphore ?? new Semaphore(0);
103
97
  }
104
98
 
@@ -106,40 +100,30 @@ export class BatchTxRequester {
106
100
  * Fetches all missing transactions and yields them one by one
107
101
  * */
108
102
  public async *run(): AsyncGenerator<Tx, Tx | undefined, unknown> {
109
- // Our timeout is represented in milliseconds but queue expects seconds
110
- // We also want to make sure we wait at least 1 second in case of very low timeouts
111
- const timeoutQueueAfter = Math.max(Math.ceil(this.timeoutMs / 1_000), 1);
112
103
  try {
113
104
  if (this.txsMetadata.getMissingTxHashes().size === 0) {
114
105
  return undefined;
115
106
  }
116
107
 
117
- // Start workers in background
118
- const workersPromise = executeTimeout(
119
- () => Promise.allSettled([this.smartRequester(), this.dumbRequester(), this.pinnedPeerRequester()]),
120
- this.timeoutMs,
121
- ).finally(() => {
108
+ // Start workers in background. Workers stop themselves via requestTracker.checkCancelled().
109
+ const workersPromise = Promise.allSettled([
110
+ this.smartRequester(),
111
+ this.dumbRequester(),
112
+ this.pinnedPeerRequester(),
113
+ ]).finally(() => {
122
114
  this.txQueue.end();
123
115
  });
124
116
 
117
+ // Yield txs as workers put them on the queue. The queue's end() drains remaining items
118
+ // before returning null, so we don't lose any txs.
125
119
  while (true) {
126
- const tx = await this.txQueue.get(timeoutQueueAfter);
120
+ const tx = await this.txQueue.get();
127
121
 
128
- // null indicates that the queue has ended
129
122
  if (tx === null) {
130
123
  break;
131
124
  }
132
125
 
133
126
  yield tx;
134
-
135
- if (this.shouldStop()) {
136
- // Drain queue before ending
137
- let remaining;
138
- while ((remaining = this.txQueue.getImmediate()) !== undefined) {
139
- yield remaining;
140
- }
141
- break;
142
- }
143
127
  }
144
128
 
145
129
  this.unlockSmartRequesterSemaphores();
@@ -360,7 +344,10 @@ export class BatchTxRequester {
360
344
  ) {
361
345
  try {
362
346
  this.logger.trace(`Smart worker ${workerIndex} started`);
363
- await executeTimeout((_: AbortSignal) => this.smartRequesterSemaphore.acquire(), this.timeoutMs);
347
+ await Promise.race([this.smartRequesterSemaphore.acquire(), this.requestTracker.cancellationToken]);
348
+ if (this.requestTracker.checkCancelled()) {
349
+ return;
350
+ }
364
351
  this.logger.trace(`Smart worker ${workerIndex} acquired semaphore`);
365
352
 
366
353
  while (!this.shouldStop()) {
@@ -384,7 +371,10 @@ export class BatchTxRequester {
384
371
  //
385
372
  // When a dumb peer responds with valid txIndices, it gets
386
373
  // promoted to smart and releases the semaphore, waking this worker.
387
- await executeTimeout((_: AbortSignal) => this.smartRequesterSemaphore.acquire(), this.timeoutMs);
374
+ await Promise.race([this.smartRequesterSemaphore.acquire(), this.requestTracker.cancellationToken]);
375
+ if (this.requestTracker.checkCancelled()) {
376
+ break;
377
+ }
388
378
  this.logger.debug(`Worker loop smart: acquired next smart peer`);
389
379
  continue;
390
380
  }
@@ -411,11 +401,7 @@ export class BatchTxRequester {
411
401
  });
412
402
  }
413
403
  } catch (err: any) {
414
- if (err instanceof TimeoutError) {
415
- this.logger.debug(`Smart worker ${workerIndex} timed out waiting for semaphore`);
416
- } else {
417
- this.logger.error(`Smart worker ${workerIndex} encountered an error: ${err}`);
418
- }
404
+ this.logger.error(`Smart worker ${workerIndex} encountered an error: ${err}`);
419
405
  } finally {
420
406
  this.logger.debug(`Smart worker ${workerIndex} finished`);
421
407
  }
@@ -651,27 +637,14 @@ export class BatchTxRequester {
651
637
  }
652
638
 
653
639
  /*
654
- * @returns true if all missing txs have been fetched */
655
- private fetchedAllTxs() {
656
- return this.txsMetadata.getMissingTxHashes().size == 0;
657
- }
658
-
659
- /*
660
- * Checks if the BatchTxRequester should stop fetching missing txs
661
- * Conditions for stopping are:
662
- * - There have been no missing transactions to start with
663
- * - All transactions have been fetched
664
- * - The deadline has been hit (no more time to fetch)
665
- * - This process has been cancelled via abortSignal
666
- *
667
- * @returns true if BatchTxRequester should stop, otherwise false*/
640
+ * Checks if the BatchTxRequester should stop fetching missing txs.
641
+ * Delegates to requestTracker which covers: deadline hit, all txs fetched, or external cancellation. */
668
642
  private shouldStop() {
669
- const aborted = this.opts.abortSignal?.aborted ?? false;
670
- if (aborted) {
643
+ if (this.requestTracker.checkCancelled()) {
671
644
  this.unlockSmartRequesterSemaphores();
672
645
  }
673
646
 
674
- return aborted || this.fetchedAllTxs() || this.dateProvider.now() > this.deadline;
647
+ return this.requestTracker.checkCancelled();
675
648
  }
676
649
 
677
650
  /*
@@ -689,10 +662,9 @@ export class BatchTxRequester {
689
662
  * This ensures we don't sleep past the deadline.
690
663
  * */
691
664
  private async sleepClampedToDeadline(durationMs: number) {
692
- const remaining = this.deadline - this.dateProvider.now();
693
- const thereIsTimeRemaining = remaining > 0;
694
- if (thereIsTimeRemaining) {
695
- await sleep(Math.min(durationMs, remaining));
665
+ if (this.requestTracker.checkCancelled()) {
666
+ return;
696
667
  }
668
+ await Promise.race([sleep(durationMs), this.requestTracker.cancellationToken]);
697
669
  }
698
670
  }
@@ -49,7 +49,6 @@ export interface BatchTxRequesterOptions {
49
49
  //Injectable for testing purposes
50
50
  semaphore?: ISemaphore;
51
51
  peerCollection?: IPeerCollection;
52
- abortSignal?: AbortSignal;
53
52
  /** Optional tx validator for testing - if not provided, one is created from p2pService.txValidatorConfig */
54
53
  txValidator?: IBatchRequestTxValidator;
55
54
  }
@@ -2,7 +2,7 @@ import { type Tx, TxHash } from '@aztec/stdlib/tx';
2
2
 
3
3
  import type { PeerId } from '@libp2p/interface';
4
4
 
5
- import type { IMissingTxsTracker } from '../../tx_collection/missing_txs_tracker.js';
5
+ import type { IRequestTracker } from '../../tx_collection/request_tracker.js';
6
6
  import { DEFAULT_BATCH_TX_REQUESTER_TX_BATCH_SIZE } from './config.js';
7
7
  import type { ITxMetadataCollection } from './interface.js';
8
8
 
@@ -41,10 +41,10 @@ export class MissingTxMetadataCollection implements ITxMetadataCollection {
41
41
  private txMetadata = new Map<string, MissingTxMetadata>();
42
42
 
43
43
  constructor(
44
- private missingTxsTracker: IMissingTxsTracker,
44
+ private requestTracker: IRequestTracker,
45
45
  private readonly txBatchSize: number = DEFAULT_BATCH_TX_REQUESTER_TX_BATCH_SIZE,
46
46
  ) {
47
- missingTxsTracker.missingTxHashes.forEach(hash => this.txMetadata.set(hash, new MissingTxMetadata(hash)));
47
+ requestTracker.missingTxHashes.forEach(hash => this.txMetadata.set(hash, new MissingTxMetadata(hash)));
48
48
  }
49
49
 
50
50
  public getPrioritizingNotInFlightAndLowerRequestCount(txs: string[]): MissingTxMetadata[] {
@@ -65,7 +65,7 @@ export class MissingTxMetadataCollection implements ITxMetadataCollection {
65
65
  }
66
66
 
67
67
  public getMissingTxHashes(): Set<string> {
68
- return this.missingTxsTracker.missingTxHashes;
68
+ return this.requestTracker.missingTxHashes;
69
69
  }
70
70
 
71
71
  public getTxsPeerHas(peer: PeerId): Set<string> {
@@ -128,7 +128,7 @@ export class MissingTxMetadataCollection implements ITxMetadataCollection {
128
128
  }
129
129
 
130
130
  public alreadyFetched(txHash: TxHash): boolean {
131
- return !this.missingTxsTracker.isMissing(txHash.toString());
131
+ return !this.requestTracker.isMissing(txHash.toString());
132
132
  }
133
133
 
134
134
  public markFetched(peerId: PeerId, tx: Tx): boolean {
@@ -144,7 +144,7 @@ export class MissingTxMetadataCollection implements ITxMetadataCollection {
144
144
  }
145
145
 
146
146
  txMeta.peers.add(peerId.toString());
147
- return this.missingTxsTracker.markFetched(tx);
147
+ return this.requestTracker.markFetched(tx);
148
148
  }
149
149
 
150
150
  public markPeerHas(peerId: PeerId, txHash: TxHash[]) {
@@ -16,7 +16,7 @@ import {
16
16
  IndividualReqRespTimeoutError,
17
17
  InvalidResponseError,
18
18
  } from '../../errors/reqresp.error.js';
19
- import { SnappyTransform } from '../encoding.js';
19
+ import { OversizedSnappyResponseError, SnappyTransform } from '../encoding.js';
20
20
  import type { PeerScoring } from '../peer-manager/peer_scoring.js';
21
21
  import {
22
22
  DEFAULT_INDIVIDUAL_REQUEST_TIMEOUT_MS,
@@ -553,16 +553,10 @@ export class ReqResp implements ReqRespInterface {
553
553
  data: message,
554
554
  };
555
555
  } catch (e: any) {
556
+ // All errors (invalid status bytes, oversized snappy responses, corrupt data, etc.)
557
+ // are re-thrown so the caller can penalize the peer via handleResponseError.
556
558
  this.logger.debug(`Reading message failed: ${e.message}`);
557
-
558
- let status = ReqRespStatus.UNKNOWN;
559
- if (e instanceof ReqRespStatusError) {
560
- status = e.status;
561
- }
562
-
563
- return {
564
- status,
565
- };
559
+ throw e;
566
560
  }
567
561
  }
568
562
 
@@ -780,6 +774,20 @@ export class ReqResp implements ReqRespInterface {
780
774
  return undefined;
781
775
  }
782
776
 
777
+ // Invalid status byte: the peer sent a status byte that doesn't match any known status code.
778
+ // This is a protocol violation, penalize harshly.
779
+ if (e instanceof ReqRespStatusError) {
780
+ this.logger.warn(`Invalid status byte from peer ${peerId.toString()} in ${subProtocol}: ${e.message}`, logTags);
781
+ return PeerErrorSeverity.LowToleranceError;
782
+ }
783
+
784
+ // Oversized snappy response: the peer is sending data that exceeds the allowed size.
785
+ // This is a protocol violation that wastes bandwidth, so penalize harshly.
786
+ if (e instanceof OversizedSnappyResponseError) {
787
+ this.logger.warn(`Oversized response from peer ${peerId.toString()} in ${subProtocol}: ${e.message}`, logTags);
788
+ return PeerErrorSeverity.LowToleranceError;
789
+ }
790
+
783
791
  return this.categorizeConnectionErrors(e, peerId, subProtocol);
784
792
  }
785
793