@aztec/p2p 0.0.1-commit.8f9871590 → 0.0.1-commit.9117c5f5a

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 (218) hide show
  1. package/dest/client/factory.d.ts +6 -6
  2. package/dest/client/factory.d.ts.map +1 -1
  3. package/dest/client/factory.js +18 -28
  4. package/dest/client/interface.d.ts +10 -19
  5. package/dest/client/interface.d.ts.map +1 -1
  6. package/dest/client/p2p_client.d.ts +7 -18
  7. package/dest/client/p2p_client.d.ts.map +1 -1
  8. package/dest/client/p2p_client.js +41 -72
  9. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.js +6 -7
  10. package/dest/config.d.ts +4 -6
  11. package/dest/config.d.ts.map +1 -1
  12. package/dest/config.js +0 -5
  13. package/dest/errors/tx-pool.error.d.ts +8 -0
  14. package/dest/errors/tx-pool.error.d.ts.map +1 -0
  15. package/dest/errors/tx-pool.error.js +9 -0
  16. package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts +3 -1
  17. package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts.map +1 -1
  18. package/dest/mem_pools/tx_pool_v2/deleted_pool.js +9 -0
  19. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts +3 -3
  20. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts.map +1 -1
  21. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.js +18 -9
  22. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.js +3 -3
  23. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts +3 -3
  24. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts.map +1 -1
  25. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.js +12 -4
  26. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts +2 -2
  27. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts.map +1 -1
  28. package/dest/mem_pools/tx_pool_v2/eviction/index.js +1 -1
  29. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts +48 -5
  30. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts.map +1 -1
  31. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.js +8 -0
  32. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.js +2 -2
  33. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.js +5 -5
  34. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts +2 -2
  35. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts.map +1 -1
  36. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.js +12 -6
  37. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts +4 -4
  38. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts.map +1 -1
  39. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.js +14 -4
  40. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts +3 -3
  41. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts.map +1 -1
  42. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.js +2 -2
  43. package/dest/mem_pools/tx_pool_v2/index.d.ts +2 -2
  44. package/dest/mem_pools/tx_pool_v2/index.d.ts.map +1 -1
  45. package/dest/mem_pools/tx_pool_v2/index.js +1 -1
  46. package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts +15 -0
  47. package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts.map +1 -0
  48. package/dest/mem_pools/tx_pool_v2/instrumentation.js +43 -0
  49. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts +16 -6
  50. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts.map +1 -1
  51. package/dest/mem_pools/tx_pool_v2/interfaces.js +3 -1
  52. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts +29 -5
  53. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts.map +1 -1
  54. package/dest/mem_pools/tx_pool_v2/tx_metadata.js +71 -6
  55. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts +5 -2
  56. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts.map +1 -1
  57. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.js +21 -12
  58. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts +6 -3
  59. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts.map +1 -1
  60. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.js +6 -5
  61. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts +11 -5
  62. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts.map +1 -1
  63. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.js +274 -149
  64. package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts +2 -2
  65. package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts.map +1 -1
  66. package/dest/msg_validators/tx_validator/aggregate_tx_validator.js +3 -3
  67. package/dest/msg_validators/tx_validator/factory.d.ts +114 -6
  68. package/dest/msg_validators/tx_validator/factory.d.ts.map +1 -1
  69. package/dest/msg_validators/tx_validator/factory.js +219 -58
  70. package/dest/msg_validators/tx_validator/gas_validator.d.ts +58 -3
  71. package/dest/msg_validators/tx_validator/gas_validator.d.ts.map +1 -1
  72. package/dest/msg_validators/tx_validator/gas_validator.js +73 -36
  73. package/dest/msg_validators/tx_validator/index.d.ts +2 -1
  74. package/dest/msg_validators/tx_validator/index.d.ts.map +1 -1
  75. package/dest/msg_validators/tx_validator/index.js +1 -0
  76. package/dest/msg_validators/tx_validator/nullifier_cache.d.ts +14 -0
  77. package/dest/msg_validators/tx_validator/nullifier_cache.d.ts.map +1 -0
  78. package/dest/msg_validators/tx_validator/nullifier_cache.js +24 -0
  79. package/dest/services/dummy_service.d.ts +4 -4
  80. package/dest/services/dummy_service.d.ts.map +1 -1
  81. package/dest/services/dummy_service.js +4 -4
  82. package/dest/services/encoding.d.ts +2 -2
  83. package/dest/services/encoding.d.ts.map +1 -1
  84. package/dest/services/encoding.js +9 -8
  85. package/dest/services/gossipsub/topic_score_params.d.ts +18 -6
  86. package/dest/services/gossipsub/topic_score_params.d.ts.map +1 -1
  87. package/dest/services/gossipsub/topic_score_params.js +32 -10
  88. package/dest/services/libp2p/libp2p_service.d.ts +16 -13
  89. package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
  90. package/dest/services/libp2p/libp2p_service.js +69 -81
  91. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts +4 -3
  92. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts.map +1 -1
  93. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.js +19 -46
  94. package/dest/services/reqresp/batch-tx-requester/interface.d.ts +2 -6
  95. package/dest/services/reqresp/batch-tx-requester/interface.d.ts.map +1 -1
  96. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts +10 -13
  97. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts.map +1 -1
  98. package/dest/services/reqresp/batch-tx-requester/missing_txs.js +25 -46
  99. package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts +17 -11
  100. package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts.map +1 -1
  101. package/dest/services/reqresp/batch-tx-requester/peer_collection.js +49 -15
  102. package/dest/services/reqresp/batch-tx-requester/tx_validator.js +2 -2
  103. package/dest/services/reqresp/reqresp.d.ts +1 -1
  104. package/dest/services/reqresp/reqresp.d.ts.map +1 -1
  105. package/dest/services/reqresp/reqresp.js +2 -1
  106. package/dest/services/service.d.ts +5 -3
  107. package/dest/services/service.d.ts.map +1 -1
  108. package/dest/services/tx_collection/fast_tx_collection.d.ts +1 -1
  109. package/dest/services/tx_collection/fast_tx_collection.d.ts.map +1 -1
  110. package/dest/services/tx_collection/fast_tx_collection.js +39 -33
  111. package/dest/services/tx_collection/file_store_tx_collection.d.ts +1 -1
  112. package/dest/services/tx_collection/file_store_tx_collection.d.ts.map +1 -1
  113. package/dest/services/tx_collection/file_store_tx_collection.js +4 -2
  114. package/dest/services/tx_collection/file_store_tx_source.d.ts +15 -6
  115. package/dest/services/tx_collection/file_store_tx_source.d.ts.map +1 -1
  116. package/dest/services/tx_collection/file_store_tx_source.js +47 -16
  117. package/dest/services/tx_collection/instrumentation.d.ts +1 -1
  118. package/dest/services/tx_collection/instrumentation.d.ts.map +1 -1
  119. package/dest/services/tx_collection/instrumentation.js +2 -1
  120. package/dest/services/tx_collection/missing_txs_tracker.d.ts +32 -0
  121. package/dest/services/tx_collection/missing_txs_tracker.d.ts.map +1 -0
  122. package/dest/services/tx_collection/missing_txs_tracker.js +27 -0
  123. package/dest/services/tx_collection/proposal_tx_collector.d.ts +7 -6
  124. package/dest/services/tx_collection/proposal_tx_collector.d.ts.map +1 -1
  125. package/dest/services/tx_collection/proposal_tx_collector.js +5 -4
  126. package/dest/services/tx_collection/slow_tx_collection.d.ts +2 -2
  127. package/dest/services/tx_collection/slow_tx_collection.d.ts.map +1 -1
  128. package/dest/services/tx_collection/slow_tx_collection.js +10 -8
  129. package/dest/services/tx_collection/tx_collection.d.ts +5 -4
  130. package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
  131. package/dest/services/tx_collection/tx_collection_sink.d.ts +6 -5
  132. package/dest/services/tx_collection/tx_collection_sink.d.ts.map +1 -1
  133. package/dest/services/tx_collection/tx_collection_sink.js +13 -22
  134. package/dest/services/tx_collection/tx_source.d.ts +8 -3
  135. package/dest/services/tx_collection/tx_source.d.ts.map +1 -1
  136. package/dest/services/tx_collection/tx_source.js +19 -2
  137. package/dest/services/tx_file_store/tx_file_store.js +1 -1
  138. package/dest/services/tx_provider.d.ts +3 -3
  139. package/dest/services/tx_provider.d.ts.map +1 -1
  140. package/dest/services/tx_provider.js +4 -4
  141. package/dest/test-helpers/make-test-p2p-clients.d.ts +5 -6
  142. package/dest/test-helpers/make-test-p2p-clients.d.ts.map +1 -1
  143. package/dest/test-helpers/make-test-p2p-clients.js +1 -2
  144. package/dest/test-helpers/mock-pubsub.d.ts +4 -4
  145. package/dest/test-helpers/mock-pubsub.d.ts.map +1 -1
  146. package/dest/test-helpers/mock-pubsub.js +8 -2
  147. package/dest/test-helpers/reqresp-nodes.d.ts +2 -3
  148. package/dest/test-helpers/reqresp-nodes.d.ts.map +1 -1
  149. package/dest/test-helpers/reqresp-nodes.js +2 -2
  150. package/dest/test-helpers/testbench-utils.d.ts +6 -3
  151. package/dest/test-helpers/testbench-utils.d.ts.map +1 -1
  152. package/dest/test-helpers/testbench-utils.js +1 -1
  153. package/dest/testbench/p2p_client_testbench_worker.d.ts +2 -2
  154. package/dest/testbench/p2p_client_testbench_worker.d.ts.map +1 -1
  155. package/dest/testbench/p2p_client_testbench_worker.js +11 -11
  156. package/dest/util.d.ts +2 -2
  157. package/dest/util.d.ts.map +1 -1
  158. package/package.json +14 -14
  159. package/src/client/factory.ts +27 -48
  160. package/src/client/interface.ts +11 -20
  161. package/src/client/p2p_client.ts +47 -104
  162. package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker.ts +19 -10
  163. package/src/config.ts +2 -10
  164. package/src/errors/tx-pool.error.ts +12 -0
  165. package/src/mem_pools/tx_pool_v2/deleted_pool.ts +11 -0
  166. package/src/mem_pools/tx_pool_v2/eviction/eviction_manager.ts +21 -8
  167. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.ts +3 -3
  168. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.ts +18 -4
  169. package/src/mem_pools/tx_pool_v2/eviction/index.ts +4 -0
  170. package/src/mem_pools/tx_pool_v2/eviction/interfaces.ts +49 -4
  171. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.ts +2 -2
  172. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.ts +5 -5
  173. package/src/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.ts +12 -9
  174. package/src/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.ts +24 -6
  175. package/src/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.ts +3 -3
  176. package/src/mem_pools/tx_pool_v2/index.ts +1 -1
  177. package/src/mem_pools/tx_pool_v2/instrumentation.ts +69 -0
  178. package/src/mem_pools/tx_pool_v2/interfaces.ts +15 -6
  179. package/src/mem_pools/tx_pool_v2/tx_metadata.ts +96 -10
  180. package/src/mem_pools/tx_pool_v2/tx_pool_indices.ts +25 -14
  181. package/src/mem_pools/tx_pool_v2/tx_pool_v2.ts +12 -7
  182. package/src/mem_pools/tx_pool_v2/tx_pool_v2_impl.ts +303 -144
  183. package/src/msg_validators/tx_validator/README.md +115 -0
  184. package/src/msg_validators/tx_validator/aggregate_tx_validator.ts +3 -3
  185. package/src/msg_validators/tx_validator/factory.ts +353 -77
  186. package/src/msg_validators/tx_validator/gas_validator.ts +90 -27
  187. package/src/msg_validators/tx_validator/index.ts +1 -0
  188. package/src/msg_validators/tx_validator/nullifier_cache.ts +30 -0
  189. package/src/services/dummy_service.ts +6 -6
  190. package/src/services/encoding.ts +7 -7
  191. package/src/services/gossipsub/README.md +29 -14
  192. package/src/services/gossipsub/topic_score_params.ts +49 -13
  193. package/src/services/libp2p/libp2p_service.ts +80 -90
  194. package/src/services/reqresp/batch-tx-requester/batch_tx_requester.ts +20 -48
  195. package/src/services/reqresp/batch-tx-requester/interface.ts +1 -5
  196. package/src/services/reqresp/batch-tx-requester/missing_txs.ts +23 -71
  197. package/src/services/reqresp/batch-tx-requester/peer_collection.ts +63 -24
  198. package/src/services/reqresp/batch-tx-requester/tx_validator.ts +2 -2
  199. package/src/services/reqresp/reqresp.ts +3 -1
  200. package/src/services/service.ts +11 -2
  201. package/src/services/tx_collection/fast_tx_collection.ts +51 -30
  202. package/src/services/tx_collection/file_store_tx_collection.ts +7 -3
  203. package/src/services/tx_collection/file_store_tx_source.ts +61 -17
  204. package/src/services/tx_collection/instrumentation.ts +7 -1
  205. package/src/services/tx_collection/missing_txs_tracker.ts +52 -0
  206. package/src/services/tx_collection/proposal_tx_collector.ts +8 -7
  207. package/src/services/tx_collection/slow_tx_collection.ts +8 -9
  208. package/src/services/tx_collection/tx_collection.ts +4 -3
  209. package/src/services/tx_collection/tx_collection_sink.ts +15 -29
  210. package/src/services/tx_collection/tx_source.ts +22 -3
  211. package/src/services/tx_file_store/tx_file_store.ts +1 -1
  212. package/src/services/tx_provider.ts +2 -2
  213. package/src/test-helpers/make-test-p2p-clients.ts +0 -2
  214. package/src/test-helpers/mock-pubsub.ts +13 -6
  215. package/src/test-helpers/reqresp-nodes.ts +2 -5
  216. package/src/test-helpers/testbench-utils.ts +3 -3
  217. package/src/testbench/p2p_client_testbench_worker.ts +20 -17
  218. package/src/util.ts +7 -1
@@ -2,16 +2,19 @@ import { BlockNumber } from '@aztec/foundation/branded-types';
2
2
  import { Fr } from '@aztec/foundation/curves/bn254';
3
3
  import { ProtocolContractAddress } from '@aztec/protocol-contracts';
4
4
  import { BlockHash, type L2BlockId } from '@aztec/stdlib/block';
5
- import type { Tx } from '@aztec/stdlib/tx';
5
+ import { Gas } from '@aztec/stdlib/gas';
6
+ import { type Tx, TxHash } from '@aztec/stdlib/tx';
6
7
 
7
8
  import { getFeePayerBalanceDelta } from '../../msg_validators/tx_validator/fee_payer_balance.js';
8
9
  import { getTxPriorityFee } from '../tx_pool/priority.js';
9
- import type { PreAddResult } from './eviction/interfaces.js';
10
+ import { type PreAddResult, TxPoolRejectionCode } from './eviction/interfaces.js';
10
11
 
11
12
  /** Validator-compatible data interface, mirroring the subset of PrivateKernelTailCircuitPublicInputs used by validators. */
12
13
  export type TxMetaValidationData = {
13
14
  getNonEmptyNullifiers(): Fr[];
14
15
  expirationTimestamp: bigint;
16
+ /** Whether the tx has public calls. Used to select the correct L2 gas minimum. */
17
+ forPublic?: unknown;
15
18
  constants: {
16
19
  anchorBlockHeader: {
17
20
  hash(): Promise<BlockHash>;
@@ -19,6 +22,9 @@ export type TxMetaValidationData = {
19
22
  blockNumber: BlockNumber;
20
23
  };
21
24
  };
25
+ txContext: {
26
+ gasSettings: { gasLimits: Gas };
27
+ };
22
28
  };
23
29
  };
24
30
 
@@ -34,6 +40,9 @@ export type TxMetaData = {
34
40
  /** The transaction hash as hex string */
35
41
  readonly txHash: string;
36
42
 
43
+ /** The transaction hash as bigint (for efficient Fr conversion in comparisons) */
44
+ readonly txHashBigInt: bigint;
45
+
37
46
  /** Block ID (number and hash) in which the transaction was mined (undefined if not mined) */
38
47
  minedL2BlockId?: L2BlockId;
39
48
 
@@ -63,6 +72,9 @@ export type TxMetaData = {
63
72
 
64
73
  /** Timestamp (ms) when the tx was received into the pool. 0 for hydrated txs (always eligible). */
65
74
  receivedAt: number;
75
+
76
+ /** Estimated memory footprint of this metadata object in bytes */
77
+ readonly estimatedSizeBytes: number;
66
78
  };
67
79
 
68
80
  /** Transaction state derived from TxMetaData fields and pool protection status */
@@ -74,7 +86,9 @@ export type TxState = 'pending' | 'protected' | 'mined' | 'deleted';
74
86
  * Fr values are captured in closures for zero-cost re-validation.
75
87
  */
76
88
  export async function buildTxMetaData(tx: Tx): Promise<TxMetaData> {
77
- const txHash = tx.getTxHash().toString();
89
+ const txHashObj = tx.getTxHash();
90
+ const txHash = txHashObj.toString();
91
+ const txHashBigInt = txHashObj.toBigInt();
78
92
  const nullifierFrs = tx.data.getNonEmptyNullifiers();
79
93
  const nullifiers = nullifierFrs.map(n => n.toString());
80
94
  const anchorBlockHeaderHashFr = await tx.data.constants.anchorBlockHeader.hash();
@@ -86,8 +100,11 @@ export async function buildTxMetaData(tx: Tx): Promise<TxMetaData> {
86
100
 
87
101
  const { feeLimit, claimAmount } = await getFeePayerBalanceDelta(tx, ProtocolContractAddress.FeeJuice);
88
102
 
103
+ const estimatedSizeBytes = estimateTxMetaDataSize(nullifiers.length);
104
+
89
105
  return {
90
106
  txHash,
107
+ txHashBigInt,
91
108
  anchorBlockHeaderHash,
92
109
  priorityFee,
93
110
  feePayer,
@@ -96,21 +113,52 @@ export async function buildTxMetaData(tx: Tx): Promise<TxMetaData> {
96
113
  nullifiers,
97
114
  expirationTimestamp,
98
115
  receivedAt: 0,
116
+ estimatedSizeBytes,
99
117
  data: {
100
118
  getNonEmptyNullifiers: () => nullifierFrs,
101
119
  expirationTimestamp,
120
+ forPublic: !!tx.data.forPublic,
102
121
  constants: {
103
122
  anchorBlockHeader: {
104
123
  hash: () => Promise.resolve(anchorBlockHeaderHashFr),
105
124
  globalVariables: { blockNumber: anchorBlockNumber },
106
125
  },
126
+ txContext: {
127
+ gasSettings: { gasLimits: tx.data.constants.txContext.gasSettings.gasLimits },
128
+ },
107
129
  },
108
130
  },
109
131
  };
110
132
  }
111
133
 
134
+ // V8 JS object overhead (~64 bytes for a plain object with hidden class).
135
+ // String overhead: ~32 bytes header + 1 byte per ASCII char (V8 one-byte strings).
136
+ // Hex string (0x + 64 hex chars = 66 chars): ~98 bytes per string.
137
+ // bigint: ~32 bytes. number: 8 bytes. Fr: ~80 bytes (32 data + object overhead).
138
+ const OBJECT_OVERHEAD = 64;
139
+ const HEX_STRING_BYTES = 98;
140
+ const BIGINT_BYTES = 32;
141
+ const FR_BYTES = 80;
142
+ // Fixed cost: object shell + txHash + anchorBlockHeaderHash + feePayer (3 hex strings)
143
+ // + txHashBigInt + priorityFee + claimAmount + feeLimit + includeByTimestamp (5 bigints)
144
+ // + receivedAt (number, 8 bytes) + estimatedSizeBytes (number, 8 bytes)
145
+ // + data closure object (~OBJECT_OVERHEAD + anchorBlockHeaderHashFr Fr + anchorBlockNumber number)
146
+ const FIXED_METADATA_BYTES =
147
+ OBJECT_OVERHEAD + 3 * HEX_STRING_BYTES + 5 * BIGINT_BYTES + 8 + 8 + OBJECT_OVERHEAD + FR_BYTES + 8;
148
+
149
+ /** Estimates the in-memory size of a TxMetaData object based on the number of nullifiers. */
150
+ function estimateTxMetaDataSize(nullifierCount: number): number {
151
+ // Per nullifier: one hex string in nullifiers[] + one Fr in the captured nullifierFrs[]
152
+ return FIXED_METADATA_BYTES + nullifierCount * (HEX_STRING_BYTES + FR_BYTES);
153
+ }
154
+
155
+ /** Converts a txHash bigint back to the canonical 0x-prefixed 64-char hex string. */
156
+ export function txHashFromBigInt(value: bigint): string {
157
+ return TxHash.fromBigInt(value).toString();
158
+ }
159
+
112
160
  /** Minimal fields required for priority comparison. */
113
- type PriorityComparable = Pick<TxMetaData, 'txHash' | 'priorityFee'>;
161
+ type PriorityComparable = Pick<TxMetaData, 'txHashBigInt' | 'priorityFee'>;
114
162
 
115
163
  /**
116
164
  * Compares two priority fees in ascending order.
@@ -125,10 +173,8 @@ export function compareFee(a: bigint, b: bigint): number {
125
173
  * Uses field element comparison for deterministic ordering.
126
174
  * Returns negative if a < b, positive if a > b, 0 if equal.
127
175
  */
128
- export function compareTxHash(a: string, b: string): number {
129
- const fieldA = Fr.fromHexString(a);
130
- const fieldB = Fr.fromHexString(b);
131
- return fieldA.cmp(fieldB);
176
+ export function compareTxHash(a: bigint, b: bigint): number {
177
+ return Fr.cmpAsBigInt(a, b);
132
178
  }
133
179
 
134
180
  /**
@@ -141,7 +187,7 @@ export function comparePriority(a: PriorityComparable, b: PriorityComparable): n
141
187
  if (feeComparison !== 0) {
142
188
  return feeComparison;
143
189
  }
144
- return compareTxHash(a.txHash, b.txHash);
190
+ return compareTxHash(a.txHashBigInt, b.txHashBigInt);
145
191
  }
146
192
 
147
193
  /**
@@ -188,7 +234,11 @@ export function checkNullifierConflict(
188
234
  return {
189
235
  shouldIgnore: true,
190
236
  txHashesToEvict: [],
191
- reason: `nullifier conflict with ${conflictingHashStr}`,
237
+ reason: {
238
+ code: TxPoolRejectionCode.NULLIFIER_CONFLICT,
239
+ message: `Nullifier conflict with existing tx ${conflictingHashStr}`,
240
+ conflictingTxHash: conflictingHashStr,
241
+ },
192
242
  };
193
243
  }
194
244
  }
@@ -206,6 +256,42 @@ export function stubTxMetaValidationData(overrides: { expirationTimestamp?: bigi
206
256
  hash: () => Promise.resolve(new BlockHash(Fr.ZERO)),
207
257
  globalVariables: { blockNumber: BlockNumber(0) },
208
258
  },
259
+ txContext: {
260
+ gasSettings: { gasLimits: Gas.empty() },
261
+ },
209
262
  },
210
263
  };
211
264
  }
265
+
266
+ /** Creates a stub TxMetaData for tests. All fields have sensible defaults and can be overridden. */
267
+ export function stubTxMetaData(
268
+ txHash: string,
269
+ overrides: {
270
+ priorityFee?: bigint;
271
+ feePayer?: string;
272
+ claimAmount?: bigint;
273
+ feeLimit?: bigint;
274
+ nullifiers?: string[];
275
+ expirationTimestamp?: bigint;
276
+ anchorBlockHeaderHash?: string;
277
+ } = {},
278
+ ): TxMetaData {
279
+ const txHashBigInt = Fr.fromHexString(txHash).toBigInt();
280
+ // Normalize to canonical zero-padded hex so txHashFromBigInt(txHashBigInt) === normalizedTxHash
281
+ const normalizedTxHash = txHashFromBigInt(txHashBigInt);
282
+ const expirationTimestamp = overrides.expirationTimestamp ?? 0n;
283
+ return {
284
+ txHash: normalizedTxHash,
285
+ txHashBigInt,
286
+ anchorBlockHeaderHash: overrides.anchorBlockHeaderHash ?? '0x1234',
287
+ priorityFee: overrides.priorityFee ?? 100n,
288
+ feePayer: overrides.feePayer ?? '0xfeepayer',
289
+ claimAmount: overrides.claimAmount ?? 0n,
290
+ feeLimit: overrides.feeLimit ?? 100n,
291
+ nullifiers: overrides.nullifiers ?? [`0x${normalizedTxHash.slice(2)}null1`],
292
+ expirationTimestamp,
293
+ receivedAt: 0,
294
+ estimatedSizeBytes: 0,
295
+ data: stubTxMetaValidationData({ expirationTimestamp }),
296
+ };
297
+ }
@@ -1,7 +1,7 @@
1
1
  import { SlotNumber } from '@aztec/foundation/branded-types';
2
2
  import type { L2BlockId } from '@aztec/stdlib/block';
3
3
 
4
- import { type TxMetaData, type TxState, compareFee, compareTxHash } from './tx_metadata.js';
4
+ import { type TxMetaData, type TxState, compareFee, compareTxHash, txHashFromBigInt } from './tx_metadata.js';
5
5
 
6
6
  /**
7
7
  * Manages in-memory indices for the transaction pool.
@@ -22,8 +22,8 @@ export class TxPoolIndices {
22
22
  #nullifierToTxHash: Map<string, string> = new Map();
23
23
  /** Fee payer to txHashes index (pending txs only) */
24
24
  #feePayerToTxHashes: Map<string, Set<string>> = new Map();
25
- /** Pending txHashes grouped by priority fee */
26
- #pendingByPriority: Map<bigint, Set<string>> = new Map();
25
+ /** Pending txHash bigints grouped by priority fee */
26
+ #pendingByPriority: Map<bigint, Set<bigint>> = new Map();
27
27
  /** Protected transactions: txHash -> slotNumber */
28
28
  #protectedTransactions: Map<string, SlotNumber> = new Map();
29
29
 
@@ -73,17 +73,17 @@ export class TxPoolIndices {
73
73
  * @param order - 'desc' for highest priority first, 'asc' for lowest priority first
74
74
  */
75
75
  *iteratePendingByPriority(order: 'asc' | 'desc', filter?: (hash: string) => boolean): Generator<string> {
76
- // Use compareFee from tx_metadata, swap args for descending order
77
76
  const feeCompareFn = order === 'desc' ? (a: bigint, b: bigint) => compareFee(b, a) : compareFee;
78
- const hashCompareFn = order === 'desc' ? (a: string, b: string) => compareTxHash(b, a) : compareTxHash;
77
+ const hashCompareFn =
78
+ order === 'desc' ? (a: bigint, b: bigint) => compareTxHash(b, a) : (a: bigint, b: bigint) => compareTxHash(a, b);
79
79
 
80
80
  const sortedFees = [...this.#pendingByPriority.keys()].sort(feeCompareFn);
81
81
 
82
82
  for (const fee of sortedFees) {
83
83
  const hashesAtFee = this.#pendingByPriority.get(fee)!;
84
- // Use compareTxHash from tx_metadata, swap args for descending order
85
84
  const sortedHashes = [...hashesAtFee].sort(hashCompareFn);
86
- for (const hash of sortedHashes) {
85
+ for (const hashBigInt of sortedHashes) {
86
+ const hash = txHashFromBigInt(hashBigInt);
87
87
  if (filter === undefined || filter(hash)) {
88
88
  yield hash;
89
89
  }
@@ -265,8 +265,8 @@ export class TxPoolIndices {
265
265
  getPendingTxs(): TxMetaData[] {
266
266
  const result: TxMetaData[] = [];
267
267
  for (const hashSet of this.#pendingByPriority.values()) {
268
- for (const txHash of hashSet) {
269
- const meta = this.#metadata.get(txHash);
268
+ for (const txHashBigInt of hashSet) {
269
+ const meta = this.#metadata.get(txHashFromBigInt(txHashBigInt));
270
270
  if (meta) {
271
271
  result.push(meta);
272
272
  }
@@ -348,13 +348,15 @@ export class TxPoolIndices {
348
348
  // METRICS
349
349
  // ============================================================================
350
350
 
351
- /** Counts transactions by state */
352
- countTxs(): { pending: number; protected: number; mined: number } {
351
+ /** Counts transactions by state and estimates total metadata memory usage */
352
+ countTxs(): { pending: number; protected: number; mined: number; totalMetadataBytes: number } {
353
353
  let pending = 0;
354
354
  let protected_ = 0;
355
355
  let mined = 0;
356
+ let totalMetadataBytes = 0;
356
357
 
357
358
  for (const meta of this.#metadata.values()) {
359
+ totalMetadataBytes += meta.estimatedSizeBytes;
358
360
  const state = this.getTxState(meta);
359
361
  if (state === 'pending') {
360
362
  pending++;
@@ -365,7 +367,16 @@ export class TxPoolIndices {
365
367
  }
366
368
  }
367
369
 
368
- return { pending, protected: protected_, mined };
370
+ return { pending, protected: protected_, mined, totalMetadataBytes };
371
+ }
372
+
373
+ /** Returns the estimated total memory consumed by all metadata objects */
374
+ getTotalMetadataBytes(): number {
375
+ let total = 0;
376
+ for (const meta of this.#metadata.values()) {
377
+ total += meta.estimatedSizeBytes;
378
+ }
379
+ return total;
369
380
  }
370
381
 
371
382
  /** Gets all mined transactions with their block IDs */
@@ -403,7 +414,7 @@ export class TxPoolIndices {
403
414
  prioritySet = new Set();
404
415
  this.#pendingByPriority.set(meta.priorityFee, prioritySet);
405
416
  }
406
- prioritySet.add(meta.txHash);
417
+ prioritySet.add(meta.txHashBigInt);
407
418
  }
408
419
 
409
420
  #removeFromPendingIndices(meta: TxMetaData): void {
@@ -424,7 +435,7 @@ export class TxPoolIndices {
424
435
  // Remove from priority map
425
436
  const hashSet = this.#pendingByPriority.get(meta.priorityFee);
426
437
  if (hashSet) {
427
- hashSet.delete(meta.txHash);
438
+ hashSet.delete(meta.txHashBigInt);
428
439
  if (hashSet.size === 0) {
429
440
  this.#pendingByPriority.delete(meta.priorityFee);
430
441
  }
@@ -61,7 +61,7 @@ export class AztecKVTxPoolV2 extends (EventEmitter as new () => TypedEventEmitte
61
61
  };
62
62
 
63
63
  // Create the implementation
64
- this.#impl = new TxPoolV2Impl(store, archiveStore, deps, callbacks, config, dateProvider, log);
64
+ this.#impl = new TxPoolV2Impl(store, archiveStore, deps, callbacks, telemetry, config, dateProvider, log);
65
65
  }
66
66
 
67
67
  // ============================================================================
@@ -70,11 +70,11 @@ export class AztecKVTxPoolV2 extends (EventEmitter as new () => TypedEventEmitte
70
70
 
71
71
  // === Core Operations ===
72
72
 
73
- addPendingTxs(txs: Tx[], opts: { source?: string } = {}): Promise<AddTxsResult> {
73
+ addPendingTxs(txs: Tx[], opts: { source?: string; feeComparisonOnly?: boolean } = {}): Promise<AddTxsResult> {
74
74
  return this.#queue.put(() => this.#impl.addPendingTxs(txs, opts));
75
75
  }
76
76
 
77
- canAddPendingTx(tx: Tx): Promise<'accepted' | 'ignored' | 'rejected'> {
77
+ canAddPendingTx(tx: Tx): Promise<'accepted' | 'ignored'> {
78
78
  return this.#queue.put(() => this.#impl.canAddPendingTx(tx));
79
79
  }
80
80
 
@@ -83,7 +83,7 @@ export class AztecKVTxPoolV2 extends (EventEmitter as new () => TypedEventEmitte
83
83
  }
84
84
 
85
85
  protectTxs(txHashes: TxHash[], block: BlockHeader): Promise<TxHash[]> {
86
- return this.#queue.put(() => Promise.resolve(this.#impl.protectTxs(txHashes, block)));
86
+ return this.#queue.put(() => this.#impl.protectTxs(txHashes, block));
87
87
  }
88
88
 
89
89
  addMinedTxs(txs: Tx[], block: BlockHeader, opts: { source?: string } = {}): Promise<void> {
@@ -100,8 +100,8 @@ export class AztecKVTxPoolV2 extends (EventEmitter as new () => TypedEventEmitte
100
100
  return this.#queue.put(() => this.#impl.prepareForSlot(slotNumber));
101
101
  }
102
102
 
103
- handlePrunedBlocks(latestBlock: L2BlockId): Promise<void> {
104
- return this.#queue.put(() => this.#impl.handlePrunedBlocks(latestBlock));
103
+ handlePrunedBlocks(latestBlock: L2BlockId, options?: { deleteAllTxs?: boolean }): Promise<void> {
104
+ return this.#queue.put(() => this.#impl.handlePrunedBlocks(latestBlock, options));
105
105
  }
106
106
 
107
107
  handleFailedExecution(txHashes: TxHash[]): Promise<void> {
@@ -195,7 +195,12 @@ export class AztecKVTxPoolV2 extends (EventEmitter as new () => TypedEventEmitte
195
195
  this.#queue.put(() => {
196
196
  const counts = this.#impl.countTxs();
197
197
  return Promise.resolve({
198
- itemCount: { pending: counts.pending, protected: counts.protected, mined: counts.mined },
198
+ itemCount: {
199
+ pending: counts.pending,
200
+ protected: counts.protected,
201
+ mined: counts.mined,
202
+ softDeleted: counts.softDeleted,
203
+ },
199
204
  });
200
205
  }),
201
206
  () => this.#store.estimateSize(),