@aztec/p2p 0.0.1-commit.6d3c34e → 0.0.1-commit.7035c9bd6

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 (535) hide show
  1. package/README.md +129 -3
  2. package/dest/bootstrap/bootstrap.d.ts +4 -3
  3. package/dest/bootstrap/bootstrap.d.ts.map +1 -1
  4. package/dest/bootstrap/bootstrap.js +4 -4
  5. package/dest/client/factory.d.ts +11 -11
  6. package/dest/client/factory.d.ts.map +1 -1
  7. package/dest/client/factory.js +58 -18
  8. package/dest/client/interface.d.ts +46 -33
  9. package/dest/client/interface.d.ts.map +1 -1
  10. package/dest/client/p2p_client.d.ts +41 -51
  11. package/dest/client/p2p_client.d.ts.map +1 -1
  12. package/dest/client/p2p_client.js +169 -225
  13. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.d.ts +2 -0
  14. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.d.ts.map +1 -0
  15. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.js +305 -0
  16. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.d.ts +73 -0
  17. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.d.ts.map +1 -0
  18. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.js +8 -0
  19. package/dest/config.d.ts +65 -16
  20. package/dest/config.d.ts.map +1 -1
  21. package/dest/config.js +108 -40
  22. package/dest/errors/tx-pool.error.d.ts +8 -0
  23. package/dest/errors/tx-pool.error.d.ts.map +1 -0
  24. package/dest/errors/tx-pool.error.js +9 -0
  25. package/dest/index.d.ts +2 -1
  26. package/dest/index.d.ts.map +1 -1
  27. package/dest/index.js +1 -0
  28. package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +104 -88
  29. package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -1
  30. package/dest/mem_pools/attestation_pool/attestation_pool.js +445 -3
  31. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.d.ts +2 -2
  32. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.d.ts.map +1 -1
  33. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.js +373 -96
  34. package/dest/mem_pools/attestation_pool/index.d.ts +2 -3
  35. package/dest/mem_pools/attestation_pool/index.d.ts.map +1 -1
  36. package/dest/mem_pools/attestation_pool/index.js +1 -2
  37. package/dest/mem_pools/attestation_pool/mocks.d.ts +4 -2
  38. package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -1
  39. package/dest/mem_pools/attestation_pool/mocks.js +8 -5
  40. package/dest/mem_pools/index.d.ts +3 -2
  41. package/dest/mem_pools/index.d.ts.map +1 -1
  42. package/dest/mem_pools/index.js +1 -1
  43. package/dest/mem_pools/instrumentation.d.ts +4 -2
  44. package/dest/mem_pools/instrumentation.d.ts.map +1 -1
  45. package/dest/mem_pools/instrumentation.js +18 -16
  46. package/dest/mem_pools/interface.d.ts +5 -5
  47. package/dest/mem_pools/interface.d.ts.map +1 -1
  48. package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts +15 -10
  49. package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts.map +1 -1
  50. package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.js +91 -50
  51. package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts +19 -5
  52. package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts.map +1 -1
  53. package/dest/mem_pools/tx_pool/eviction/eviction_manager.js +59 -3
  54. package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts +79 -5
  55. package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts.map +1 -1
  56. package/dest/mem_pools/tx_pool/eviction/eviction_strategy.js +47 -0
  57. package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts +16 -0
  58. package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -0
  59. package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.js +123 -0
  60. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts +2 -2
  61. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts.map +1 -1
  62. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.js +3 -3
  63. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts +4 -4
  64. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts.map +1 -1
  65. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.js +2 -0
  66. package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts +2 -2
  67. package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts.map +1 -1
  68. package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.d.ts +25 -0
  69. package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.d.ts.map +1 -0
  70. package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.js +57 -0
  71. package/dest/mem_pools/tx_pool/priority.d.ts +2 -2
  72. package/dest/mem_pools/tx_pool/priority.d.ts.map +1 -1
  73. package/dest/mem_pools/tx_pool/priority.js +4 -4
  74. package/dest/mem_pools/tx_pool/tx_pool_test_suite.d.ts +1 -1
  75. package/dest/mem_pools/tx_pool/tx_pool_test_suite.d.ts.map +1 -1
  76. package/dest/mem_pools/tx_pool/tx_pool_test_suite.js +3 -1
  77. package/dest/mem_pools/tx_pool_v2/archive/index.d.ts +2 -0
  78. package/dest/mem_pools/tx_pool_v2/archive/index.d.ts.map +1 -0
  79. package/dest/mem_pools/tx_pool_v2/archive/index.js +1 -0
  80. package/dest/mem_pools/tx_pool_v2/archive/tx_archive.d.ts +43 -0
  81. package/dest/mem_pools/tx_pool_v2/archive/tx_archive.d.ts.map +1 -0
  82. package/dest/mem_pools/tx_pool_v2/archive/tx_archive.js +103 -0
  83. package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts +104 -0
  84. package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts.map +1 -0
  85. package/dest/mem_pools/tx_pool_v2/deleted_pool.js +251 -0
  86. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts +47 -0
  87. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts.map +1 -0
  88. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.js +128 -0
  89. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts +17 -0
  90. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -0
  91. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.js +94 -0
  92. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts +19 -0
  93. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts.map +1 -0
  94. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.js +97 -0
  95. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts +10 -0
  96. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts.map +1 -0
  97. package/dest/mem_pools/tx_pool_v2/eviction/index.js +11 -0
  98. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts +180 -0
  99. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts.map +1 -0
  100. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.js +25 -0
  101. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.d.ts +15 -0
  102. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.d.ts.map +1 -0
  103. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.js +65 -0
  104. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.d.ts +17 -0
  105. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.d.ts.map +1 -0
  106. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.js +93 -0
  107. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts +16 -0
  108. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts.map +1 -0
  109. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.js +78 -0
  110. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts +20 -0
  111. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts.map +1 -0
  112. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.js +75 -0
  113. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts +15 -0
  114. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts.map +1 -0
  115. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.js +19 -0
  116. package/dest/mem_pools/tx_pool_v2/index.d.ts +6 -0
  117. package/dest/mem_pools/tx_pool_v2/index.d.ts.map +1 -0
  118. package/dest/mem_pools/tx_pool_v2/index.js +5 -0
  119. package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts +15 -0
  120. package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts.map +1 -0
  121. package/dest/mem_pools/tx_pool_v2/instrumentation.js +43 -0
  122. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts +215 -0
  123. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts.map +1 -0
  124. package/dest/mem_pools/tx_pool_v2/interfaces.js +10 -0
  125. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts +134 -0
  126. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts.map +1 -0
  127. package/dest/mem_pools/tx_pool_v2/tx_metadata.js +216 -0
  128. package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.d.ts +26 -0
  129. package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.d.ts.map +1 -0
  130. package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.js +70 -0
  131. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts +108 -0
  132. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts.map +1 -0
  133. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.js +337 -0
  134. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts +62 -0
  135. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts.map +1 -0
  136. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.js +167 -0
  137. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts +78 -0
  138. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts.map +1 -0
  139. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.js +923 -0
  140. package/dest/msg_validators/attestation_validator/attestation_validator.d.ts +3 -3
  141. package/dest/msg_validators/attestation_validator/attestation_validator.d.ts.map +1 -1
  142. package/dest/msg_validators/attestation_validator/attestation_validator.js +43 -11
  143. package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts +5 -5
  144. package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts.map +1 -1
  145. package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.js +18 -6
  146. package/dest/msg_validators/clock_tolerance.d.ts +21 -0
  147. package/dest/msg_validators/clock_tolerance.d.ts.map +1 -0
  148. package/dest/msg_validators/clock_tolerance.js +38 -0
  149. package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts +6 -4
  150. package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts.map +1 -1
  151. package/dest/msg_validators/proposal_validator/block_proposal_validator.js +10 -2
  152. package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts +6 -4
  153. package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts.map +1 -1
  154. package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.js +16 -2
  155. package/dest/msg_validators/proposal_validator/proposal_validator.d.ts +13 -8
  156. package/dest/msg_validators/proposal_validator/proposal_validator.d.ts.map +1 -1
  157. package/dest/msg_validators/proposal_validator/proposal_validator.js +90 -54
  158. package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts +4 -4
  159. package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts.map +1 -1
  160. package/dest/msg_validators/tx_validator/aggregate_tx_validator.js +3 -3
  161. package/dest/msg_validators/tx_validator/allowed_public_setup.d.ts +2 -1
  162. package/dest/msg_validators/tx_validator/allowed_public_setup.d.ts.map +1 -1
  163. package/dest/msg_validators/tx_validator/allowed_public_setup.js +24 -20
  164. package/dest/msg_validators/tx_validator/allowed_setup_helpers.d.ts +17 -0
  165. package/dest/msg_validators/tx_validator/allowed_setup_helpers.d.ts.map +1 -0
  166. package/dest/msg_validators/tx_validator/allowed_setup_helpers.js +24 -0
  167. package/dest/msg_validators/tx_validator/archive_cache.d.ts +3 -3
  168. package/dest/msg_validators/tx_validator/archive_cache.d.ts.map +1 -1
  169. package/dest/msg_validators/tx_validator/archive_cache.js +1 -1
  170. package/dest/msg_validators/tx_validator/block_header_validator.d.ts +20 -6
  171. package/dest/msg_validators/tx_validator/block_header_validator.d.ts.map +1 -1
  172. package/dest/msg_validators/tx_validator/block_header_validator.js +4 -3
  173. package/dest/msg_validators/tx_validator/data_validator.d.ts +3 -1
  174. package/dest/msg_validators/tx_validator/data_validator.d.ts.map +1 -1
  175. package/dest/msg_validators/tx_validator/data_validator.js +4 -1
  176. package/dest/msg_validators/tx_validator/double_spend_validator.d.ts +15 -4
  177. package/dest/msg_validators/tx_validator/double_spend_validator.d.ts.map +1 -1
  178. package/dest/msg_validators/tx_validator/double_spend_validator.js +7 -6
  179. package/dest/msg_validators/tx_validator/factory.d.ts +137 -5
  180. package/dest/msg_validators/tx_validator/factory.d.ts.map +1 -1
  181. package/dest/msg_validators/tx_validator/factory.js +249 -58
  182. package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts +10 -0
  183. package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts.map +1 -0
  184. package/dest/msg_validators/tx_validator/fee_payer_balance.js +24 -0
  185. package/dest/msg_validators/tx_validator/gas_validator.d.ts +68 -3
  186. package/dest/msg_validators/tx_validator/gas_validator.d.ts.map +1 -1
  187. package/dest/msg_validators/tx_validator/gas_validator.js +114 -52
  188. package/dest/msg_validators/tx_validator/index.d.ts +4 -1
  189. package/dest/msg_validators/tx_validator/index.d.ts.map +1 -1
  190. package/dest/msg_validators/tx_validator/index.js +3 -0
  191. package/dest/msg_validators/tx_validator/metadata_validator.d.ts +3 -2
  192. package/dest/msg_validators/tx_validator/metadata_validator.d.ts.map +1 -1
  193. package/dest/msg_validators/tx_validator/metadata_validator.js +6 -6
  194. package/dest/msg_validators/tx_validator/nullifier_cache.d.ts +14 -0
  195. package/dest/msg_validators/tx_validator/nullifier_cache.d.ts.map +1 -0
  196. package/dest/msg_validators/tx_validator/nullifier_cache.js +24 -0
  197. package/dest/msg_validators/tx_validator/phases_validator.d.ts +24 -3
  198. package/dest/msg_validators/tx_validator/phases_validator.d.ts.map +1 -1
  199. package/dest/msg_validators/tx_validator/phases_validator.js +74 -26
  200. package/dest/msg_validators/tx_validator/size_validator.d.ts +8 -0
  201. package/dest/msg_validators/tx_validator/size_validator.d.ts.map +1 -0
  202. package/dest/msg_validators/tx_validator/size_validator.js +23 -0
  203. package/dest/msg_validators/tx_validator/timestamp_validator.d.ts +22 -5
  204. package/dest/msg_validators/tx_validator/timestamp_validator.d.ts.map +1 -1
  205. package/dest/msg_validators/tx_validator/timestamp_validator.js +8 -8
  206. package/dest/msg_validators/tx_validator/tx_permitted_validator.d.ts +3 -2
  207. package/dest/msg_validators/tx_validator/tx_permitted_validator.d.ts.map +1 -1
  208. package/dest/msg_validators/tx_validator/tx_permitted_validator.js +2 -2
  209. package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts +3 -2
  210. package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts.map +1 -1
  211. package/dest/msg_validators/tx_validator/tx_proof_validator.js +2 -2
  212. package/dest/services/data_store.d.ts +1 -1
  213. package/dest/services/data_store.d.ts.map +1 -1
  214. package/dest/services/data_store.js +10 -6
  215. package/dest/services/discv5/discV5_service.js +1 -1
  216. package/dest/services/dummy_service.d.ts +24 -4
  217. package/dest/services/dummy_service.d.ts.map +1 -1
  218. package/dest/services/dummy_service.js +46 -1
  219. package/dest/services/encoding.d.ts +7 -3
  220. package/dest/services/encoding.d.ts.map +1 -1
  221. package/dest/services/encoding.js +20 -14
  222. package/dest/services/gossipsub/index.d.ts +3 -0
  223. package/dest/services/gossipsub/index.d.ts.map +1 -0
  224. package/dest/services/gossipsub/index.js +2 -0
  225. package/dest/services/gossipsub/scoring.d.ts +21 -3
  226. package/dest/services/gossipsub/scoring.d.ts.map +1 -1
  227. package/dest/services/gossipsub/scoring.js +24 -7
  228. package/dest/services/gossipsub/topic_score_params.d.ts +173 -0
  229. package/dest/services/gossipsub/topic_score_params.d.ts.map +1 -0
  230. package/dest/services/gossipsub/topic_score_params.js +346 -0
  231. package/dest/services/index.d.ts +2 -1
  232. package/dest/services/index.d.ts.map +1 -1
  233. package/dest/services/index.js +1 -0
  234. package/dest/services/libp2p/instrumentation.d.ts +1 -1
  235. package/dest/services/libp2p/instrumentation.d.ts.map +1 -1
  236. package/dest/services/libp2p/instrumentation.js +19 -8
  237. package/dest/services/libp2p/libp2p_service.d.ts +97 -46
  238. package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
  239. package/dest/services/libp2p/libp2p_service.js +497 -392
  240. package/dest/services/peer-manager/metrics.d.ts +4 -2
  241. package/dest/services/peer-manager/metrics.d.ts.map +1 -1
  242. package/dest/services/peer-manager/metrics.js +26 -5
  243. package/dest/services/peer-manager/peer_manager.d.ts +1 -1
  244. package/dest/services/peer-manager/peer_manager.d.ts.map +1 -1
  245. package/dest/services/peer-manager/peer_manager.js +4 -3
  246. package/dest/services/peer-manager/peer_scoring.d.ts +1 -1
  247. package/dest/services/peer-manager/peer_scoring.d.ts.map +1 -1
  248. package/dest/services/peer-manager/peer_scoring.js +33 -4
  249. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts +51 -0
  250. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts.map +1 -0
  251. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.js +540 -0
  252. package/dest/services/reqresp/batch-tx-requester/config.d.ts +17 -0
  253. package/dest/services/reqresp/batch-tx-requester/config.d.ts.map +1 -0
  254. package/dest/services/reqresp/batch-tx-requester/config.js +27 -0
  255. package/dest/services/reqresp/batch-tx-requester/interface.d.ts +47 -0
  256. package/dest/services/reqresp/batch-tx-requester/interface.d.ts.map +1 -0
  257. package/dest/services/reqresp/batch-tx-requester/interface.js +1 -0
  258. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts +35 -0
  259. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts.map +1 -0
  260. package/dest/services/reqresp/batch-tx-requester/missing_txs.js +136 -0
  261. package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts +62 -0
  262. package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts.map +1 -0
  263. package/dest/services/reqresp/batch-tx-requester/peer_collection.js +176 -0
  264. package/dest/services/reqresp/batch-tx-requester/tx_validator.d.ts +20 -0
  265. package/dest/services/reqresp/batch-tx-requester/tx_validator.d.ts.map +1 -0
  266. package/dest/services/reqresp/batch-tx-requester/tx_validator.js +21 -0
  267. package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts +22 -3
  268. package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts.map +1 -1
  269. package/dest/services/reqresp/connection-sampler/batch_connection_sampler.js +63 -4
  270. package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts +2 -1
  271. package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts.map +1 -1
  272. package/dest/services/reqresp/connection-sampler/connection_sampler.js +12 -0
  273. package/dest/services/reqresp/constants.d.ts +12 -0
  274. package/dest/services/reqresp/constants.d.ts.map +1 -0
  275. package/dest/services/reqresp/constants.js +7 -0
  276. package/dest/services/reqresp/interface.d.ts +12 -1
  277. package/dest/services/reqresp/interface.d.ts.map +1 -1
  278. package/dest/services/reqresp/interface.js +15 -1
  279. package/dest/services/reqresp/metrics.d.ts +6 -5
  280. package/dest/services/reqresp/metrics.d.ts.map +1 -1
  281. package/dest/services/reqresp/metrics.js +17 -5
  282. package/dest/services/reqresp/protocols/block_txs/bitvector.d.ts +5 -1
  283. package/dest/services/reqresp/protocols/block_txs/bitvector.d.ts.map +1 -1
  284. package/dest/services/reqresp/protocols/block_txs/bitvector.js +12 -0
  285. package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts +7 -5
  286. package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts.map +1 -1
  287. package/dest/services/reqresp/protocols/block_txs/block_txs_handler.js +27 -9
  288. package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts +29 -6
  289. package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts.map +1 -1
  290. package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.js +59 -13
  291. package/dest/services/reqresp/protocols/status.d.ts +1 -1
  292. package/dest/services/reqresp/protocols/status.d.ts.map +1 -1
  293. package/dest/services/reqresp/protocols/status.js +2 -1
  294. package/dest/services/reqresp/protocols/tx.d.ts +7 -1
  295. package/dest/services/reqresp/protocols/tx.d.ts.map +1 -1
  296. package/dest/services/reqresp/protocols/tx.js +20 -0
  297. package/dest/services/reqresp/reqresp.d.ts +6 -1
  298. package/dest/services/reqresp/reqresp.d.ts.map +1 -1
  299. package/dest/services/reqresp/reqresp.js +86 -34
  300. package/dest/services/service.d.ts +42 -3
  301. package/dest/services/service.d.ts.map +1 -1
  302. package/dest/services/tx_collection/config.d.ts +22 -1
  303. package/dest/services/tx_collection/config.d.ts.map +1 -1
  304. package/dest/services/tx_collection/config.js +55 -1
  305. package/dest/services/tx_collection/fast_tx_collection.d.ts +7 -7
  306. package/dest/services/tx_collection/fast_tx_collection.d.ts.map +1 -1
  307. package/dest/services/tx_collection/fast_tx_collection.js +95 -84
  308. package/dest/services/tx_collection/file_store_tx_collection.d.ts +53 -0
  309. package/dest/services/tx_collection/file_store_tx_collection.d.ts.map +1 -0
  310. package/dest/services/tx_collection/file_store_tx_collection.js +167 -0
  311. package/dest/services/tx_collection/file_store_tx_source.d.ts +37 -0
  312. package/dest/services/tx_collection/file_store_tx_source.d.ts.map +1 -0
  313. package/dest/services/tx_collection/file_store_tx_source.js +90 -0
  314. package/dest/services/tx_collection/index.d.ts +3 -1
  315. package/dest/services/tx_collection/index.d.ts.map +1 -1
  316. package/dest/services/tx_collection/index.js +2 -0
  317. package/dest/services/tx_collection/instrumentation.d.ts +1 -1
  318. package/dest/services/tx_collection/instrumentation.d.ts.map +1 -1
  319. package/dest/services/tx_collection/instrumentation.js +10 -2
  320. package/dest/services/tx_collection/proposal_tx_collector.d.ts +48 -0
  321. package/dest/services/tx_collection/proposal_tx_collector.d.ts.map +1 -0
  322. package/dest/services/tx_collection/proposal_tx_collector.js +50 -0
  323. package/dest/services/tx_collection/request_tracker.d.ts +53 -0
  324. package/dest/services/tx_collection/request_tracker.d.ts.map +1 -0
  325. package/dest/services/tx_collection/request_tracker.js +84 -0
  326. package/dest/services/tx_collection/slow_tx_collection.d.ts +9 -5
  327. package/dest/services/tx_collection/slow_tx_collection.d.ts.map +1 -1
  328. package/dest/services/tx_collection/slow_tx_collection.js +60 -26
  329. package/dest/services/tx_collection/tx_collection.d.ts +29 -19
  330. package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
  331. package/dest/services/tx_collection/tx_collection.js +79 -7
  332. package/dest/services/tx_collection/tx_collection_sink.d.ts +18 -8
  333. package/dest/services/tx_collection/tx_collection_sink.d.ts.map +1 -1
  334. package/dest/services/tx_collection/tx_collection_sink.js +26 -29
  335. package/dest/services/tx_collection/tx_source.d.ts +8 -3
  336. package/dest/services/tx_collection/tx_source.d.ts.map +1 -1
  337. package/dest/services/tx_collection/tx_source.js +19 -2
  338. package/dest/services/tx_file_store/config.d.ts +16 -0
  339. package/dest/services/tx_file_store/config.d.ts.map +1 -0
  340. package/dest/services/tx_file_store/config.js +22 -0
  341. package/dest/services/tx_file_store/index.d.ts +4 -0
  342. package/dest/services/tx_file_store/index.d.ts.map +1 -0
  343. package/dest/services/tx_file_store/index.js +3 -0
  344. package/dest/services/tx_file_store/instrumentation.d.ts +15 -0
  345. package/dest/services/tx_file_store/instrumentation.d.ts.map +1 -0
  346. package/dest/services/tx_file_store/instrumentation.js +29 -0
  347. package/dest/services/tx_file_store/tx_file_store.d.ts +48 -0
  348. package/dest/services/tx_file_store/tx_file_store.d.ts.map +1 -0
  349. package/dest/services/tx_file_store/tx_file_store.js +152 -0
  350. package/dest/services/tx_provider.d.ts +6 -6
  351. package/dest/services/tx_provider.d.ts.map +1 -1
  352. package/dest/services/tx_provider.js +9 -8
  353. package/dest/services/tx_provider_instrumentation.d.ts +1 -1
  354. package/dest/services/tx_provider_instrumentation.d.ts.map +1 -1
  355. package/dest/services/tx_provider_instrumentation.js +5 -5
  356. package/dest/test-helpers/index.d.ts +3 -1
  357. package/dest/test-helpers/index.d.ts.map +1 -1
  358. package/dest/test-helpers/index.js +2 -0
  359. package/dest/test-helpers/make-test-p2p-clients.d.ts +7 -8
  360. package/dest/test-helpers/make-test-p2p-clients.d.ts.map +1 -1
  361. package/dest/test-helpers/make-test-p2p-clients.js +1 -2
  362. package/dest/test-helpers/mock-pubsub.d.ts +30 -4
  363. package/dest/test-helpers/mock-pubsub.d.ts.map +1 -1
  364. package/dest/test-helpers/mock-pubsub.js +105 -4
  365. package/dest/test-helpers/reqresp-nodes.d.ts +2 -3
  366. package/dest/test-helpers/reqresp-nodes.d.ts.map +1 -1
  367. package/dest/test-helpers/reqresp-nodes.js +4 -3
  368. package/dest/test-helpers/test_tx_provider.d.ts +40 -0
  369. package/dest/test-helpers/test_tx_provider.d.ts.map +1 -0
  370. package/dest/test-helpers/test_tx_provider.js +41 -0
  371. package/dest/test-helpers/testbench-utils.d.ts +163 -0
  372. package/dest/test-helpers/testbench-utils.d.ts.map +1 -0
  373. package/dest/test-helpers/testbench-utils.js +385 -0
  374. package/dest/testbench/p2p_client_testbench_worker.d.ts +28 -2
  375. package/dest/testbench/p2p_client_testbench_worker.d.ts.map +1 -1
  376. package/dest/testbench/p2p_client_testbench_worker.js +222 -140
  377. package/dest/testbench/worker_client_manager.d.ts +53 -6
  378. package/dest/testbench/worker_client_manager.d.ts.map +1 -1
  379. package/dest/testbench/worker_client_manager.js +231 -45
  380. package/dest/util.d.ts +3 -3
  381. package/dest/util.d.ts.map +1 -1
  382. package/package.json +16 -16
  383. package/src/bootstrap/bootstrap.ts +7 -4
  384. package/src/client/factory.ts +108 -37
  385. package/src/client/interface.ts +56 -34
  386. package/src/client/p2p_client.ts +207 -274
  387. package/src/client/test/tx_proposal_collector/README.md +227 -0
  388. package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker.ts +344 -0
  389. package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.ts +43 -0
  390. package/src/config.ts +172 -46
  391. package/src/errors/tx-pool.error.ts +12 -0
  392. package/src/index.ts +1 -0
  393. package/src/mem_pools/attestation_pool/attestation_pool.ts +497 -91
  394. package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +460 -112
  395. package/src/mem_pools/attestation_pool/index.ts +9 -2
  396. package/src/mem_pools/attestation_pool/mocks.ts +7 -4
  397. package/src/mem_pools/index.ts +4 -1
  398. package/src/mem_pools/instrumentation.ts +19 -14
  399. package/src/mem_pools/interface.ts +4 -4
  400. package/src/mem_pools/tx_pool/README.md +29 -14
  401. package/src/mem_pools/tx_pool/aztec_kv_tx_pool.ts +130 -75
  402. package/src/mem_pools/tx_pool/eviction/eviction_manager.ts +66 -5
  403. package/src/mem_pools/tx_pool/eviction/eviction_strategy.ts +119 -4
  404. package/src/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.ts +163 -0
  405. package/src/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.ts +3 -3
  406. package/src/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.ts +4 -2
  407. package/src/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.ts +75 -0
  408. package/src/mem_pools/tx_pool/priority.ts +4 -4
  409. package/src/mem_pools/tx_pool/tx_pool_test_suite.ts +3 -1
  410. package/src/mem_pools/tx_pool_v2/README.md +283 -0
  411. package/src/mem_pools/tx_pool_v2/archive/index.ts +1 -0
  412. package/src/mem_pools/tx_pool_v2/archive/tx_archive.ts +120 -0
  413. package/src/mem_pools/tx_pool_v2/deleted_pool.ts +321 -0
  414. package/src/mem_pools/tx_pool_v2/eviction/eviction_manager.ts +160 -0
  415. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.ts +122 -0
  416. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.ts +125 -0
  417. package/src/mem_pools/tx_pool_v2/eviction/index.ts +27 -0
  418. package/src/mem_pools/tx_pool_v2/eviction/interfaces.ts +219 -0
  419. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.ts +74 -0
  420. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.ts +101 -0
  421. package/src/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.ts +91 -0
  422. package/src/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.ts +99 -0
  423. package/src/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.ts +32 -0
  424. package/src/mem_pools/tx_pool_v2/index.ts +12 -0
  425. package/src/mem_pools/tx_pool_v2/instrumentation.ts +69 -0
  426. package/src/mem_pools/tx_pool_v2/interfaces.ts +247 -0
  427. package/src/mem_pools/tx_pool_v2/tx_metadata.ts +337 -0
  428. package/src/mem_pools/tx_pool_v2/tx_pool_bench_metrics.ts +77 -0
  429. package/src/mem_pools/tx_pool_v2/tx_pool_indices.ts +430 -0
  430. package/src/mem_pools/tx_pool_v2/tx_pool_v2.ts +238 -0
  431. package/src/mem_pools/tx_pool_v2/tx_pool_v2_impl.ts +1105 -0
  432. package/src/msg_validators/attestation_validator/README.md +49 -0
  433. package/src/msg_validators/attestation_validator/attestation_validator.ts +28 -15
  434. package/src/msg_validators/attestation_validator/fisherman_attestation_validator.ts +16 -10
  435. package/src/msg_validators/clock_tolerance.ts +52 -0
  436. package/src/msg_validators/proposal_validator/README.md +123 -0
  437. package/src/msg_validators/proposal_validator/block_proposal_validator.ts +14 -4
  438. package/src/msg_validators/proposal_validator/checkpoint_proposal_validator.ts +20 -7
  439. package/src/msg_validators/proposal_validator/proposal_validator.ts +89 -65
  440. package/src/msg_validators/tx_validator/README.md +119 -0
  441. package/src/msg_validators/tx_validator/aggregate_tx_validator.ts +5 -5
  442. package/src/msg_validators/tx_validator/allowed_public_setup.ts +22 -27
  443. package/src/msg_validators/tx_validator/allowed_setup_helpers.ts +31 -0
  444. package/src/msg_validators/tx_validator/archive_cache.ts +3 -3
  445. package/src/msg_validators/tx_validator/block_header_validator.ts +21 -8
  446. package/src/msg_validators/tx_validator/data_validator.ts +6 -2
  447. package/src/msg_validators/tx_validator/double_spend_validator.ts +15 -9
  448. package/src/msg_validators/tx_validator/factory.ts +406 -56
  449. package/src/msg_validators/tx_validator/fee_payer_balance.ts +44 -0
  450. package/src/msg_validators/tx_validator/gas_validator.ts +140 -55
  451. package/src/msg_validators/tx_validator/index.ts +3 -0
  452. package/src/msg_validators/tx_validator/metadata_validator.ts +18 -7
  453. package/src/msg_validators/tx_validator/nullifier_cache.ts +30 -0
  454. package/src/msg_validators/tx_validator/phases_validator.ts +86 -29
  455. package/src/msg_validators/tx_validator/size_validator.ts +22 -0
  456. package/src/msg_validators/tx_validator/timestamp_validator.ts +29 -21
  457. package/src/msg_validators/tx_validator/tx_permitted_validator.ts +8 -3
  458. package/src/msg_validators/tx_validator/tx_proof_validator.ts +8 -3
  459. package/src/services/data_store.ts +10 -7
  460. package/src/services/discv5/discV5_service.ts +1 -1
  461. package/src/services/dummy_service.ts +59 -2
  462. package/src/services/encoding.ts +20 -13
  463. package/src/services/gossipsub/README.md +641 -0
  464. package/src/services/gossipsub/index.ts +2 -0
  465. package/src/services/gossipsub/scoring.ts +29 -5
  466. package/src/services/gossipsub/topic_score_params.ts +487 -0
  467. package/src/services/index.ts +1 -0
  468. package/src/services/libp2p/instrumentation.ts +20 -7
  469. package/src/services/libp2p/libp2p_service.ts +541 -420
  470. package/src/services/peer-manager/metrics.ts +28 -4
  471. package/src/services/peer-manager/peer_manager.ts +5 -3
  472. package/src/services/peer-manager/peer_scoring.ts +29 -1
  473. package/src/services/reqresp/README.md +229 -0
  474. package/src/services/reqresp/batch-tx-requester/README.md +344 -0
  475. package/src/services/reqresp/batch-tx-requester/batch_tx_requester.ts +670 -0
  476. package/src/services/reqresp/batch-tx-requester/config.ts +40 -0
  477. package/src/services/reqresp/batch-tx-requester/interface.ts +54 -0
  478. package/src/services/reqresp/batch-tx-requester/missing_txs.ts +168 -0
  479. package/src/services/reqresp/batch-tx-requester/peer_collection.ts +249 -0
  480. package/src/services/reqresp/batch-tx-requester/tx_validator.ts +37 -0
  481. package/src/services/reqresp/connection-sampler/batch_connection_sampler.ts +65 -4
  482. package/src/services/reqresp/connection-sampler/connection_sampler.ts +16 -0
  483. package/src/services/reqresp/constants.ts +14 -0
  484. package/src/services/reqresp/interface.ts +29 -1
  485. package/src/services/reqresp/metrics.ts +34 -9
  486. package/src/services/reqresp/protocols/block_txs/bitvector.ts +16 -0
  487. package/src/services/reqresp/protocols/block_txs/block_txs_handler.ts +35 -12
  488. package/src/services/reqresp/protocols/block_txs/block_txs_reqresp.ts +74 -9
  489. package/src/services/reqresp/protocols/status.ts +5 -3
  490. package/src/services/reqresp/protocols/tx.ts +22 -0
  491. package/src/services/reqresp/reqresp.ts +99 -32
  492. package/src/services/service.ts +55 -2
  493. package/src/services/tx_collection/config.ts +83 -1
  494. package/src/services/tx_collection/fast_tx_collection.ts +111 -91
  495. package/src/services/tx_collection/file_store_tx_collection.ts +202 -0
  496. package/src/services/tx_collection/file_store_tx_source.ts +117 -0
  497. package/src/services/tx_collection/index.ts +6 -0
  498. package/src/services/tx_collection/instrumentation.ts +17 -2
  499. package/src/services/tx_collection/proposal_tx_collector.ts +108 -0
  500. package/src/services/tx_collection/request_tracker.ts +127 -0
  501. package/src/services/tx_collection/slow_tx_collection.ts +68 -35
  502. package/src/services/tx_collection/tx_collection.ts +122 -27
  503. package/src/services/tx_collection/tx_collection_sink.ts +30 -34
  504. package/src/services/tx_collection/tx_source.ts +22 -3
  505. package/src/services/tx_file_store/config.ts +37 -0
  506. package/src/services/tx_file_store/index.ts +3 -0
  507. package/src/services/tx_file_store/instrumentation.ts +36 -0
  508. package/src/services/tx_file_store/tx_file_store.ts +175 -0
  509. package/src/services/tx_provider.ts +12 -11
  510. package/src/services/tx_provider_instrumentation.ts +11 -5
  511. package/src/test-helpers/index.ts +2 -0
  512. package/src/test-helpers/make-test-p2p-clients.ts +4 -6
  513. package/src/test-helpers/mock-pubsub.ts +146 -9
  514. package/src/test-helpers/reqresp-nodes.ts +5 -7
  515. package/src/test-helpers/test_tx_provider.ts +64 -0
  516. package/src/test-helpers/testbench-utils.ts +456 -0
  517. package/src/testbench/p2p_client_testbench_worker.ts +334 -137
  518. package/src/testbench/worker_client_manager.ts +312 -47
  519. package/src/util.ts +8 -2
  520. package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts +0 -40
  521. package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts.map +0 -1
  522. package/dest/mem_pools/attestation_pool/kv_attestation_pool.js +0 -218
  523. package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts +0 -31
  524. package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts.map +0 -1
  525. package/dest/mem_pools/attestation_pool/memory_attestation_pool.js +0 -180
  526. package/dest/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.d.ts +0 -15
  527. package/dest/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.d.ts.map +0 -1
  528. package/dest/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.js +0 -88
  529. package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts +0 -23
  530. package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts.map +0 -1
  531. package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.js +0 -183
  532. package/src/mem_pools/attestation_pool/kv_attestation_pool.ts +0 -320
  533. package/src/mem_pools/attestation_pool/memory_attestation_pool.ts +0 -264
  534. package/src/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.ts +0 -108
  535. package/src/msg_validators/proposal_validator/proposal_validator_test_suite.ts +0 -206
@@ -0,0 +1,49 @@
1
+ # Attestation Validation
2
+
3
+ This module validates `CheckpointAttestation` gossipsub messages. Attestations are signatures from committee members endorsing a checkpoint proposal.
4
+
5
+ **Topic**: `checkpoint_attestation` | **Snappy size limit**: 5 KB
6
+
7
+ ## Stage 1: AttestationValidator (Gossipsub Validation)
8
+
9
+ | # | Rule | Consequence | Severity | File |
10
+ |---|------|-------------|----------|------|
11
+ | 1 | **Slot timeliness**: `currentSlot` or `nextSlot`. Previous slot within 500ms: IGNORE. Older: REJECT. | REJECT or IGNORE | HighToleranceError | `attestation_validator.ts` |
12
+ | 2 | **Attester signature**: `getSender()` must recover valid address | REJECT | LowToleranceError | same |
13
+ | 3 | **Attester in committee**: recovered address in committee for slot | REJECT | HighToleranceError | same |
14
+ | 4 | **Proposer exists**: `getProposerAttesterAddressInSlot` must return defined | REJECT | HighToleranceError | same |
15
+ | 5 | **Proposer signature**: `getProposer()` must recover valid address | REJECT | LowToleranceError | same |
16
+ | 6 | **Proposer matches expected**: recovered proposer = expected for slot | REJECT | HighToleranceError | same |
17
+ | 7 | **NoCommitteeError**: committee unavailable | REJECT | LowToleranceError | same |
18
+
19
+ **Fisherman mode extension** (`FishermanAttestationValidator`): if a checkpoint proposal for the same archive exists in pool, the attestation's `ConsensusPayload` must `.equals()` the stored proposal's payload. On mismatch: REJECT + LowToleranceError.
20
+
21
+ ## Stage 2: Pool Admission
22
+
23
+ | # | Rule | Consequence |
24
+ |---|------|-------------|
25
+ | 8 | Sender recoverable (pool-side) | Silent drop |
26
+ | 9 | Not a duplicate (same slot + proposalId + signer) | IGNORE |
27
+ | 10 | Per-signer cap: `MAX_ATTESTATIONS_PER_SLOT_AND_SIGNER` = 2 | IGNORE |
28
+
29
+ Own attestations added via `addOwnCheckpointAttestations` bypass the per-signer cap.
30
+
31
+ ## Stage 3: Equivocation Detection
32
+
33
+ When a signer's attestation count for a slot reaches exactly 2 (different proposals): `duplicateAttestationCallback` fires -> `WANT_TO_SLASH_EVENT` with `OffenseType.DUPLICATE_ATTESTATION`. Attestation still ACCEPTED and rebroadcast. Callback fires once (not again at count 3+).
34
+
35
+ ## Validation at L1 Checkpoint Submission (Archiver)
36
+
37
+ | Rule | Consequence | File |
38
+ |------|-------------|------|
39
+ | Each attestation must have recoverable signature (or address-only is allowed but does not count toward quorum) | Checkpoint rejected as invalid | `archiver/src/modules/validation.ts` |
40
+ | Attestation at index `i` must correspond to committee member at index `i` | Checkpoint rejected as invalid | same |
41
+ | Valid attestation count >= floor(committee * 2/3) + 1 | Checkpoint rejected as invalid | same |
42
+ | No committee / escape hatch open | Accepted unconditionally | same |
43
+
44
+ Note: `skipValidateCheckpointAttestations` config flag bypasses all archiver attestation validation.
45
+
46
+ ## Gossipsub Topic Scoring
47
+
48
+ P3 enabled with expected messages per slot = `targetCommitteeSize`. Conservative threshold (30% of convergence value). Max P3 penalty = -34 per topic.
49
+
@@ -1,7 +1,14 @@
1
1
  import type { EpochCacheInterface } from '@aztec/epoch-cache';
2
2
  import { NoCommitteeError } from '@aztec/ethereum/contracts';
3
3
  import { type Logger, createLogger } from '@aztec/foundation/log';
4
- import { type CheckpointAttestation, type P2PValidator, PeerErrorSeverity } from '@aztec/stdlib/p2p';
4
+ import {
5
+ type CheckpointAttestation,
6
+ type P2PValidator,
7
+ PeerErrorSeverity,
8
+ type ValidationResult,
9
+ } from '@aztec/stdlib/p2p';
10
+
11
+ import { isWithinClockTolerance } from '../clock_tolerance.js';
5
12
 
6
13
  export class CheckpointAttestationValidator implements P2PValidator<CheckpointAttestation> {
7
14
  protected epochCache: EpochCacheInterface;
@@ -12,30 +19,36 @@ export class CheckpointAttestationValidator implements P2PValidator<CheckpointAt
12
19
  this.logger = createLogger('p2p:checkpoint-attestation-validator');
13
20
  }
14
21
 
15
- async validate(message: CheckpointAttestation): Promise<PeerErrorSeverity | undefined> {
22
+ async validate(message: CheckpointAttestation): Promise<ValidationResult> {
16
23
  const slotNumber = message.payload.header.slotNumber;
17
24
 
18
25
  try {
19
- const { currentSlot, nextSlot } = await this.epochCache.getProposerAttesterAddressInCurrentOrNextSlot();
26
+ // Use target slots since proposals target pipeline slots (slot + 1 when pipelining)
27
+ const { targetSlot, nextSlot } = this.epochCache.getTargetAndNextSlot();
20
28
 
21
- if (slotNumber !== currentSlot && slotNumber !== nextSlot) {
22
- this.logger.warn(
23
- `Checkpoint attestation slot ${slotNumber} is not current (${currentSlot}) or next (${nextSlot}) slot`,
24
- );
25
- return PeerErrorSeverity.HighToleranceError;
29
+ if (slotNumber !== targetSlot && slotNumber !== nextSlot) {
30
+ // Check if message is for previous slot and within clock tolerance
31
+ if (!isWithinClockTolerance(slotNumber, targetSlot, this.epochCache)) {
32
+ this.logger.warn(
33
+ `Checkpoint attestation slot ${slotNumber} is not current (${targetSlot}) or next (${nextSlot}) slot`,
34
+ );
35
+ return { result: 'reject', severity: PeerErrorSeverity.HighToleranceError };
36
+ }
37
+ this.logger.debug(`Ignoring checkpoint attestation for previous slot ${slotNumber} within clock tolerance`);
38
+ return { result: 'ignore' };
26
39
  }
27
40
 
28
41
  // Verify the signature is valid
29
42
  const attester = message.getSender();
30
43
  if (attester === undefined) {
31
44
  this.logger.warn(`Invalid signature in checkpoint attestation for slot ${slotNumber}`);
32
- return PeerErrorSeverity.LowToleranceError;
45
+ return { result: 'reject', severity: PeerErrorSeverity.LowToleranceError };
33
46
  }
34
47
 
35
48
  // Verify the attester is in the committee for this slot
36
49
  if (!(await this.epochCache.isInCommittee(slotNumber, attester))) {
37
50
  this.logger.warn(`Attester ${attester.toString()} is not in committee for slot ${slotNumber}`);
38
- return PeerErrorSeverity.HighToleranceError;
51
+ return { result: 'reject', severity: PeerErrorSeverity.HighToleranceError };
39
52
  }
40
53
 
41
54
  // Verify the proposer signature matches the expected proposer for the attestation's slot
@@ -45,26 +58,26 @@ export class CheckpointAttestationValidator implements P2PValidator<CheckpointAt
45
58
  const expectedProposer = await this.epochCache.getProposerAttesterAddressInSlot(slotNumber);
46
59
  if (!expectedProposer) {
47
60
  this.logger.warn(`No proposer defined for slot ${slotNumber}`);
48
- return PeerErrorSeverity.HighToleranceError;
61
+ return { result: 'reject', severity: PeerErrorSeverity.HighToleranceError };
49
62
  }
50
63
  if (!proposer) {
51
64
  this.logger.warn(`Invalid proposer signature in checkpoint attestation for slot ${slotNumber}`);
52
- return PeerErrorSeverity.LowToleranceError;
65
+ return { result: 'reject', severity: PeerErrorSeverity.LowToleranceError };
53
66
  }
54
67
  if (!proposer.equals(expectedProposer)) {
55
68
  this.logger.warn(
56
69
  `Proposer signature mismatch in checkpoint attestation. ` +
57
70
  `Expected ${expectedProposer?.toString() ?? 'none'} but got ${proposer.toString()} for slot ${slotNumber}`,
58
71
  );
59
- return PeerErrorSeverity.HighToleranceError;
72
+ return { result: 'reject', severity: PeerErrorSeverity.HighToleranceError };
60
73
  }
61
74
 
62
- return undefined;
75
+ return { result: 'accept' };
63
76
  } catch (e) {
64
77
  // People shouldn't be sending us attestations if the committee doesn't exist
65
78
  if (e instanceof NoCommitteeError) {
66
79
  this.logger.warn(`No committee exists for checkpoint attestation for slot ${slotNumber}`);
67
- return PeerErrorSeverity.LowToleranceError;
80
+ return { result: 'reject', severity: PeerErrorSeverity.LowToleranceError };
68
81
  }
69
82
  throw e;
70
83
  }
@@ -1,8 +1,8 @@
1
1
  import type { EpochCacheInterface } from '@aztec/epoch-cache';
2
- import { type CheckpointAttestation, PeerErrorSeverity } from '@aztec/stdlib/p2p';
3
- import { Attributes, Metrics, type TelemetryClient } from '@aztec/telemetry-client';
2
+ import { type CheckpointAttestation, PeerErrorSeverity, type ValidationResult } from '@aztec/stdlib/p2p';
3
+ import { Attributes, Metrics, type TelemetryClient, createUpDownCounterWithDefault } from '@aztec/telemetry-client';
4
4
 
5
- import type { AttestationPool } from '../../mem_pools/attestation_pool/attestation_pool.js';
5
+ import type { AttestationPoolApi } from '../../mem_pools/attestation_pool/attestation_pool.js';
6
6
  import { CheckpointAttestationValidator } from './attestation_validator.js';
7
7
 
8
8
  /**
@@ -18,20 +18,26 @@ export class FishermanAttestationValidator extends CheckpointAttestationValidato
18
18
 
19
19
  constructor(
20
20
  epochCache: EpochCacheInterface,
21
- private attestationPool: AttestationPool,
21
+ private attestationPool: AttestationPoolApi,
22
22
  telemetryClient: TelemetryClient,
23
23
  ) {
24
24
  super(epochCache);
25
25
  this.logger = this.logger.createChild('[FISHERMAN]');
26
26
 
27
27
  const meter = telemetryClient.getMeter('FishermanAttestationValidator');
28
- this.invalidAttestationCounter = meter.createUpDownCounter(Metrics.VALIDATOR_INVALID_ATTESTATION_RECEIVED_COUNT);
28
+ this.invalidAttestationCounter = createUpDownCounterWithDefault(
29
+ meter,
30
+ Metrics.VALIDATOR_INVALID_ATTESTATION_RECEIVED_COUNT,
31
+ {
32
+ [Attributes.ERROR_TYPE]: ['base_validation_failed', 'payload_mismatch'],
33
+ },
34
+ );
29
35
  }
30
36
 
31
- override async validate(message: CheckpointAttestation): Promise<PeerErrorSeverity | undefined> {
37
+ override async validate(message: CheckpointAttestation): Promise<ValidationResult> {
32
38
  // First run the standard validation
33
39
  const baseValidationResult = await super.validate(message);
34
- if (baseValidationResult !== undefined) {
40
+ if (baseValidationResult.result !== 'accept') {
35
41
  // Track base validation failures (invalid signature, wrong committee, etc.)
36
42
  this.invalidAttestationCounter.add(1, {
37
43
  [Attributes.ERROR_TYPE]: 'base_validation_failed',
@@ -45,7 +51,7 @@ export class FishermanAttestationValidator extends CheckpointAttestationValidato
45
51
  const proposer = message.getProposer();
46
52
 
47
53
  if (!attester || !proposer) {
48
- return undefined;
54
+ return { result: 'accept' };
49
55
  }
50
56
 
51
57
  const proposalId = message.archive.toString();
@@ -74,7 +80,7 @@ export class FishermanAttestationValidator extends CheckpointAttestationValidato
74
80
  });
75
81
 
76
82
  // Return error to reject the message, but LibP2PService won't penalize in fisherman mode
77
- return PeerErrorSeverity.LowToleranceError;
83
+ return { result: 'reject', severity: PeerErrorSeverity.LowToleranceError };
78
84
  }
79
85
  } else {
80
86
  // We might receive attestations before proposals in some cases
@@ -83,6 +89,6 @@ export class FishermanAttestationValidator extends CheckpointAttestationValidato
83
89
  );
84
90
  }
85
91
 
86
- return undefined;
92
+ return { result: 'accept' };
87
93
  }
88
94
  }
@@ -0,0 +1,52 @@
1
+ import type { EpochCacheInterface } from '@aztec/epoch-cache';
2
+ import { SlotNumber } from '@aztec/foundation/branded-types';
3
+
4
+ /**
5
+ * Maximum clock disparity tolerance for P2P message validation (in milliseconds).
6
+ * Messages for the previous slot are accepted if we're within this many milliseconds
7
+ * of the current slot start. This prevents penalizing peers for messages that
8
+ * were valid when sent but arrived slightly late due to network latency.
9
+ *
10
+ * This follows Ethereum's MAXIMUM_GOSSIP_CLOCK_DISPARITY approach.
11
+ */
12
+ export const MAXIMUM_GOSSIP_CLOCK_DISPARITY_MS = 500;
13
+
14
+ /**
15
+ * Checks if a message for the previous slot should be accepted due to clock tolerance.
16
+ *
17
+ * @param messageSlot - The slot number from the received message
18
+ * @param currentSlot - The current slot number
19
+ * @param epochCache - EpochCache to get timing information
20
+ * @returns true if the message is for the previous slot AND we're within the clock tolerance window
21
+ */
22
+ export function isWithinClockTolerance(
23
+ messageSlot: SlotNumber,
24
+ currentSlot: SlotNumber,
25
+ epochCache: EpochCacheInterface,
26
+ ): boolean {
27
+ // Guard against slot 0 edge case (genesis)
28
+ if (currentSlot === SlotNumber.ZERO) {
29
+ return false;
30
+ }
31
+
32
+ // Only apply tolerance to messages for the previous slot
33
+ const previousSlot = SlotNumber(currentSlot - 1);
34
+ if (messageSlot !== previousSlot) {
35
+ return false;
36
+ }
37
+
38
+ // Check how far we are into the current slot (in milliseconds)
39
+ const { ts: slotStartTs, nowMs } = epochCache.getEpochAndSlotNow();
40
+ const targetSlot = epochCache.getTargetSlot();
41
+
42
+ // Sanity check: ensure the epoch cache's target slot matches the expected current slot
43
+ if (targetSlot !== currentSlot) {
44
+ return false;
45
+ }
46
+
47
+ // ts is in seconds, convert to ms; nowMs is already in milliseconds
48
+ const slotStartMs = slotStartTs * 1000n;
49
+ const elapsedMs = Number(nowMs - slotStartMs);
50
+
51
+ return elapsedMs < MAXIMUM_GOSSIP_CLOCK_DISPARITY_MS;
52
+ }
@@ -0,0 +1,123 @@
1
+ # Proposal Validation
2
+
3
+ This module validates `BlockProposal` and `CheckpointProposal` gossipsub messages. Both share the same base `ProposalValidator` (neither subclass overrides `validate()`), with checkpoint-specific logic layered on top in the gossipsub handler.
4
+
5
+ ## BlockProposal
6
+
7
+ **Topic**: `block_proposal` | **Snappy size limit**: 10 MB
8
+
9
+ ### Stage 1: Gossipsub Validation (ProposalValidator)
10
+
11
+ File: `proposal_validator.ts`
12
+
13
+ | # | Rule | Consequence | Severity |
14
+ |---|------|-------------|----------|
15
+ | 1 | **Slot check**: must be `currentSlot` or `nextSlot`. Previous slot within 500ms tolerance: IGNORE. | REJECT | HighToleranceError |
16
+ | 2 | **Signature**: `getSender()` must recover a valid address. If `signedTxs` present, its recovered sender must match. | REJECT | MidToleranceError |
17
+ | 3 | **Txs permitted**: if `disableTransactions`, must have 0 txHashes and 0 embedded txs | REJECT | MidToleranceError |
18
+ | 4 | **Max txs**: `txHashes.length <= maxTxsPerBlock` | REJECT | MidToleranceError |
19
+ | 5 | **Embedded txs in txHashes**: every embedded tx's hash must appear in `txHashes` | REJECT | MidToleranceError |
20
+ | 6 | **Proposer check**: signer must match expected proposer for slot (skipped if committee size = 0) | REJECT | MidToleranceError |
21
+ | 7 | **Tx hash integrity**: each embedded tx's recomputed hash must match declared hash | REJECT | LowToleranceError |
22
+ | 8 | **NoCommitteeError**: epoch cache cannot determine committee | REJECT | LowToleranceError |
23
+
24
+ Deserialization guards: `BlockProposal.fromBuffer` and `SignedTxs.fromBuffer` both enforce `txCount <= MAX_TXS_PER_BLOCK` (65536). Violation -> REJECT + LowToleranceError.
25
+
26
+ ### Stage 2: Mempool (Attestation Pool)
27
+
28
+ | # | Rule | Consequence |
29
+ |---|------|-------------|
30
+ | 9 | **Duplicate**: same archive root already stored | IGNORE (no penalty) |
31
+ | 10 | **Per-position cap**: max 2 proposals per (slot, indexWithinCheckpoint) | REJECT + HighToleranceError |
32
+ | 11 | **Equivocation**: >1 distinct proposal for same (slot, index) | ACCEPT (rebroadcast for detection). At count=2: `duplicateProposalCallback` fires -> slash event (`OffenseType.DUPLICATE_PROPOSAL`, configured via `slashDuplicateProposalPenalty`) |
33
+
34
+ ### Stage 3: Validator-Client Processing (BlockProposalHandler)
35
+
36
+ Only runs on validator nodes. Non-validator nodes use a default handler that triggers tx collection without deep validation.
37
+
38
+ | # | Rule | Failure Reason |
39
+ |---|------|----------------|
40
+ | 12 | Signature re-check | `invalid_proposal` |
41
+ | 13 | ProposalValidator re-run | `invalid_proposal` |
42
+ | 14 | Self-proposal filter | Ignored silently |
43
+ | 15 | Parent block exists (`lastArchive.root` matches known block or genesis) | `parent_block_not_found` |
44
+ | 16 | Parent block slot <= proposal slot | `parent_block_wrong_slot` |
45
+ | 17 | Block number not already in archiver | `block_number_already_exists` |
46
+ | 18 | Checkpoint number consistency (multiple sub-rules for first/non-first blocks) | `invalid_proposal` |
47
+ | 19 | Global variables consistency (non-first block: chainId, version, slot, timestamp, coinbase, feeRecipient, gasFees match parent) | `global_variables_mismatch` |
48
+ | 20 | L1-to-L2 message hash matches `proposal.inHash` | `in_hash_mismatch` |
49
+ | 21 | All txs referenced by `txHashes` obtainable | `txs_not_available` |
50
+ | 22 | **Re-execution**: processed tx count matches `txHashes.length` | `timeout` (ReExTimeoutError) |
51
+ | 23 | **Re-execution**: no failed txs | `failed_txs` (ReExFailedTxsError) -- **SLASHABLE** |
52
+ | 24 | **Re-execution**: archive root and header match proposal | `state_mismatch` (ReExStateMismatchError) -- **SLASHABLE** |
53
+
54
+ **Escape hatch**: during escape hatch periods (`isEscapeHatchOpenAtSlot`), re-execution and slashing are both disabled, and the proposal is rejected locally.
55
+
56
+ **Conditional re-execution**: rules 22-24 only run when at least one condition is true: `fishermanMode` enabled, `slashBroadcastedInvalidBlockPenalty > 0` with `validatorReexecute`, committee membership with `validatorReexecute`, `alwaysReexecuteBlockProposals`, or `blobClient.canUpload()`.
57
+
58
+ **Slashing**: only `state_mismatch` and `failed_txs` trigger on-chain slashing (`OffenseType.BROADCASTED_INVALID_BLOCK_PROPOSAL`, gated by `slashBroadcastedInvalidBlockPenalty > 0`). Unknown errors during re-execution do NOT slash.
59
+
60
+ **Embedded tx validation**: txs in `signedTxs` are validated via `createTxValidatorForBlockProposalReceivedTxs` (well-formedness only) when stored in the tx pool. Invalid embedded txs are rejected from the pool but do not cause the block proposal itself to be rejected at gossipsub level.
61
+
62
+ ### Gossipsub Topic Scoring
63
+
64
+ | Parameter | Effect |
65
+ |-----------|--------|
66
+ | P4 (invalidMessageDeliveries) | weight = -20, decay over 4 slots |
67
+ | P3 (meshMessageDeliveries) | Enabled only when `expectedBlockProposalsPerSlot > 0` (MBPS mode) |
68
+ | P1/P2 | Only active when P3 is enabled |
69
+
70
+ ---
71
+
72
+ ## CheckpointProposal
73
+
74
+ **Topic**: `checkpoint_proposal` | **Snappy size limit**: 10 MB
75
+
76
+ ### Stage 1: Gossipsub Validation (ProposalValidator)
77
+
78
+ Same `ProposalValidator.validate()` as BlockProposal (shared implementation, neither subclass overrides it). See BlockProposal Stage 1 rules 1-8.
79
+
80
+ ### Stage 2: Embedded Block Proposal Validation (if `lastBlock` present)
81
+
82
+ The checkpoint's embedded `lastBlock` is extracted via `getBlockProposal()` and validated through `BlockProposalValidator.validate()` plus block mempool checks.
83
+
84
+ | Rule | Consequence | File |
85
+ |------|-------------|------|
86
+ | Block proposal must pass `BlockProposalValidator.validate()` | If REJECT: entire checkpoint REJECTED | `libp2p_service.ts` |
87
+ | Block proposal must not exceed per-position cap (2) | Checkpoint REJECTED + HighToleranceError | same |
88
+ | Block equivocation detected (>1 proposals for same slot+index) | Checkpoint REJECTED (block itself is ACCEPT for re-broadcast) | same |
89
+
90
+ ### Stage 3: Mempool (Attestation Pool)
91
+
92
+ | Rule | Consequence | File |
93
+ |------|-------------|------|
94
+ | Duplicate (same archive ID) | IGNORE (no penalty). Embedded block still processed if valid. | `attestation_pool.ts` |
95
+ | Per-slot cap: `MAX_CHECKPOINT_PROPOSALS_PER_SLOT` = 2 | REJECT + HighToleranceError. Embedded block still processed. | same |
96
+
97
+ ### Stage 4: Equivocation Detection
98
+
99
+ When >1 checkpoint proposals exist for same slot (count > 1): ACCEPT (re-broadcast). At count == 2 (exactly): `duplicateProposalCallback` fires. Proposal NOT further processed. Callback fires only once per equivocation pair.
100
+
101
+ ### Stage 5: Validator-Client Consensus Validation
102
+
103
+ Determines whether the validator signs an attestation.
104
+
105
+ | Rule | Consequence | File |
106
+ |------|-------------|------|
107
+ | Escape hatch open | No attestation | `validator-client/src/validator.ts` |
108
+ | Signature invalid (re-check) | No attestation | same |
109
+ | Self-proposal | No attestation (ignored) | same |
110
+ | `feeAssetPriceModifier` outside [-100, +100] bps | No attestation | same |
111
+ | Not in committee (unless fisherman mode) | No attestation | same |
112
+ | Checkpoint header mismatch (computed vs proposal) | No attestation | same |
113
+ | Archive root mismatch | No attestation | same |
114
+ | Epoch out hash mismatch | No attestation | same |
115
+ | Last block not found / not matching | No attestation | same |
116
+ | Already attested to this or earlier slot | No attestation (unless `attestToEquivocatedProposals`) | same |
117
+
118
+ **`skipCheckpointProposalValidation` config**: when true, the re-execution checks (header/archive/epoch hash) are all skipped. Signature, fee modifier, committee, escape hatch, and equivocation checks still apply.
119
+
120
+ ### Gossipsub Topic Scoring
121
+
122
+ P3 enabled with expected rate of 1 message per slot. P4 weight = -20, max P3 penalty = -34 per topic.
123
+
@@ -1,10 +1,20 @@
1
1
  import type { EpochCacheInterface } from '@aztec/epoch-cache';
2
- import type { BlockProposal, P2PValidator } from '@aztec/stdlib/p2p';
2
+ import type { BlockProposal, P2PValidator, ValidationResult } from '@aztec/stdlib/p2p';
3
3
 
4
4
  import { ProposalValidator } from '../proposal_validator/proposal_validator.js';
5
5
 
6
- export class BlockProposalValidator extends ProposalValidator<BlockProposal> implements P2PValidator<BlockProposal> {
7
- constructor(epochCache: EpochCacheInterface, opts: { txsPermitted: boolean }) {
8
- super(epochCache, opts, 'p2p:block_proposal_validator');
6
+ export class BlockProposalValidator implements P2PValidator<BlockProposal> {
7
+ private proposalValidator: ProposalValidator;
8
+
9
+ constructor(epochCache: EpochCacheInterface, opts: { txsPermitted: boolean; maxTxsPerBlock?: number }) {
10
+ this.proposalValidator = new ProposalValidator(epochCache, opts, 'p2p:block_proposal_validator');
11
+ }
12
+
13
+ async validate(proposal: BlockProposal): Promise<ValidationResult> {
14
+ const headerResult = await this.proposalValidator.validate(proposal);
15
+ if (headerResult.result !== 'accept') {
16
+ return headerResult;
17
+ }
18
+ return this.proposalValidator.validateTxs(proposal);
9
19
  }
10
20
  }
@@ -1,13 +1,26 @@
1
1
  import type { EpochCacheInterface } from '@aztec/epoch-cache';
2
- import type { CheckpointProposal, P2PValidator } from '@aztec/stdlib/p2p';
2
+ import type { CheckpointProposal, P2PValidator, ValidationResult } from '@aztec/stdlib/p2p';
3
3
 
4
4
  import { ProposalValidator } from '../proposal_validator/proposal_validator.js';
5
5
 
6
- export class CheckpointProposalValidator
7
- extends ProposalValidator<CheckpointProposal>
8
- implements P2PValidator<CheckpointProposal>
9
- {
10
- constructor(epochCache: EpochCacheInterface, opts: { txsPermitted: boolean }) {
11
- super(epochCache, opts, 'p2p:checkpoint_proposal_validator');
6
+ export class CheckpointProposalValidator implements P2PValidator<CheckpointProposal> {
7
+ private proposalValidator: ProposalValidator;
8
+
9
+ constructor(epochCache: EpochCacheInterface, opts: { txsPermitted: boolean; maxTxsPerBlock?: number }) {
10
+ this.proposalValidator = new ProposalValidator(epochCache, opts, 'p2p:checkpoint_proposal_validator');
11
+ }
12
+
13
+ async validate(proposal: CheckpointProposal): Promise<ValidationResult> {
14
+ const headerResult = await this.proposalValidator.validate(proposal);
15
+ if (headerResult.result !== 'accept') {
16
+ return headerResult;
17
+ }
18
+
19
+ const blockProposal = proposal.getBlockProposal();
20
+ if (blockProposal) {
21
+ return this.proposalValidator.validateTxs(blockProposal);
22
+ }
23
+
24
+ return { result: 'accept' };
12
25
  }
13
26
  }
@@ -1,92 +1,116 @@
1
1
  import type { EpochCacheInterface } from '@aztec/epoch-cache';
2
2
  import { NoCommitteeError } from '@aztec/ethereum/contracts';
3
3
  import { type Logger, createLogger } from '@aztec/foundation/log';
4
- import { BlockProposal, CheckpointProposal, PeerErrorSeverity } from '@aztec/stdlib/p2p';
4
+ import {
5
+ type BlockProposal,
6
+ type CheckpointProposalCore,
7
+ PeerErrorSeverity,
8
+ type ValidationResult,
9
+ } from '@aztec/stdlib/p2p';
5
10
 
6
- export abstract class ProposalValidator<TProposal extends BlockProposal | CheckpointProposal> {
7
- protected epochCache: EpochCacheInterface;
8
- protected logger: Logger;
9
- protected txsPermitted: boolean;
11
+ import { isWithinClockTolerance } from '../clock_tolerance.js';
10
12
 
11
- constructor(epochCache: EpochCacheInterface, opts: { txsPermitted: boolean }, loggerName: string) {
13
+ /** Validates header-level and tx-level fields of block and checkpoint proposals. */
14
+ export class ProposalValidator {
15
+ private epochCache: EpochCacheInterface;
16
+ private logger: Logger;
17
+ private txsPermitted: boolean;
18
+ private maxTxsPerBlock?: number;
19
+
20
+ constructor(
21
+ epochCache: EpochCacheInterface,
22
+ opts: { txsPermitted: boolean; maxTxsPerBlock?: number },
23
+ loggerName: string,
24
+ ) {
12
25
  this.epochCache = epochCache;
13
26
  this.txsPermitted = opts.txsPermitted;
27
+ this.maxTxsPerBlock = opts.maxTxsPerBlock;
14
28
  this.logger = createLogger(loggerName);
15
29
  }
16
30
 
17
- public async validate(proposal: TProposal): Promise<PeerErrorSeverity | undefined> {
31
+ /** Validates header-level fields: slot, signature, and proposer. */
32
+ public async validate(proposal: BlockProposal | CheckpointProposalCore): Promise<ValidationResult> {
18
33
  try {
19
- // Signature validity
20
- const proposer = proposal.getSender();
21
- if (!proposer) {
22
- this.logger.debug(`Penalizing peer for proposal with invalid signature`);
23
- return PeerErrorSeverity.MidToleranceError;
24
- }
34
+ // Slot check: use target slots since proposals target pipeline slots (slot + 1 when pipelining)
35
+ const { targetSlot, nextSlot } = this.epochCache.getTargetAndNextSlot();
25
36
 
26
- // Transactions permitted check
27
- const embeddedTxCount = proposal.txs?.length ?? 0;
28
- if (!this.txsPermitted && (proposal.txHashes.length > 0 || embeddedTxCount > 0)) {
29
- this.logger.debug(
30
- `Penalizing peer for proposal with ${proposal.txHashes.length} transaction(s) when transactions are not permitted`,
31
- );
32
- return PeerErrorSeverity.MidToleranceError;
37
+ const slotNumber = proposal.slotNumber;
38
+ if (slotNumber !== targetSlot && slotNumber !== nextSlot) {
39
+ // Check if message is for previous slot and within clock tolerance
40
+ if (!isWithinClockTolerance(slotNumber, targetSlot, this.epochCache)) {
41
+ this.logger.warn(`Penalizing peer for invalid slot number ${slotNumber}`, { targetSlot, nextSlot });
42
+ return { result: 'reject', severity: PeerErrorSeverity.HighToleranceError };
43
+ }
44
+ this.logger.verbose(`Ignoring proposal for previous slot ${slotNumber} within clock tolerance`);
45
+ return { result: 'ignore' };
33
46
  }
34
47
 
35
- // Embedded txs must be listed in txHashes
36
- const hashSet = new Set(proposal.txHashes.map(h => h.toString()));
37
- const missingTxHashes =
38
- embeddedTxCount > 0
39
- ? proposal.txs!.filter(tx => !hashSet.has(tx.getTxHash().toString())).map(tx => tx.getTxHash().toString())
40
- : [];
41
- if (embeddedTxCount > 0 && missingTxHashes.length > 0) {
42
- this.logger.warn('Penalizing peer for embedded transaction(s) not included in txHashes', {
43
- embeddedTxCount,
44
- txHashesLength: proposal.txHashes.length,
45
- missingTxHashes,
46
- });
47
- return PeerErrorSeverity.MidToleranceError;
48
+ // Signature validity
49
+ const proposer = proposal.getSender();
50
+ if (!proposer) {
51
+ this.logger.warn(`Penalizing peer for proposal with invalid signature`);
52
+ return { result: 'reject', severity: PeerErrorSeverity.MidToleranceError };
48
53
  }
49
54
 
50
- // Slot and proposer checks
51
- const { currentProposer, nextProposer, currentSlot, nextSlot } =
52
- await this.epochCache.getProposerAttesterAddressInCurrentOrNextSlot();
53
- const slotNumber = proposal.slotNumber;
54
- if (slotNumber !== currentSlot && slotNumber !== nextSlot) {
55
- this.logger.debug(`Penalizing peer for invalid slot number ${slotNumber}`, { currentSlot, nextSlot });
56
- return PeerErrorSeverity.HighToleranceError;
57
- }
58
- if (slotNumber === currentSlot && currentProposer !== undefined && !proposer.equals(currentProposer)) {
59
- this.logger.debug(`Penalizing peer for invalid proposer for current slot ${slotNumber}`, {
60
- currentProposer,
61
- nextProposer,
62
- proposer: proposer.toString(),
63
- });
64
- return PeerErrorSeverity.MidToleranceError;
65
- }
66
- if (slotNumber === nextSlot && nextProposer !== undefined && !proposer.equals(nextProposer)) {
67
- this.logger.debug(`Penalizing peer for invalid proposer for next slot ${slotNumber}`, {
68
- currentProposer,
69
- nextProposer,
55
+ // Proposer check
56
+ const expectedProposer = await this.epochCache.getProposerAttesterAddressInSlot(slotNumber);
57
+ if (expectedProposer !== undefined && !proposer.equals(expectedProposer)) {
58
+ this.logger.warn(`Penalizing peer for invalid proposer for current slot ${slotNumber}`, {
59
+ expectedProposer,
70
60
  proposer: proposer.toString(),
71
61
  });
72
- return PeerErrorSeverity.MidToleranceError;
73
- }
74
-
75
- // Validate tx hashes for all txs embedded in the proposal
76
- if (!(await Promise.all(proposal.txs?.map(tx => tx.validateTxHash()) ?? [])).every(v => v)) {
77
- this.logger.warn(`Penalizing peer for invalid tx hashes in proposal`, {
78
- proposer,
79
- slotNumber,
80
- });
81
- return PeerErrorSeverity.LowToleranceError;
62
+ return { result: 'reject', severity: PeerErrorSeverity.MidToleranceError };
82
63
  }
83
64
 
84
- return undefined;
65
+ return { result: 'accept' };
85
66
  } catch (e) {
86
67
  if (e instanceof NoCommitteeError) {
87
- return PeerErrorSeverity.LowToleranceError;
68
+ return { result: 'reject', severity: PeerErrorSeverity.LowToleranceError };
88
69
  }
89
70
  throw e;
90
71
  }
91
72
  }
73
+
74
+ /** Validates transaction-related fields of a block proposal. */
75
+ public async validateTxs(proposal: BlockProposal): Promise<ValidationResult> {
76
+ // Transactions permitted check
77
+ const embeddedTxCount = proposal.txs?.length ?? 0;
78
+ if (!this.txsPermitted && (proposal.txHashes.length > 0 || embeddedTxCount > 0)) {
79
+ this.logger.warn(
80
+ `Penalizing peer for proposal with ${proposal.txHashes.length} transaction(s) when transactions are not permitted`,
81
+ );
82
+ return { result: 'reject', severity: PeerErrorSeverity.MidToleranceError };
83
+ }
84
+
85
+ // Max txs per block check
86
+ if (this.maxTxsPerBlock !== undefined && proposal.txHashes.length > this.maxTxsPerBlock) {
87
+ this.logger.warn(
88
+ `Penalizing peer for proposal with ${proposal.txHashes.length} transaction(s) when max is ${this.maxTxsPerBlock}`,
89
+ );
90
+ return { result: 'reject', severity: PeerErrorSeverity.MidToleranceError };
91
+ }
92
+
93
+ // Embedded txs must be listed in txHashes
94
+ const hashSet = new Set(proposal.txHashes.map(h => h.toString()));
95
+ const missingTxHashes =
96
+ embeddedTxCount > 0
97
+ ? proposal.txs!.filter(tx => !hashSet.has(tx.getTxHash().toString())).map(tx => tx.getTxHash().toString())
98
+ : [];
99
+ if (embeddedTxCount > 0 && missingTxHashes.length > 0) {
100
+ this.logger.warn('Penalizing peer for embedded transaction(s) not included in txHashes', {
101
+ embeddedTxCount,
102
+ txHashesLength: proposal.txHashes.length,
103
+ missingTxHashes,
104
+ });
105
+ return { result: 'reject', severity: PeerErrorSeverity.MidToleranceError };
106
+ }
107
+
108
+ // Validate tx hashes for all txs embedded in the proposal
109
+ if (!(await Promise.all(proposal.txs?.map(tx => tx.validateTxHash()) ?? [])).every(v => v)) {
110
+ this.logger.warn(`Penalizing peer for invalid tx hashes in proposal`);
111
+ return { result: 'reject', severity: PeerErrorSeverity.LowToleranceError };
112
+ }
113
+
114
+ return { result: 'accept' };
115
+ }
92
116
  }