@aztec/p2p 0.0.1-commit.c7c42ec → 0.0.1-commit.c949de6bc

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 (531) hide show
  1. package/dest/bootstrap/bootstrap.d.ts +4 -3
  2. package/dest/bootstrap/bootstrap.d.ts.map +1 -1
  3. package/dest/bootstrap/bootstrap.js +4 -4
  4. package/dest/client/factory.d.ts +10 -10
  5. package/dest/client/factory.d.ts.map +1 -1
  6. package/dest/client/factory.js +45 -18
  7. package/dest/client/interface.d.ts +61 -35
  8. package/dest/client/interface.d.ts.map +1 -1
  9. package/dest/client/p2p_client.d.ts +47 -60
  10. package/dest/client/p2p_client.d.ts.map +1 -1
  11. package/dest/client/p2p_client.js +600 -313
  12. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.d.ts +2 -0
  13. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.d.ts.map +1 -0
  14. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.js +304 -0
  15. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.d.ts +73 -0
  16. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.d.ts.map +1 -0
  17. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.js +8 -0
  18. package/dest/config.d.ts +35 -7
  19. package/dest/config.d.ts.map +1 -1
  20. package/dest/config.js +23 -9
  21. package/dest/errors/tx-pool.error.d.ts +8 -0
  22. package/dest/errors/tx-pool.error.d.ts.map +1 -0
  23. package/dest/errors/tx-pool.error.js +9 -0
  24. package/dest/index.d.ts +2 -1
  25. package/dest/index.d.ts.map +1 -1
  26. package/dest/index.js +1 -0
  27. package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +111 -76
  28. package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -1
  29. package/dest/mem_pools/attestation_pool/attestation_pool.js +441 -3
  30. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.d.ts +2 -2
  31. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.d.ts.map +1 -1
  32. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.js +527 -287
  33. package/dest/mem_pools/attestation_pool/index.d.ts +2 -3
  34. package/dest/mem_pools/attestation_pool/index.d.ts.map +1 -1
  35. package/dest/mem_pools/attestation_pool/index.js +1 -2
  36. package/dest/mem_pools/attestation_pool/mocks.d.ts +9 -6
  37. package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -1
  38. package/dest/mem_pools/attestation_pool/mocks.js +16 -12
  39. package/dest/mem_pools/index.d.ts +3 -2
  40. package/dest/mem_pools/index.d.ts.map +1 -1
  41. package/dest/mem_pools/index.js +1 -1
  42. package/dest/mem_pools/instrumentation.d.ts +1 -1
  43. package/dest/mem_pools/instrumentation.d.ts.map +1 -1
  44. package/dest/mem_pools/instrumentation.js +5 -14
  45. package/dest/mem_pools/interface.d.ts +5 -5
  46. package/dest/mem_pools/interface.d.ts.map +1 -1
  47. package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts +15 -10
  48. package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts.map +1 -1
  49. package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.js +91 -50
  50. package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts +19 -5
  51. package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts.map +1 -1
  52. package/dest/mem_pools/tx_pool/eviction/eviction_manager.js +59 -3
  53. package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts +79 -5
  54. package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts.map +1 -1
  55. package/dest/mem_pools/tx_pool/eviction/eviction_strategy.js +47 -0
  56. package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts +16 -0
  57. package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -0
  58. package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.js +122 -0
  59. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts +2 -2
  60. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts.map +1 -1
  61. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.js +3 -3
  62. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts +4 -4
  63. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts.map +1 -1
  64. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.js +2 -0
  65. package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts +2 -2
  66. package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts.map +1 -1
  67. package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.d.ts +25 -0
  68. package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.d.ts.map +1 -0
  69. package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.js +57 -0
  70. package/dest/mem_pools/tx_pool_v2/archive/index.d.ts +2 -0
  71. package/dest/mem_pools/tx_pool_v2/archive/index.d.ts.map +1 -0
  72. package/dest/mem_pools/tx_pool_v2/archive/index.js +1 -0
  73. package/dest/mem_pools/tx_pool_v2/archive/tx_archive.d.ts +43 -0
  74. package/dest/mem_pools/tx_pool_v2/archive/tx_archive.d.ts.map +1 -0
  75. package/dest/mem_pools/tx_pool_v2/archive/tx_archive.js +103 -0
  76. package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts +104 -0
  77. package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts.map +1 -0
  78. package/dest/mem_pools/tx_pool_v2/deleted_pool.js +251 -0
  79. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts +47 -0
  80. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts.map +1 -0
  81. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.js +128 -0
  82. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts +17 -0
  83. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -0
  84. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.js +93 -0
  85. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts +19 -0
  86. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts.map +1 -0
  87. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.js +97 -0
  88. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts +10 -0
  89. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts.map +1 -0
  90. package/dest/mem_pools/tx_pool_v2/eviction/index.js +11 -0
  91. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts +174 -0
  92. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts.map +1 -0
  93. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.js +25 -0
  94. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.d.ts +15 -0
  95. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.d.ts.map +1 -0
  96. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.js +65 -0
  97. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.d.ts +17 -0
  98. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.d.ts.map +1 -0
  99. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.js +93 -0
  100. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts +16 -0
  101. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts.map +1 -0
  102. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.js +78 -0
  103. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts +20 -0
  104. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts.map +1 -0
  105. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.js +73 -0
  106. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts +15 -0
  107. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts.map +1 -0
  108. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.js +19 -0
  109. package/dest/mem_pools/tx_pool_v2/index.d.ts +6 -0
  110. package/dest/mem_pools/tx_pool_v2/index.d.ts.map +1 -0
  111. package/dest/mem_pools/tx_pool_v2/index.js +5 -0
  112. package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts +15 -0
  113. package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts.map +1 -0
  114. package/dest/mem_pools/tx_pool_v2/instrumentation.js +43 -0
  115. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts +211 -0
  116. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts.map +1 -0
  117. package/dest/mem_pools/tx_pool_v2/interfaces.js +9 -0
  118. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts +119 -0
  119. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts.map +1 -0
  120. package/dest/mem_pools/tx_pool_v2/tx_metadata.js +193 -0
  121. package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.d.ts +26 -0
  122. package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.d.ts.map +1 -0
  123. package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.js +70 -0
  124. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts +108 -0
  125. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts.map +1 -0
  126. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.js +354 -0
  127. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts +60 -0
  128. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts.map +1 -0
  129. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.js +161 -0
  130. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts +77 -0
  131. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts.map +1 -0
  132. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.js +905 -0
  133. package/dest/msg_validators/attestation_validator/attestation_validator.d.ts +4 -4
  134. package/dest/msg_validators/attestation_validator/attestation_validator.d.ts.map +1 -1
  135. package/dest/msg_validators/attestation_validator/attestation_validator.js +51 -18
  136. package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts +7 -7
  137. package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts.map +1 -1
  138. package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.js +22 -13
  139. package/dest/msg_validators/clock_tolerance.d.ts +21 -0
  140. package/dest/msg_validators/clock_tolerance.d.ts.map +1 -0
  141. package/dest/msg_validators/clock_tolerance.js +37 -0
  142. package/dest/msg_validators/index.d.ts +2 -2
  143. package/dest/msg_validators/index.d.ts.map +1 -1
  144. package/dest/msg_validators/index.js +1 -1
  145. package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts +9 -0
  146. package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts.map +1 -0
  147. package/dest/msg_validators/proposal_validator/block_proposal_validator.js +6 -0
  148. package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts +9 -0
  149. package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts.map +1 -0
  150. package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.js +6 -0
  151. package/dest/msg_validators/proposal_validator/index.d.ts +4 -0
  152. package/dest/msg_validators/proposal_validator/index.d.ts.map +1 -0
  153. package/dest/msg_validators/proposal_validator/index.js +3 -0
  154. package/dest/msg_validators/proposal_validator/proposal_validator.d.ts +13 -0
  155. package/dest/msg_validators/proposal_validator/proposal_validator.d.ts.map +1 -0
  156. package/dest/msg_validators/proposal_validator/proposal_validator.js +104 -0
  157. package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts +23 -0
  158. package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts.map +1 -0
  159. package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.js +212 -0
  160. package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts +4 -4
  161. package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts.map +1 -1
  162. package/dest/msg_validators/tx_validator/aggregate_tx_validator.js +3 -3
  163. package/dest/msg_validators/tx_validator/archive_cache.d.ts +3 -3
  164. package/dest/msg_validators/tx_validator/archive_cache.d.ts.map +1 -1
  165. package/dest/msg_validators/tx_validator/archive_cache.js +1 -1
  166. package/dest/msg_validators/tx_validator/block_header_validator.d.ts +20 -6
  167. package/dest/msg_validators/tx_validator/block_header_validator.d.ts.map +1 -1
  168. package/dest/msg_validators/tx_validator/block_header_validator.js +4 -3
  169. package/dest/msg_validators/tx_validator/data_validator.d.ts +3 -1
  170. package/dest/msg_validators/tx_validator/data_validator.d.ts.map +1 -1
  171. package/dest/msg_validators/tx_validator/data_validator.js +4 -1
  172. package/dest/msg_validators/tx_validator/double_spend_validator.d.ts +15 -4
  173. package/dest/msg_validators/tx_validator/double_spend_validator.d.ts.map +1 -1
  174. package/dest/msg_validators/tx_validator/double_spend_validator.js +7 -6
  175. package/dest/msg_validators/tx_validator/factory.d.ts +118 -5
  176. package/dest/msg_validators/tx_validator/factory.d.ts.map +1 -1
  177. package/dest/msg_validators/tx_validator/factory.js +228 -57
  178. package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts +10 -0
  179. package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts.map +1 -0
  180. package/dest/msg_validators/tx_validator/fee_payer_balance.js +20 -0
  181. package/dest/msg_validators/tx_validator/gas_validator.d.ts +59 -3
  182. package/dest/msg_validators/tx_validator/gas_validator.d.ts.map +1 -1
  183. package/dest/msg_validators/tx_validator/gas_validator.js +84 -52
  184. package/dest/msg_validators/tx_validator/index.d.ts +3 -1
  185. package/dest/msg_validators/tx_validator/index.d.ts.map +1 -1
  186. package/dest/msg_validators/tx_validator/index.js +2 -0
  187. package/dest/msg_validators/tx_validator/metadata_validator.d.ts +3 -2
  188. package/dest/msg_validators/tx_validator/metadata_validator.d.ts.map +1 -1
  189. package/dest/msg_validators/tx_validator/metadata_validator.js +2 -2
  190. package/dest/msg_validators/tx_validator/nullifier_cache.d.ts +14 -0
  191. package/dest/msg_validators/tx_validator/nullifier_cache.d.ts.map +1 -0
  192. package/dest/msg_validators/tx_validator/nullifier_cache.js +24 -0
  193. package/dest/msg_validators/tx_validator/phases_validator.d.ts +3 -2
  194. package/dest/msg_validators/tx_validator/phases_validator.d.ts.map +1 -1
  195. package/dest/msg_validators/tx_validator/phases_validator.js +3 -3
  196. package/dest/msg_validators/tx_validator/size_validator.d.ts +8 -0
  197. package/dest/msg_validators/tx_validator/size_validator.d.ts.map +1 -0
  198. package/dest/msg_validators/tx_validator/size_validator.js +23 -0
  199. package/dest/msg_validators/tx_validator/timestamp_validator.d.ts +22 -5
  200. package/dest/msg_validators/tx_validator/timestamp_validator.d.ts.map +1 -1
  201. package/dest/msg_validators/tx_validator/timestamp_validator.js +8 -8
  202. package/dest/msg_validators/tx_validator/tx_permitted_validator.d.ts +3 -2
  203. package/dest/msg_validators/tx_validator/tx_permitted_validator.d.ts.map +1 -1
  204. package/dest/msg_validators/tx_validator/tx_permitted_validator.js +2 -2
  205. package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts +3 -2
  206. package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts.map +1 -1
  207. package/dest/msg_validators/tx_validator/tx_proof_validator.js +2 -2
  208. package/dest/services/data_store.d.ts +1 -1
  209. package/dest/services/data_store.d.ts.map +1 -1
  210. package/dest/services/data_store.js +10 -6
  211. package/dest/services/discv5/discV5_service.js +1 -1
  212. package/dest/services/dummy_service.d.ts +28 -4
  213. package/dest/services/dummy_service.d.ts.map +1 -1
  214. package/dest/services/dummy_service.js +49 -1
  215. package/dest/services/encoding.d.ts +3 -3
  216. package/dest/services/encoding.d.ts.map +1 -1
  217. package/dest/services/encoding.js +16 -14
  218. package/dest/services/gossipsub/index.d.ts +3 -0
  219. package/dest/services/gossipsub/index.d.ts.map +1 -0
  220. package/dest/services/gossipsub/index.js +2 -0
  221. package/dest/services/gossipsub/scoring.d.ts +21 -3
  222. package/dest/services/gossipsub/scoring.d.ts.map +1 -1
  223. package/dest/services/gossipsub/scoring.js +24 -7
  224. package/dest/services/gossipsub/topic_score_params.d.ts +173 -0
  225. package/dest/services/gossipsub/topic_score_params.d.ts.map +1 -0
  226. package/dest/services/gossipsub/topic_score_params.js +346 -0
  227. package/dest/services/index.d.ts +2 -1
  228. package/dest/services/index.d.ts.map +1 -1
  229. package/dest/services/index.js +1 -0
  230. package/dest/services/libp2p/instrumentation.d.ts +1 -1
  231. package/dest/services/libp2p/instrumentation.d.ts.map +1 -1
  232. package/dest/services/libp2p/instrumentation.js +30 -72
  233. package/dest/services/libp2p/libp2p_service.d.ts +115 -40
  234. package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
  235. package/dest/services/libp2p/libp2p_service.js +1020 -369
  236. package/dest/services/peer-manager/metrics.d.ts +2 -2
  237. package/dest/services/peer-manager/metrics.d.ts.map +1 -1
  238. package/dest/services/peer-manager/metrics.js +21 -26
  239. package/dest/services/peer-manager/peer_manager.d.ts +2 -2
  240. package/dest/services/peer-manager/peer_manager.d.ts.map +1 -1
  241. package/dest/services/peer-manager/peer_manager.js +0 -10
  242. package/dest/services/peer-manager/peer_scoring.d.ts +1 -1
  243. package/dest/services/peer-manager/peer_scoring.d.ts.map +1 -1
  244. package/dest/services/peer-manager/peer_scoring.js +32 -6
  245. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts +48 -0
  246. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts.map +1 -0
  247. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.js +539 -0
  248. package/dest/services/reqresp/batch-tx-requester/config.d.ts +17 -0
  249. package/dest/services/reqresp/batch-tx-requester/config.d.ts.map +1 -0
  250. package/dest/services/reqresp/batch-tx-requester/config.js +27 -0
  251. package/dest/services/reqresp/batch-tx-requester/interface.d.ts +46 -0
  252. package/dest/services/reqresp/batch-tx-requester/interface.d.ts.map +1 -0
  253. package/dest/services/reqresp/batch-tx-requester/interface.js +1 -0
  254. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts +34 -0
  255. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts.map +1 -0
  256. package/dest/services/reqresp/batch-tx-requester/missing_txs.js +130 -0
  257. package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts +60 -0
  258. package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts.map +1 -0
  259. package/dest/services/reqresp/batch-tx-requester/peer_collection.js +173 -0
  260. package/dest/services/reqresp/batch-tx-requester/tx_validator.d.ts +20 -0
  261. package/dest/services/reqresp/batch-tx-requester/tx_validator.d.ts.map +1 -0
  262. package/dest/services/reqresp/batch-tx-requester/tx_validator.js +21 -0
  263. package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts +22 -3
  264. package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts.map +1 -1
  265. package/dest/services/reqresp/connection-sampler/batch_connection_sampler.js +63 -4
  266. package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts +2 -1
  267. package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts.map +1 -1
  268. package/dest/services/reqresp/connection-sampler/connection_sampler.js +12 -0
  269. package/dest/services/reqresp/constants.d.ts +12 -0
  270. package/dest/services/reqresp/constants.d.ts.map +1 -0
  271. package/dest/services/reqresp/constants.js +7 -0
  272. package/dest/services/reqresp/interface.d.ts +12 -1
  273. package/dest/services/reqresp/interface.d.ts.map +1 -1
  274. package/dest/services/reqresp/interface.js +15 -1
  275. package/dest/services/reqresp/metrics.d.ts +6 -5
  276. package/dest/services/reqresp/metrics.d.ts.map +1 -1
  277. package/dest/services/reqresp/metrics.js +17 -21
  278. package/dest/services/reqresp/protocols/block_txs/bitvector.d.ts +5 -1
  279. package/dest/services/reqresp/protocols/block_txs/bitvector.d.ts.map +1 -1
  280. package/dest/services/reqresp/protocols/block_txs/bitvector.js +12 -0
  281. package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts +7 -5
  282. package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts.map +1 -1
  283. package/dest/services/reqresp/protocols/block_txs/block_txs_handler.js +27 -9
  284. package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts +29 -6
  285. package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts.map +1 -1
  286. package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.js +59 -13
  287. package/dest/services/reqresp/protocols/status.d.ts +1 -1
  288. package/dest/services/reqresp/protocols/status.d.ts.map +1 -1
  289. package/dest/services/reqresp/protocols/status.js +4 -1
  290. package/dest/services/reqresp/protocols/tx.d.ts +7 -1
  291. package/dest/services/reqresp/protocols/tx.d.ts.map +1 -1
  292. package/dest/services/reqresp/protocols/tx.js +20 -0
  293. package/dest/services/reqresp/reqresp.d.ts +6 -1
  294. package/dest/services/reqresp/reqresp.d.ts.map +1 -1
  295. package/dest/services/reqresp/reqresp.js +473 -51
  296. package/dest/services/service.d.ts +56 -4
  297. package/dest/services/service.d.ts.map +1 -1
  298. package/dest/services/tx_collection/config.d.ts +22 -1
  299. package/dest/services/tx_collection/config.d.ts.map +1 -1
  300. package/dest/services/tx_collection/config.js +55 -1
  301. package/dest/services/tx_collection/fast_tx_collection.d.ts +7 -4
  302. package/dest/services/tx_collection/fast_tx_collection.d.ts.map +1 -1
  303. package/dest/services/tx_collection/fast_tx_collection.js +71 -44
  304. package/dest/services/tx_collection/file_store_tx_collection.d.ts +53 -0
  305. package/dest/services/tx_collection/file_store_tx_collection.d.ts.map +1 -0
  306. package/dest/services/tx_collection/file_store_tx_collection.js +167 -0
  307. package/dest/services/tx_collection/file_store_tx_source.d.ts +37 -0
  308. package/dest/services/tx_collection/file_store_tx_source.d.ts.map +1 -0
  309. package/dest/services/tx_collection/file_store_tx_source.js +90 -0
  310. package/dest/services/tx_collection/index.d.ts +3 -1
  311. package/dest/services/tx_collection/index.d.ts.map +1 -1
  312. package/dest/services/tx_collection/index.js +2 -0
  313. package/dest/services/tx_collection/instrumentation.d.ts +1 -1
  314. package/dest/services/tx_collection/instrumentation.d.ts.map +1 -1
  315. package/dest/services/tx_collection/instrumentation.js +11 -13
  316. package/dest/services/tx_collection/missing_txs_tracker.d.ts +32 -0
  317. package/dest/services/tx_collection/missing_txs_tracker.d.ts.map +1 -0
  318. package/dest/services/tx_collection/missing_txs_tracker.js +27 -0
  319. package/dest/services/tx_collection/proposal_tx_collector.d.ts +49 -0
  320. package/dest/services/tx_collection/proposal_tx_collector.d.ts.map +1 -0
  321. package/dest/services/tx_collection/proposal_tx_collector.js +50 -0
  322. package/dest/services/tx_collection/slow_tx_collection.d.ts +9 -5
  323. package/dest/services/tx_collection/slow_tx_collection.d.ts.map +1 -1
  324. package/dest/services/tx_collection/slow_tx_collection.js +60 -26
  325. package/dest/services/tx_collection/tx_collection.d.ts +29 -16
  326. package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
  327. package/dest/services/tx_collection/tx_collection.js +79 -7
  328. package/dest/services/tx_collection/tx_collection_sink.d.ts +18 -8
  329. package/dest/services/tx_collection/tx_collection_sink.d.ts.map +1 -1
  330. package/dest/services/tx_collection/tx_collection_sink.js +26 -29
  331. package/dest/services/tx_collection/tx_source.d.ts +8 -3
  332. package/dest/services/tx_collection/tx_source.d.ts.map +1 -1
  333. package/dest/services/tx_collection/tx_source.js +19 -2
  334. package/dest/services/tx_file_store/config.d.ts +16 -0
  335. package/dest/services/tx_file_store/config.d.ts.map +1 -0
  336. package/dest/services/tx_file_store/config.js +22 -0
  337. package/dest/services/tx_file_store/index.d.ts +4 -0
  338. package/dest/services/tx_file_store/index.d.ts.map +1 -0
  339. package/dest/services/tx_file_store/index.js +3 -0
  340. package/dest/services/tx_file_store/instrumentation.d.ts +15 -0
  341. package/dest/services/tx_file_store/instrumentation.d.ts.map +1 -0
  342. package/dest/services/tx_file_store/instrumentation.js +29 -0
  343. package/dest/services/tx_file_store/tx_file_store.d.ts +48 -0
  344. package/dest/services/tx_file_store/tx_file_store.d.ts.map +1 -0
  345. package/dest/services/tx_file_store/tx_file_store.js +152 -0
  346. package/dest/services/tx_provider.d.ts +6 -6
  347. package/dest/services/tx_provider.d.ts.map +1 -1
  348. package/dest/services/tx_provider.js +9 -8
  349. package/dest/services/tx_provider_instrumentation.d.ts +1 -1
  350. package/dest/services/tx_provider_instrumentation.d.ts.map +1 -1
  351. package/dest/services/tx_provider_instrumentation.js +7 -20
  352. package/dest/test-helpers/index.d.ts +3 -1
  353. package/dest/test-helpers/index.d.ts.map +1 -1
  354. package/dest/test-helpers/index.js +2 -0
  355. package/dest/test-helpers/make-test-p2p-clients.d.ts +7 -8
  356. package/dest/test-helpers/make-test-p2p-clients.d.ts.map +1 -1
  357. package/dest/test-helpers/make-test-p2p-clients.js +1 -2
  358. package/dest/test-helpers/mock-pubsub.d.ts +30 -4
  359. package/dest/test-helpers/mock-pubsub.d.ts.map +1 -1
  360. package/dest/test-helpers/mock-pubsub.js +105 -4
  361. package/dest/test-helpers/reqresp-nodes.d.ts +2 -3
  362. package/dest/test-helpers/reqresp-nodes.d.ts.map +1 -1
  363. package/dest/test-helpers/reqresp-nodes.js +4 -3
  364. package/dest/test-helpers/test_tx_provider.d.ts +40 -0
  365. package/dest/test-helpers/test_tx_provider.d.ts.map +1 -0
  366. package/dest/test-helpers/test_tx_provider.js +41 -0
  367. package/dest/test-helpers/testbench-utils.d.ts +163 -0
  368. package/dest/test-helpers/testbench-utils.d.ts.map +1 -0
  369. package/dest/test-helpers/testbench-utils.js +366 -0
  370. package/dest/testbench/p2p_client_testbench_worker.d.ts +28 -2
  371. package/dest/testbench/p2p_client_testbench_worker.d.ts.map +1 -1
  372. package/dest/testbench/p2p_client_testbench_worker.js +225 -131
  373. package/dest/testbench/worker_client_manager.d.ts +51 -6
  374. package/dest/testbench/worker_client_manager.d.ts.map +1 -1
  375. package/dest/testbench/worker_client_manager.js +226 -39
  376. package/dest/util.d.ts +2 -2
  377. package/dest/util.d.ts.map +1 -1
  378. package/package.json +16 -16
  379. package/src/bootstrap/bootstrap.ts +7 -4
  380. package/src/client/factory.ts +83 -36
  381. package/src/client/interface.ts +73 -36
  382. package/src/client/p2p_client.ts +259 -364
  383. package/src/client/test/tx_proposal_collector/README.md +227 -0
  384. package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker.ts +345 -0
  385. package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.ts +43 -0
  386. package/src/config.ts +49 -13
  387. package/src/errors/tx-pool.error.ts +12 -0
  388. package/src/index.ts +1 -0
  389. package/src/mem_pools/attestation_pool/attestation_pool.ts +510 -78
  390. package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +612 -320
  391. package/src/mem_pools/attestation_pool/index.ts +9 -2
  392. package/src/mem_pools/attestation_pool/mocks.ts +20 -13
  393. package/src/mem_pools/index.ts +4 -1
  394. package/src/mem_pools/instrumentation.ts +10 -18
  395. package/src/mem_pools/interface.ts +4 -4
  396. package/src/mem_pools/tx_pool/README.md +29 -14
  397. package/src/mem_pools/tx_pool/aztec_kv_tx_pool.ts +130 -75
  398. package/src/mem_pools/tx_pool/eviction/eviction_manager.ts +66 -5
  399. package/src/mem_pools/tx_pool/eviction/eviction_strategy.ts +119 -4
  400. package/src/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.ts +162 -0
  401. package/src/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.ts +3 -3
  402. package/src/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.ts +4 -2
  403. package/src/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.ts +75 -0
  404. package/src/mem_pools/tx_pool_v2/README.md +275 -0
  405. package/src/mem_pools/tx_pool_v2/archive/index.ts +1 -0
  406. package/src/mem_pools/tx_pool_v2/archive/tx_archive.ts +120 -0
  407. package/src/mem_pools/tx_pool_v2/deleted_pool.ts +321 -0
  408. package/src/mem_pools/tx_pool_v2/eviction/eviction_manager.ts +160 -0
  409. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.ts +121 -0
  410. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.ts +125 -0
  411. package/src/mem_pools/tx_pool_v2/eviction/index.ts +27 -0
  412. package/src/mem_pools/tx_pool_v2/eviction/interfaces.ts +209 -0
  413. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.ts +74 -0
  414. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.ts +101 -0
  415. package/src/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.ts +91 -0
  416. package/src/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.ts +90 -0
  417. package/src/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.ts +31 -0
  418. package/src/mem_pools/tx_pool_v2/index.ts +12 -0
  419. package/src/mem_pools/tx_pool_v2/instrumentation.ts +69 -0
  420. package/src/mem_pools/tx_pool_v2/interfaces.ts +242 -0
  421. package/src/mem_pools/tx_pool_v2/tx_metadata.ts +297 -0
  422. package/src/mem_pools/tx_pool_v2/tx_pool_bench_metrics.ts +77 -0
  423. package/src/mem_pools/tx_pool_v2/tx_pool_indices.ts +444 -0
  424. package/src/mem_pools/tx_pool_v2/tx_pool_v2.ts +223 -0
  425. package/src/mem_pools/tx_pool_v2/tx_pool_v2_impl.ts +1083 -0
  426. package/src/msg_validators/attestation_validator/attestation_validator.ts +36 -21
  427. package/src/msg_validators/attestation_validator/fisherman_attestation_validator.ts +21 -18
  428. package/src/msg_validators/clock_tolerance.ts +51 -0
  429. package/src/msg_validators/index.ts +1 -1
  430. package/src/msg_validators/proposal_validator/block_proposal_validator.ts +10 -0
  431. package/src/msg_validators/proposal_validator/checkpoint_proposal_validator.ts +13 -0
  432. package/src/msg_validators/proposal_validator/index.ts +3 -0
  433. package/src/msg_validators/proposal_validator/proposal_validator.ts +92 -0
  434. package/src/msg_validators/proposal_validator/proposal_validator_test_suite.ts +230 -0
  435. package/src/msg_validators/tx_validator/README.md +115 -0
  436. package/src/msg_validators/tx_validator/aggregate_tx_validator.ts +5 -5
  437. package/src/msg_validators/tx_validator/archive_cache.ts +3 -3
  438. package/src/msg_validators/tx_validator/block_header_validator.ts +21 -8
  439. package/src/msg_validators/tx_validator/data_validator.ts +18 -6
  440. package/src/msg_validators/tx_validator/double_spend_validator.ts +15 -9
  441. package/src/msg_validators/tx_validator/factory.ts +372 -55
  442. package/src/msg_validators/tx_validator/fee_payer_balance.ts +40 -0
  443. package/src/msg_validators/tx_validator/gas_validator.ts +106 -54
  444. package/src/msg_validators/tx_validator/index.ts +2 -0
  445. package/src/msg_validators/tx_validator/metadata_validator.ts +18 -7
  446. package/src/msg_validators/tx_validator/nullifier_cache.ts +30 -0
  447. package/src/msg_validators/tx_validator/phases_validator.ts +5 -3
  448. package/src/msg_validators/tx_validator/size_validator.ts +22 -0
  449. package/src/msg_validators/tx_validator/timestamp_validator.ts +29 -19
  450. package/src/msg_validators/tx_validator/tx_permitted_validator.ts +8 -3
  451. package/src/msg_validators/tx_validator/tx_proof_validator.ts +8 -3
  452. package/src/services/data_store.ts +10 -7
  453. package/src/services/discv5/discV5_service.ts +1 -1
  454. package/src/services/dummy_service.ts +65 -2
  455. package/src/services/encoding.ts +13 -12
  456. package/src/services/gossipsub/README.md +641 -0
  457. package/src/services/gossipsub/index.ts +2 -0
  458. package/src/services/gossipsub/scoring.ts +29 -5
  459. package/src/services/gossipsub/topic_score_params.ts +487 -0
  460. package/src/services/index.ts +1 -0
  461. package/src/services/libp2p/instrumentation.ts +32 -73
  462. package/src/services/libp2p/libp2p_service.ts +706 -371
  463. package/src/services/peer-manager/metrics.ts +22 -26
  464. package/src/services/peer-manager/peer_manager.ts +1 -2
  465. package/src/services/peer-manager/peer_scoring.ts +28 -4
  466. package/src/services/reqresp/batch-tx-requester/README.md +305 -0
  467. package/src/services/reqresp/batch-tx-requester/batch_tx_requester.ts +678 -0
  468. package/src/services/reqresp/batch-tx-requester/config.ts +40 -0
  469. package/src/services/reqresp/batch-tx-requester/interface.ts +53 -0
  470. package/src/services/reqresp/batch-tx-requester/missing_txs.ts +161 -0
  471. package/src/services/reqresp/batch-tx-requester/peer_collection.ts +244 -0
  472. package/src/services/reqresp/batch-tx-requester/tx_validator.ts +37 -0
  473. package/src/services/reqresp/connection-sampler/batch_connection_sampler.ts +65 -4
  474. package/src/services/reqresp/connection-sampler/connection_sampler.ts +19 -1
  475. package/src/services/reqresp/constants.ts +14 -0
  476. package/src/services/reqresp/interface.ts +29 -1
  477. package/src/services/reqresp/metrics.ts +36 -27
  478. package/src/services/reqresp/protocols/block_txs/bitvector.ts +16 -0
  479. package/src/services/reqresp/protocols/block_txs/block_txs_handler.ts +35 -12
  480. package/src/services/reqresp/protocols/block_txs/block_txs_reqresp.ts +74 -9
  481. package/src/services/reqresp/protocols/status.ts +7 -4
  482. package/src/services/reqresp/protocols/tx.ts +22 -0
  483. package/src/services/reqresp/reqresp.ts +82 -23
  484. package/src/services/service.ts +73 -5
  485. package/src/services/tx_collection/config.ts +83 -1
  486. package/src/services/tx_collection/fast_tx_collection.ts +93 -47
  487. package/src/services/tx_collection/file_store_tx_collection.ts +202 -0
  488. package/src/services/tx_collection/file_store_tx_source.ts +117 -0
  489. package/src/services/tx_collection/index.ts +6 -0
  490. package/src/services/tx_collection/instrumentation.ts +11 -13
  491. package/src/services/tx_collection/missing_txs_tracker.ts +52 -0
  492. package/src/services/tx_collection/proposal_tx_collector.ts +113 -0
  493. package/src/services/tx_collection/slow_tx_collection.ts +68 -35
  494. package/src/services/tx_collection/tx_collection.ts +121 -24
  495. package/src/services/tx_collection/tx_collection_sink.ts +30 -34
  496. package/src/services/tx_collection/tx_source.ts +22 -3
  497. package/src/services/tx_file_store/config.ts +37 -0
  498. package/src/services/tx_file_store/index.ts +3 -0
  499. package/src/services/tx_file_store/instrumentation.ts +36 -0
  500. package/src/services/tx_file_store/tx_file_store.ts +175 -0
  501. package/src/services/tx_provider.ts +12 -11
  502. package/src/services/tx_provider_instrumentation.ts +13 -20
  503. package/src/test-helpers/index.ts +2 -0
  504. package/src/test-helpers/make-test-p2p-clients.ts +3 -5
  505. package/src/test-helpers/mock-pubsub.ts +146 -9
  506. package/src/test-helpers/reqresp-nodes.ts +4 -6
  507. package/src/test-helpers/test_tx_provider.ts +64 -0
  508. package/src/test-helpers/testbench-utils.ts +430 -0
  509. package/src/testbench/p2p_client_testbench_worker.ts +349 -128
  510. package/src/testbench/worker_client_manager.ts +304 -42
  511. package/src/util.ts +7 -1
  512. package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts +0 -37
  513. package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts.map +0 -1
  514. package/dest/mem_pools/attestation_pool/kv_attestation_pool.js +0 -213
  515. package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts +0 -30
  516. package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts.map +0 -1
  517. package/dest/mem_pools/attestation_pool/memory_attestation_pool.js +0 -219
  518. package/dest/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.d.ts +0 -15
  519. package/dest/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.d.ts.map +0 -1
  520. package/dest/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.js +0 -88
  521. package/dest/msg_validators/block_proposal_validator/block_proposal_validator.d.ts +0 -12
  522. package/dest/msg_validators/block_proposal_validator/block_proposal_validator.d.ts.map +0 -1
  523. package/dest/msg_validators/block_proposal_validator/block_proposal_validator.js +0 -82
  524. package/dest/msg_validators/block_proposal_validator/index.d.ts +0 -2
  525. package/dest/msg_validators/block_proposal_validator/index.d.ts.map +0 -1
  526. package/dest/msg_validators/block_proposal_validator/index.js +0 -1
  527. package/src/mem_pools/attestation_pool/kv_attestation_pool.ts +0 -298
  528. package/src/mem_pools/attestation_pool/memory_attestation_pool.ts +0 -287
  529. package/src/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.ts +0 -108
  530. package/src/msg_validators/block_proposal_validator/block_proposal_validator.ts +0 -97
  531. package/src/msg_validators/block_proposal_validator/index.ts +0 -1
@@ -1,34 +1,41 @@
1
1
  import type { EpochCacheInterface } from '@aztec/epoch-cache';
2
- import { BlockNumber, SlotNumber } from '@aztec/foundation/branded-types';
3
- import { randomInt } from '@aztec/foundation/crypto/random';
2
+ import { BlockNumber, type SlotNumber } from '@aztec/foundation/branded-types';
4
3
  import { Fr } from '@aztec/foundation/curves/bn254';
5
4
  import { type Logger, createLibp2pComponentLogger, createLogger } from '@aztec/foundation/log';
6
5
  import { RunningPromise } from '@aztec/foundation/running-promise';
7
6
  import { Timer } from '@aztec/foundation/timer';
8
7
  import type { AztecAsyncKVStore } from '@aztec/kv-store';
9
- import { getVKTreeRoot } from '@aztec/noir-protocol-circuits-types/vk-tree';
10
8
  import { protocolContractsHash } from '@aztec/protocol-contracts';
11
9
  import type { EthAddress, L2Block, L2BlockSource } from '@aztec/stdlib/block';
12
10
  import type { ContractDataSource } from '@aztec/stdlib/contract';
13
11
  import { GasFees } from '@aztec/stdlib/gas';
14
12
  import type { ClientProtocolCircuitVerifier, PeerInfo, WorldStateSynchronizer } from '@aztec/stdlib/interfaces/server';
15
13
  import {
16
- BlockAttestation,
17
14
  BlockProposal,
15
+ CheckpointAttestation,
16
+ CheckpointProposal,
17
+ type CheckpointProposalCore,
18
18
  type Gossipable,
19
- P2PClientType,
20
19
  P2PMessage,
20
+ type ValidationResult as P2PValidationResult,
21
21
  PeerErrorSeverity,
22
22
  TopicType,
23
23
  createTopicString,
24
- getTopicsForClientAndConfig,
24
+ getTopicsForConfig,
25
25
  metricsTopicStrToLabels,
26
26
  } from '@aztec/stdlib/p2p';
27
27
  import { MerkleTreeId } from '@aztec/stdlib/trees';
28
28
  import { Tx, type TxHash, type TxValidationResult, type TxValidator } from '@aztec/stdlib/tx';
29
29
  import type { UInt64 } from '@aztec/stdlib/types';
30
30
  import { compressComponentVersions } from '@aztec/stdlib/versioning';
31
- import { Attributes, OtelMetricsAdapter, type TelemetryClient, WithTracer, trackSpan } from '@aztec/telemetry-client';
31
+ import {
32
+ Attributes,
33
+ OtelMetricsAdapter,
34
+ SpanStatusCode,
35
+ type TelemetryClient,
36
+ WithTracer,
37
+ trackSpan,
38
+ } from '@aztec/telemetry-client';
32
39
 
33
40
  import {
34
41
  type GossipSub,
@@ -36,7 +43,7 @@ import {
36
43
  type GossipsubMessage,
37
44
  gossipsub,
38
45
  } from '@chainsafe/libp2p-gossipsub';
39
- import { createPeerScoreParams, createTopicScoreParams } from '@chainsafe/libp2p-gossipsub/score';
46
+ import { createPeerScoreParams } from '@chainsafe/libp2p-gossipsub/score';
40
47
  import { SignaturePolicy } from '@chainsafe/libp2p-gossipsub/types';
41
48
  import { noise } from '@chainsafe/libp2p-noise';
42
49
  import { yamux } from '@chainsafe/libp2p-yamux';
@@ -50,58 +57,65 @@ import { ENR } from '@nethermindeth/enr';
50
57
  import { createLibp2p } from 'libp2p';
51
58
 
52
59
  import type { P2PConfig } from '../../config.js';
53
- import { ProposalSlotCapExceededError } from '../../errors/attestation-pool.error.js';
54
60
  import type { MemPools } from '../../mem_pools/interface.js';
55
61
  import {
56
- AttestationValidator,
57
62
  BlockProposalValidator,
63
+ CheckpointAttestationValidator,
64
+ CheckpointProposalValidator,
65
+ DoubleSpendTxValidator,
58
66
  FishermanAttestationValidator,
67
+ getDefaultAllowedSetupFunctions,
59
68
  } from '../../msg_validators/index.js';
60
69
  import { MessageSeenValidator } from '../../msg_validators/msg_seen_validator/msg_seen_validator.js';
61
- import { getDefaultAllowedSetupFunctions } from '../../msg_validators/tx_validator/allowed_public_setup.js';
62
- import { type MessageValidator, createTxMessageValidators } from '../../msg_validators/tx_validator/factory.js';
63
70
  import {
64
- AggregateTxValidator,
65
- DataTxValidator,
66
- DoubleSpendTxValidator,
67
- MetadataTxValidator,
68
- TxProofValidator,
69
- } from '../../msg_validators/tx_validator/index.js';
71
+ type TransactionValidator,
72
+ createFirstStageTxValidationsForGossipedTransactions,
73
+ createSecondStageTxValidationsForGossipedTransactions,
74
+ createTxValidatorForBlockProposalReceivedTxs,
75
+ createTxValidatorForReqResponseReceivedTxs,
76
+ } from '../../msg_validators/tx_validator/factory.js';
70
77
  import { GossipSubEvent } from '../../types/index.js';
71
78
  import { type PubSubLibp2p, convertToMultiaddr } from '../../util.js';
72
79
  import { getVersions } from '../../versioning.js';
73
80
  import { AztecDatastore } from '../data_store.js';
74
81
  import { DiscV5Service } from '../discv5/discV5_service.js';
75
82
  import { SnappyTransform, fastMsgIdFn, getMsgIdFn, msgIdToStrFn } from '../encoding.js';
76
- import { gossipScoreThresholds } from '../gossipsub/scoring.js';
83
+ import { APP_SPECIFIC_WEIGHT, gossipScoreThresholds } from '../gossipsub/scoring.js';
84
+ import { createAllTopicScoreParams } from '../gossipsub/topic_score_params.js';
77
85
  import type { PeerManagerInterface } from '../peer-manager/interface.js';
78
86
  import { PeerManager } from '../peer-manager/peer_manager.js';
79
87
  import { PeerScoring } from '../peer-manager/peer_scoring.js';
88
+ import type { BatchTxRequesterLibP2PService } from '../reqresp/batch-tx-requester/interface.js';
80
89
  import type { P2PReqRespConfig } from '../reqresp/config.js';
81
90
  import {
91
+ AuthRequest,
92
+ BlockTxsRequest,
93
+ BlockTxsResponse,
82
94
  DEFAULT_SUB_PROTOCOL_VALIDATORS,
83
95
  type ReqRespInterface,
96
+ type ReqRespResponse,
84
97
  ReqRespSubProtocol,
85
98
  type ReqRespSubProtocolHandler,
86
99
  type ReqRespSubProtocolHandlers,
87
100
  type ReqRespSubProtocolValidators,
101
+ StatusMessage,
88
102
  type SubProtocolMap,
89
103
  ValidationError,
90
- } from '../reqresp/interface.js';
91
- import { reqRespBlockTxsHandler } from '../reqresp/protocols/block_txs/block_txs_handler.js';
92
- import { reqGoodbyeHandler } from '../reqresp/protocols/goodbye.js';
93
- import {
94
- AuthRequest,
95
- BlockTxsRequest,
96
- BlockTxsResponse,
97
- StatusMessage,
98
104
  pingHandler,
105
+ reqGoodbyeHandler,
99
106
  reqRespBlockHandler,
107
+ reqRespBlockTxsHandler,
100
108
  reqRespStatusHandler,
101
109
  reqRespTxHandler,
102
- } from '../reqresp/protocols/index.js';
110
+ } from '../reqresp/index.js';
103
111
  import { ReqResp } from '../reqresp/reqresp.js';
104
- import type { P2PBlockReceivedCallback, P2PService, PeerDiscoveryService } from '../service.js';
112
+ import type {
113
+ P2PBlockReceivedCallback,
114
+ P2PCheckpointReceivedCallback,
115
+ P2PDuplicateAttestationCallback,
116
+ P2PService,
117
+ PeerDiscoveryService,
118
+ } from '../service.js';
105
119
  import { P2PInstrumentation } from './instrumentation.js';
106
120
 
107
121
  interface ValidationResult {
@@ -113,26 +127,37 @@ interface ValidationResult {
113
127
  type ValidationOutcome = { allPassed: true } | { allPassed: false; failure: ValidationResult };
114
128
 
115
129
  // REFACTOR: Unify with the type above
116
- type ReceivedMessageValidationResult<T> =
117
- | { obj: T; result: Exclude<TopicValidatorResult, TopicValidatorResult.Reject> }
118
- | { obj?: undefined; result: TopicValidatorResult.Reject };
130
+ type ReceivedMessageValidationResult<T, M = undefined> =
131
+ | { obj: T; result: Exclude<TopicValidatorResult, TopicValidatorResult.Reject>; metadata?: M }
132
+ | { obj?: T; result: TopicValidatorResult.Reject; metadata?: M };
119
133
 
120
134
  /**
121
135
  * Lib P2P implementation of the P2PService interface.
122
136
  */
123
- export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends WithTracer implements P2PService {
137
+ export class LibP2PService extends WithTracer implements P2PService {
124
138
  private discoveryRunningPromise?: RunningPromise;
125
139
  private msgIdSeenValidators: Record<TopicType, MessageSeenValidator> = {} as Record<TopicType, MessageSeenValidator>;
126
140
 
127
141
  // Message validators
128
- private attestationValidator: AttestationValidator;
129
142
  private blockProposalValidator: BlockProposalValidator;
143
+ private checkpointProposalValidator: CheckpointProposalValidator;
144
+ private checkpointAttestationValidator: CheckpointAttestationValidator;
130
145
 
131
146
  private protocolVersion = '';
132
147
  private topicStrings: Record<TopicType, string> = {} as Record<TopicType, string>;
133
148
 
134
149
  private feesCache: { blockNumber: BlockNumber; gasFees: GasFees } | undefined;
135
150
 
151
+ /** Callback invoked when a duplicate proposal is detected (triggers slashing). */
152
+ private duplicateProposalCallback?: (info: {
153
+ slot: SlotNumber;
154
+ proposer: EthAddress;
155
+ type: 'checkpoint' | 'block';
156
+ }) => void;
157
+
158
+ /** Callback invoked when a duplicate attestation is detected (triggers slashing). */
159
+ private duplicateAttestationCallback?: P2PDuplicateAttestationCallback;
160
+
136
161
  /**
137
162
  * Callback for when a block is received from a peer.
138
163
  * @param block - The block received from the peer.
@@ -140,21 +165,29 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
140
165
  */
141
166
  private blockReceivedCallback: P2PBlockReceivedCallback;
142
167
 
168
+ /**
169
+ * Callback for when a checkpoint proposal is received from a peer.
170
+ * @param checkpoint - The checkpoint proposal received from the peer.
171
+ * @returns The attestations for the checkpoint, if any.
172
+ */
173
+ private checkpointReceivedCallback: P2PCheckpointReceivedCallback;
174
+
143
175
  private gossipSubEventHandler: (e: CustomEvent<GossipsubMessage>) => void;
144
176
 
145
177
  private instrumentation: P2PInstrumentation;
146
178
 
179
+ private telemetry: TelemetryClient;
180
+
147
181
  protected logger: Logger;
148
182
 
149
183
  constructor(
150
- private clientType: T,
151
184
  private config: P2PConfig,
152
185
  protected node: PubSubLibp2p,
153
186
  private peerDiscoveryService: PeerDiscoveryService,
154
187
  private reqresp: ReqRespInterface,
155
- private peerManager: PeerManagerInterface,
188
+ protected peerManager: PeerManagerInterface,
156
189
  protected mempools: MemPools,
157
- private archiver: L2BlockSource & ContractDataSource,
190
+ protected archiver: L2BlockSource & ContractDataSource,
158
191
  private epochCache: EpochCacheInterface,
159
192
  private proofVerifier: ClientProtocolCircuitVerifier,
160
193
  private worldStateSynchronizer: WorldStateSynchronizer,
@@ -162,6 +195,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
162
195
  logger: Logger = createLogger('p2p:libp2p_service'),
163
196
  ) {
164
197
  super(telemetry, 'LibP2PService');
198
+ this.telemetry = telemetry;
165
199
 
166
200
  // Create child logger with fisherman prefix if in fisherman mode
167
201
  this.logger = config.fishermanMode ? logger.createChild('[FISHERMAN]') : logger;
@@ -170,7 +204,8 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
170
204
 
171
205
  this.msgIdSeenValidators[TopicType.tx] = new MessageSeenValidator(config.seenMessageCacheSize);
172
206
  this.msgIdSeenValidators[TopicType.block_proposal] = new MessageSeenValidator(config.seenMessageCacheSize);
173
- this.msgIdSeenValidators[TopicType.block_attestation] = new MessageSeenValidator(config.seenMessageCacheSize);
207
+ this.msgIdSeenValidators[TopicType.checkpoint_proposal] = new MessageSeenValidator(config.seenMessageCacheSize);
208
+ this.msgIdSeenValidators[TopicType.checkpoint_attestation] = new MessageSeenValidator(config.seenMessageCacheSize);
174
209
 
175
210
  const versions = getVersions(config);
176
211
  this.protocolVersion = compressComponentVersions(versions);
@@ -178,25 +213,40 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
178
213
 
179
214
  this.topicStrings[TopicType.tx] = createTopicString(TopicType.tx, this.protocolVersion);
180
215
  this.topicStrings[TopicType.block_proposal] = createTopicString(TopicType.block_proposal, this.protocolVersion);
181
- this.topicStrings[TopicType.block_attestation] = createTopicString(
182
- TopicType.block_attestation,
216
+ this.topicStrings[TopicType.checkpoint_proposal] = createTopicString(
217
+ TopicType.checkpoint_proposal,
218
+ this.protocolVersion,
219
+ );
220
+ this.topicStrings[TopicType.checkpoint_attestation] = createTopicString(
221
+ TopicType.checkpoint_attestation,
183
222
  this.protocolVersion,
184
223
  );
185
224
 
186
- // Use FishermanAttestationValidator in fisherman mode to validate attestation payloads against proposals
187
- this.attestationValidator = config.fishermanMode
188
- ? new FishermanAttestationValidator(epochCache, mempools.attestationPool, telemetry)
189
- : new AttestationValidator(epochCache);
190
225
  this.blockProposalValidator = new BlockProposalValidator(epochCache, { txsPermitted: !config.disableTransactions });
226
+ this.checkpointProposalValidator = new CheckpointProposalValidator(epochCache, {
227
+ txsPermitted: !config.disableTransactions,
228
+ });
229
+ this.checkpointAttestationValidator = config.fishermanMode
230
+ ? new FishermanAttestationValidator(epochCache, mempools.attestationPool, telemetry)
231
+ : new CheckpointAttestationValidator(epochCache);
191
232
 
192
233
  this.gossipSubEventHandler = this.handleGossipSubEvent.bind(this);
193
234
 
194
- this.blockReceivedCallback = async (block: BlockProposal): Promise<BlockAttestation[] | undefined> => {
235
+ this.blockReceivedCallback = async (block: BlockProposal): Promise<boolean> => {
195
236
  this.logger.debug(
196
237
  `Handler not yet registered: Block received callback not set. Received block for slot ${block.slotNumber} from peer.`,
197
238
  { p2pMessageIdentifier: await block.p2pMessageLoggingIdentifier() },
198
239
  );
199
- return undefined;
240
+ return false;
241
+ };
242
+
243
+ this.checkpointReceivedCallback = (
244
+ checkpoint: CheckpointProposalCore,
245
+ ): Promise<CheckpointAttestation[] | undefined> => {
246
+ this.logger.debug(
247
+ `Handler not yet registered: Checkpoint received callback not set. Received checkpoint for slot ${checkpoint.slotNumber} from peer.`,
248
+ );
249
+ return Promise.resolve(undefined);
200
250
  };
201
251
  }
202
252
 
@@ -210,8 +260,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
210
260
  * @param txPool - The transaction pool to be accessed by the service.
211
261
  * @returns The new service.
212
262
  */
213
- public static async new<T extends P2PClientType>(
214
- clientType: T,
263
+ public static async new(
215
264
  config: P2PConfig,
216
265
  peerId: PeerId,
217
266
  deps: {
@@ -242,14 +291,14 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
242
291
 
243
292
  const datastore = new AztecDatastore(peerStore);
244
293
 
245
- const otelMetricsAdapter = new OtelMetricsAdapter(telemetry);
294
+ const otelMetricsAdapter = new OtelMetricsAdapter(telemetry, logger.getBindings());
246
295
 
247
296
  const peerDiscoveryService = new DiscV5Service(
248
297
  peerId,
249
298
  config,
250
299
  packageVersion,
251
300
  telemetry,
252
- createLogger(`${logger.module}:discv5_service`),
301
+ createLogger(`${logger.module}:discv5_service`, logger.getBindings()),
253
302
  );
254
303
 
255
304
  // Seed libp2p's bootstrap discovery with private and trusted peers
@@ -263,10 +312,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
263
312
  const versions = getVersions(config);
264
313
  const protocolVersion = compressComponentVersions(versions);
265
314
 
266
- const txTopic = createTopicString(TopicType.tx, protocolVersion);
267
- const blockProposalTopic = createTopicString(TopicType.block_proposal, protocolVersion);
268
- const blockAttestationTopic = createTopicString(TopicType.block_attestation, protocolVersion);
269
-
270
315
  const preferredPeersEnrs: ENR[] = config.preferredPeers.map(enr => ENR.decodeTxt(enr));
271
316
  const directPeers = (
272
317
  await Promise.all(
@@ -286,6 +331,16 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
286
331
 
287
332
  const announceTcpMultiaddr = config.p2pIp ? [convertToMultiaddr(config.p2pIp, p2pPort, 'tcp')] : [];
288
333
 
334
+ // Create dynamic topic score params based on network configuration
335
+ const l1Constants = epochCache.getL1Constants();
336
+ const topicScoreParams = createAllTopicScoreParams(protocolVersion, {
337
+ slotDurationMs: l1Constants.slotDuration * 1000,
338
+ heartbeatIntervalMs: config.gossipsubInterval,
339
+ targetCommitteeSize: l1Constants.targetCommitteeSize,
340
+ blockDurationMs: config.blockDurationMs,
341
+ expectedBlockProposalsPerSlot: config.expectedBlockProposalsPerSlot,
342
+ });
343
+
289
344
  const node = await createLibp2p({
290
345
  start: false,
291
346
  peerId,
@@ -381,30 +436,14 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
381
436
  scoreParams: createPeerScoreParams({
382
437
  // IPColocation factor can be disabled for local testing - default to -5
383
438
  IPColocationFactorWeight: config.debugDisableColocationPenalty ? 0 : -5.0,
384
- topics: {
385
- [txTopic]: createTopicScoreParams({
386
- topicWeight: 1,
387
- invalidMessageDeliveriesWeight: -20,
388
- invalidMessageDeliveriesDecay: 0.5,
389
- }),
390
- [blockAttestationTopic]: createTopicScoreParams({
391
- topicWeight: 1,
392
- invalidMessageDeliveriesWeight: -20,
393
- invalidMessageDeliveriesDecay: 0.5,
394
- }),
395
- [blockProposalTopic]: createTopicScoreParams({
396
- topicWeight: 1,
397
- invalidMessageDeliveriesWeight: -20,
398
- invalidMessageDeliveriesDecay: 0.5,
399
- }),
400
- },
439
+ topics: topicScoreParams,
401
440
  }),
402
441
  }) as (components: GossipSubComponents) => GossipSub,
403
442
  components: (components: { connectionManager: ConnectionManager }) => ({
404
443
  connectionManager: components.connectionManager,
405
444
  }),
406
445
  },
407
- logger: createLibp2pComponentLogger(logger.module),
446
+ logger: createLibp2pComponentLogger(logger.module, logger.getBindings()),
408
447
  });
409
448
 
410
449
  const peerScoring = new PeerScoring(config, telemetry);
@@ -423,13 +462,16 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
423
462
  epochCache,
424
463
  );
425
464
 
426
- // Update gossipsub score params
427
- node.services.pubsub.score.params.appSpecificWeight = 10;
465
+ // Configure application-specific scoring for gossipsub.
466
+ // The weight scales app score to align with gossipsub thresholds:
467
+ // - Disconnect (-50) × 10 = -500 = gossipThreshold (stops receiving gossip)
468
+ // - Ban (-100) × 10 = -1000 = publishThreshold (cannot publish)
469
+ // Note: positive topic scores can offset penalties, so alignment is best-effort.
470
+ node.services.pubsub.score.params.appSpecificWeight = APP_SPECIFIC_WEIGHT;
428
471
  node.services.pubsub.score.params.appSpecificScore = (peerId: string) =>
429
472
  peerManager.shouldDisableP2PGossip(peerId) ? -Infinity : peerManager.getPeerScore(peerId);
430
473
 
431
474
  return new LibP2PService(
432
- clientType,
433
475
  config,
434
476
  node,
435
477
  peerDiscoveryService,
@@ -462,17 +504,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
462
504
  }
463
505
  const announceTcpMultiaddr = convertToMultiaddr(p2pIp, p2pPort, 'tcp');
464
506
 
465
- await this.peerManager.initializePeers();
466
- if (!this.config.p2pDiscoveryDisabled) {
467
- await this.peerDiscoveryService.start();
468
- }
469
- await this.node.start();
470
-
471
- // Subscribe to standard GossipSub topics by default
472
- for (const topic of getTopicsForClientAndConfig(this.clientType, this.config.disableTransactions)) {
473
- this.subscribeToTopic(this.topicStrings[topic]);
474
- }
475
-
476
507
  // Create request response protocol handlers
477
508
  const txHandler = reqRespTxHandler(this.mempools);
478
509
  const goodbyeHandler = reqGoodbyeHandler(this.peerManager);
@@ -487,7 +518,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
487
518
  };
488
519
 
489
520
  if (!this.config.disableTransactions) {
490
- const blockTxsHandler = reqRespBlockTxsHandler(this.mempools.attestationPool, this.mempools.txPool);
521
+ const blockTxsHandler = reqRespBlockTxsHandler(
522
+ this.mempools.attestationPool,
523
+ this.archiver,
524
+ this.mempools.txPool,
525
+ );
491
526
  requestResponseHandlers[ReqRespSubProtocol.BLOCK_TXS] = blockTxsHandler.bind(this);
492
527
  }
493
528
 
@@ -495,10 +530,32 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
495
530
  requestResponseHandlers[ReqRespSubProtocol.TX] = txHandler.bind(this);
496
531
  }
497
532
 
533
+ // Define the sub protocol validators - This is done within this start() method to gain a callback to the existing validateTx function
534
+ const reqrespSubProtocolValidators = {
535
+ ...DEFAULT_SUB_PROTOCOL_VALIDATORS,
536
+ [ReqRespSubProtocol.TX]: this.validateRequestedTxs.bind(this),
537
+ [ReqRespSubProtocol.BLOCK_TXS]: this.validateRequestedBlockTxs.bind(this),
538
+ [ReqRespSubProtocol.BLOCK]: this.validateRequestedBlock.bind(this),
539
+ };
540
+
541
+ await this.peerManager.initializePeers();
542
+
543
+ await this.reqresp.start(requestResponseHandlers, reqrespSubProtocolValidators);
544
+
545
+ await this.node.start();
546
+
547
+ // Subscribe to standard GossipSub topics by default
548
+ for (const topic of getTopicsForConfig(this.config.disableTransactions)) {
549
+ this.subscribeToTopic(this.topicStrings[topic]);
550
+ }
551
+
498
552
  // add GossipSub listener
499
553
  this.node.services.pubsub.addEventListener(GossipSubEvent.MESSAGE, this.gossipSubEventHandler);
500
554
 
501
555
  // Start running promise for peer discovery and metrics collection
556
+ if (!this.config.p2pDiscoveryDisabled) {
557
+ await this.peerDiscoveryService.start();
558
+ }
502
559
  this.discoveryRunningPromise = new RunningPromise(
503
560
  async () => {
504
561
  await this.peerManager.heartbeat();
@@ -508,14 +565,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
508
565
  );
509
566
  this.discoveryRunningPromise.start();
510
567
 
511
- // Define the sub protocol validators - This is done within this start() method to gain a callback to the existing validateTx function
512
- const reqrespSubProtocolValidators = {
513
- ...DEFAULT_SUB_PROTOCOL_VALIDATORS,
514
- [ReqRespSubProtocol.TX]: this.validateRequestedTxs.bind(this),
515
- [ReqRespSubProtocol.BLOCK_TXS]: this.validateRequestedBlockTxs.bind(this),
516
- [ReqRespSubProtocol.BLOCK]: this.validateRequestedBlock.bind(this),
517
- };
518
- await this.reqresp.start(requestResponseHandlers, reqrespSubProtocolValidators);
519
568
  this.logger.info(`Started P2P service`, {
520
569
  listen: this.config.listenAddress,
521
570
  port: this.config.p2pPort,
@@ -562,6 +611,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
562
611
  return this.peerManager.getPeers(includePending);
563
612
  }
564
613
 
614
+ public getGossipMeshPeerCount(topicType: TopicType): number {
615
+ return this.node.services.pubsub.getMeshPeers(this.topicStrings[topicType]).length;
616
+ }
617
+
565
618
  private handleGossipSubEvent(e: CustomEvent<GossipsubMessage>) {
566
619
  this.logger.trace(`Received PUBSUB message.`);
567
620
 
@@ -589,6 +642,15 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
589
642
  return this.reqresp.sendBatchRequest(protocol, requests, pinnedPeerId);
590
643
  }
591
644
 
645
+ public sendRequestToPeer(
646
+ peerId: PeerId,
647
+ subProtocol: ReqRespSubProtocol,
648
+ payload: Buffer,
649
+ dialTimeout?: number,
650
+ ): Promise<ReqRespResponse> {
651
+ return this.reqresp.sendRequestToPeer(peerId, subProtocol, payload, dialTimeout);
652
+ }
653
+
592
654
  /**
593
655
  * Get the ENR of the node
594
656
  * @returns The ENR of the node
@@ -601,6 +663,29 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
601
663
  this.blockReceivedCallback = callback;
602
664
  }
603
665
 
666
+ public registerCheckpointReceivedCallback(callback: P2PCheckpointReceivedCallback) {
667
+ this.checkpointReceivedCallback = callback;
668
+ }
669
+
670
+ /**
671
+ * Registers a callback to be invoked when a duplicate proposal is detected.
672
+ * This callback is triggered on the first duplicate (when count goes from 1 to 2).
673
+ */
674
+ public registerDuplicateProposalCallback(
675
+ callback: (info: { slot: SlotNumber; proposer: EthAddress; type: 'checkpoint' | 'block' }) => void,
676
+ ): void {
677
+ this.duplicateProposalCallback = callback;
678
+ }
679
+
680
+ /**
681
+ * Registers a callback to be invoked when a duplicate attestation is detected.
682
+ * A validator signing attestations for different proposals at the same slot.
683
+ * This callback is triggered on the first duplicate (when count goes from 1 to 2).
684
+ */
685
+ public registerDuplicateAttestationCallback(callback: P2PDuplicateAttestationCallback): void {
686
+ this.duplicateAttestationCallback = callback;
687
+ }
688
+
604
689
  /**
605
690
  * Subscribes to a topic.
606
691
  * @param topic - The topic to subscribe to.
@@ -622,7 +707,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
622
707
  if (!this.node.services.pubsub) {
623
708
  throw new Error('Pubsub service not available.');
624
709
  }
625
- const p2pMessage = P2PMessage.fromGossipable(message, this.config.debugP2PInstrumentMessages);
710
+ const isBlockProposal = topic === this.topicStrings[TopicType.block_proposal];
711
+ const traceContext =
712
+ this.config.debugP2PInstrumentMessages && isBlockProposal ? this.telemetry.getTraceContext() : undefined;
713
+ const p2pMessage = P2PMessage.fromGossipable(message, this.config.debugP2PInstrumentMessages, traceContext);
626
714
  const result = await this.node.services.pubsub.publish(topic, p2pMessage.toMessageData());
627
715
  return result.recipients.length;
628
716
  }
@@ -643,12 +731,15 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
643
731
  case this.topicStrings[TopicType.tx]:
644
732
  topicType = TopicType.tx;
645
733
  break;
646
- case this.topicStrings[TopicType.block_attestation]:
647
- topicType = TopicType.block_attestation;
648
- break;
649
734
  case this.topicStrings[TopicType.block_proposal]:
650
735
  topicType = TopicType.block_proposal;
651
736
  break;
737
+ case this.topicStrings[TopicType.checkpoint_proposal]:
738
+ topicType = TopicType.checkpoint_proposal;
739
+ break;
740
+ case this.topicStrings[TopicType.checkpoint_attestation]:
741
+ topicType = TopicType.checkpoint_attestation;
742
+ break;
652
743
  default:
653
744
  this.logger.error(`Received message on unknown topic: ${msg.topic}`);
654
745
  break;
@@ -707,36 +798,85 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
707
798
  return;
708
799
  }
709
800
 
801
+ // Determine topic type for attributes
710
802
  if (msg.topic === this.topicStrings[TopicType.tx]) {
711
803
  topicType = TopicType.tx;
712
- await this.handleGossipedTx(p2pMessage.payload, msgId, source);
804
+ } else if (msg.topic === this.topicStrings[TopicType.checkpoint_attestation]) {
805
+ topicType = TopicType.checkpoint_attestation;
806
+ } else if (msg.topic === this.topicStrings[TopicType.block_proposal]) {
807
+ topicType = TopicType.block_proposal;
808
+ } else if (msg.topic === this.topicStrings[TopicType.checkpoint_proposal]) {
809
+ topicType = TopicType.checkpoint_proposal;
713
810
  }
714
- if (msg.topic === this.topicStrings[TopicType.block_attestation]) {
715
- topicType = TopicType.block_attestation;
716
- if (this.clientType === P2PClientType.Full) {
717
- await this.processAttestationFromPeer(p2pMessage.payload, msgId, source);
811
+
812
+ // Process the message, optionally within a linked span for trace propagation
813
+ const processMessage = async () => {
814
+ if (msg.topic === this.topicStrings[TopicType.tx]) {
815
+ await this.handleGossipedTx(p2pMessage.payload, msgId, source);
816
+ } else if (msg.topic === this.topicStrings[TopicType.checkpoint_attestation]) {
817
+ await this.processCheckpointAttestationFromPeer(p2pMessage.payload, msgId, source);
818
+ } else if (msg.topic === this.topicStrings[TopicType.block_proposal]) {
819
+ await this.processBlockFromPeer(p2pMessage.payload, msgId, source);
820
+ } else if (msg.topic === this.topicStrings[TopicType.checkpoint_proposal]) {
821
+ await this.handleGossipedCheckpointProposal(p2pMessage.payload, msgId, source);
822
+ } else {
823
+ this.logger.error(`Received message on unknown topic: ${msg.topic}`);
718
824
  }
719
- }
720
- if (msg.topic === this.topicStrings[TopicType.block_proposal]) {
721
- topicType = TopicType.block_proposal;
722
- await this.processBlockFromPeer(p2pMessage.payload, msgId, source);
825
+ };
826
+
827
+ const latency = p2pMessage.timestamp !== undefined ? msgReceivedTime - p2pMessage.timestamp.getTime() : undefined;
828
+ const propagatedContext = p2pMessage.traceContext
829
+ ? this.telemetry.extractPropagatedContext(p2pMessage.traceContext)
830
+ : undefined;
831
+
832
+ if (propagatedContext) {
833
+ await this.tracer.startActiveSpan(
834
+ 'LibP2PService.processMessage',
835
+ {
836
+ attributes: {
837
+ [Attributes.TOPIC_NAME]: topicType!,
838
+ [Attributes.PEER_ID]: source.toString(),
839
+ },
840
+ },
841
+ propagatedContext,
842
+ async span => {
843
+ try {
844
+ await processMessage();
845
+ span.setStatus({
846
+ code: SpanStatusCode.OK,
847
+ });
848
+ } catch (err) {
849
+ span.setStatus({
850
+ code: SpanStatusCode.ERROR,
851
+ message: String(err),
852
+ });
853
+ if (typeof err === 'string' || (err && err instanceof Error)) {
854
+ span.recordException(err);
855
+ }
856
+ throw err;
857
+ } finally {
858
+ span.end();
859
+ }
860
+ },
861
+ );
862
+ } else {
863
+ await processMessage();
723
864
  }
724
865
 
725
- if (p2pMessage.timestamp !== undefined && topicType !== undefined) {
726
- const latency = msgReceivedTime - p2pMessage.timestamp.getTime();
866
+ if (latency !== undefined && topicType !== undefined) {
727
867
  this.instrumentation.recordMessageLatency(topicType, latency);
728
868
  }
729
869
 
730
870
  return;
731
871
  }
732
872
 
733
- protected async validateReceivedMessage<T>(
734
- validationFunc: () => Promise<ReceivedMessageValidationResult<T>>,
873
+ protected async validateReceivedMessage<T, M = undefined>(
874
+ validationFunc: () => Promise<ReceivedMessageValidationResult<T, M>>,
735
875
  msgId: string,
736
876
  source: PeerId,
737
877
  topicType: TopicType,
738
- ): Promise<ReceivedMessageValidationResult<T>> {
739
- let resultAndObj: ReceivedMessageValidationResult<T> = { result: TopicValidatorResult.Reject };
878
+ ): Promise<ReceivedMessageValidationResult<T, M>> {
879
+ let resultAndObj: ReceivedMessageValidationResult<T, M> = { result: TopicValidatorResult.Reject };
740
880
  const timer = new Timer();
741
881
  try {
742
882
  resultAndObj = await validationFunc();
@@ -760,21 +900,63 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
760
900
  protected async handleGossipedTx(payloadData: Buffer, msgId: string, source: PeerId) {
761
901
  const validationFunc: () => Promise<ReceivedMessageValidationResult<Tx>> = async () => {
762
902
  const tx = Tx.fromBuffer(payloadData);
763
- const isValid = await this.validatePropagatedTx(tx, source);
764
- const exists = isValid && (await this.mempools.txPool.hasTx(tx.getTxHash()));
903
+
904
+ const currentBlockNumber = await this.archiver.getBlockNumber();
905
+ const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
906
+
907
+ // Stage 1: fast validators (metadata, data, timestamps, double-spend, gas, phases, block header)
908
+ const firstStageValidators = await this.createFirstStageMessageValidators(currentBlockNumber, nextSlotTimestamp);
909
+ const firstStageOutcome = await this.runValidations(tx, firstStageValidators);
910
+ if (!firstStageOutcome.allPassed) {
911
+ const { name } = firstStageOutcome.failure;
912
+ let { severity } = firstStageOutcome.failure;
913
+
914
+ // Double spend validator has a special case handler. We perform more detailed examination
915
+ // as to how recently the nullifier was entered into the tree and if the transaction should
916
+ // have 'known' the nullifier existed. This determines the severity of the penalty applied to the peer.
917
+ if (name === 'doubleSpendValidator') {
918
+ const txBlockNumber = BlockNumber(currentBlockNumber + 1);
919
+ severity = await this.handleDoubleSpendFailure(tx, txBlockNumber);
920
+ }
921
+
922
+ this.peerManager.penalizePeer(source, severity);
923
+ return { result: TopicValidatorResult.Reject };
924
+ }
925
+
926
+ // Pool pre-check: see if the pool would accept this tx before doing expensive proof verification
927
+ const canAdd = await this.mempools.txPool.canAddPendingTx(tx);
928
+ if (canAdd === 'ignored') {
929
+ return { result: TopicValidatorResult.Ignore, obj: tx };
930
+ }
931
+
932
+ // Stage 2: expensive proof verification
933
+ const secondStageValidators = this.createSecondStageMessageValidators();
934
+ const secondStageOutcome = await this.runValidations(tx, secondStageValidators);
935
+ if (!secondStageOutcome.allPassed) {
936
+ const { severity } = secondStageOutcome.failure;
937
+ this.peerManager.penalizePeer(source, severity);
938
+ return { result: TopicValidatorResult.Reject };
939
+ }
940
+
941
+ // Pool add: persist the tx
942
+ const txHash = tx.getTxHash();
943
+ const addResult = await this.mempools.txPool.addPendingTxs([tx], { source: 'gossip' });
944
+
945
+ const wasAccepted = addResult.accepted.some(h => h.equals(txHash));
946
+ const wasIgnored = addResult.ignored.some(h => h.equals(txHash));
765
947
 
766
948
  this.logger.trace(`Validate propagated tx`, {
767
- isValid,
768
- exists,
949
+ wasAccepted,
950
+ wasIgnored,
769
951
  [Attributes.P2P_ID]: source.toString(),
770
952
  });
771
953
 
772
- if (!isValid) {
773
- return { result: TopicValidatorResult.Reject };
774
- } else if (exists) {
954
+ if (wasAccepted) {
955
+ return { result: TopicValidatorResult.Accept, obj: tx };
956
+ } else if (wasIgnored) {
775
957
  return { result: TopicValidatorResult.Ignore, obj: tx };
776
958
  } else {
777
- return { result: TopicValidatorResult.Accept, obj: tx };
959
+ return { result: TopicValidatorResult.Reject };
778
960
  }
779
961
  };
780
962
 
@@ -783,6 +965,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
783
965
  return;
784
966
  }
785
967
 
968
+ // Tx was accepted into pool and will be propagated - just log and record metrics
786
969
  const txHash = tx.getTxHash();
787
970
  const txHashString = txHash.toString();
788
971
  this.logger.verbose(`Received tx ${txHashString} from external peer ${source.toString()} via gossip`, {
@@ -790,73 +973,31 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
790
973
  txHash: txHashString,
791
974
  });
792
975
 
793
- if (this.config.dropTransactions && randomInt(1000) < this.config.dropTransactionsProbability * 1000) {
794
- this.logger.warn(`Intentionally dropping tx ${txHashString} (probability rule)`);
795
- return;
796
- }
797
-
798
976
  this.instrumentation.incrementTxReceived(1);
799
- await this.mempools.txPool.addTxs([tx]);
800
977
  }
801
978
 
802
979
  /**
803
- * Process Attestation From Peer
804
- * When a proposal is received from a peer, we add it to the attestation pool, so it can be accessed by other services.
805
- *
806
- * @param attestation - The attestation to process.
980
+ * Process a checkpoint attestation from a peer.
981
+ * Validates the attestation and adds it to the pool.
807
982
  */
808
- private async processAttestationFromPeer(payloadData: Buffer, msgId: string, source: PeerId): Promise<void> {
809
- const validationFunc: () => Promise<ReceivedMessageValidationResult<BlockAttestation>> = async () => {
810
- const attestation = BlockAttestation.fromBuffer(payloadData);
811
- const pool = this.mempools.attestationPool;
812
- const isValid = await this.validateAttestation(source, attestation);
813
- const exists = isValid && (await pool.hasAttestation(attestation));
814
-
815
- let canAdd = true;
816
- if (isValid && !exists) {
817
- const slot = attestation.payload.header.slotNumber;
818
- const { committee } = await this.epochCache.getCommittee(slot);
819
- const committeeSize = committee?.length ?? 0;
820
- canAdd = await pool.canAddAttestation(attestation, committeeSize);
821
- }
822
-
823
- this.logger.trace(`Validate propagated block attestation`, {
824
- isValid,
825
- exists,
826
- canAdd,
827
- [Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber.toString(),
828
- [Attributes.P2P_ID]: source.toString(),
829
- });
830
-
831
- if (!isValid) {
832
- return { result: TopicValidatorResult.Reject };
833
- } else if (exists) {
834
- return { result: TopicValidatorResult.Ignore, obj: attestation };
835
- } else if (!canAdd) {
836
- this.logger.warn(`Dropping block attestation due to per-(slot, proposalId) attestation cap`, {
837
- slot: attestation.payload.header.slotNumber.toString(),
838
- archive: attestation.archive.toString(),
839
- source: source.toString(),
840
- });
841
- return { result: TopicValidatorResult.Ignore, obj: attestation };
842
- } else {
843
- return { result: TopicValidatorResult.Accept, obj: attestation };
844
- }
845
- };
846
-
847
- const { result, obj: attestation } = await this.validateReceivedMessage<BlockAttestation>(
848
- validationFunc,
983
+ private async processCheckpointAttestationFromPeer(
984
+ payloadData: Buffer,
985
+ msgId: string,
986
+ source: PeerId,
987
+ ): Promise<void> {
988
+ const { result, obj: attestation } = await this.validateReceivedMessage<CheckpointAttestation>(
989
+ () => this.validateAndStoreCheckpointAttestation(source, CheckpointAttestation.fromBuffer(payloadData)),
849
990
  msgId,
850
991
  source,
851
- TopicType.block_attestation,
992
+ TopicType.checkpoint_attestation,
852
993
  );
853
994
 
854
995
  if (result !== TopicValidatorResult.Accept || !attestation) {
855
996
  return;
856
997
  }
857
998
 
858
- this.logger.debug(
859
- `Received attestation for slot ${attestation.slotNumber} from external peer ${source.toString()}`,
999
+ this.logger.verbose(
1000
+ `Received valid checkpoint attestation for slot ${attestation.slotNumber} from external peer ${source.toString()}`,
860
1001
  {
861
1002
  p2pMessageIdentifier: await attestation.p2pMessageLoggingIdentifier(),
862
1003
  slot: attestation.slotNumber,
@@ -864,119 +1005,354 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
864
1005
  source: source.toString(),
865
1006
  },
866
1007
  );
867
-
868
- await this.mempools.attestationPool.addAttestations([attestation]);
869
1008
  }
870
1009
 
871
- private async processBlockFromPeer(payloadData: Buffer, msgId: string, source: PeerId): Promise<void> {
872
- const validationFunc: () => Promise<ReceivedMessageValidationResult<BlockProposal>> = async () => {
873
- const block = BlockProposal.fromBuffer(payloadData);
874
- const isValid = await this.validateBlockProposal(source, block);
875
- const pool = this.mempools.attestationPool;
1010
+ /** Validates a checkpoint attestation and adds it to the pool. Penalizes the peer if validation fails. */
1011
+ @trackSpan('Libp2pService.validateAndStoreCheckpointAttestation', (_peerId, attestation) => ({
1012
+ [Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber.toString(),
1013
+ }))
1014
+ protected async validateAndStoreCheckpointAttestation(
1015
+ peerId: PeerId,
1016
+ attestation: CheckpointAttestation,
1017
+ ): Promise<ReceivedMessageValidationResult<CheckpointAttestation>> {
1018
+ const validationResult = await this.checkpointAttestationValidator.validate(attestation);
1019
+
1020
+ if (validationResult.result === 'reject') {
1021
+ this.logger.warn(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
1022
+ this.peerManager.penalizePeer(peerId, validationResult.severity);
1023
+ return { result: TopicValidatorResult.Reject };
1024
+ }
1025
+
1026
+ if (validationResult.result === 'ignore') {
1027
+ return { result: TopicValidatorResult.Ignore, obj: attestation };
1028
+ }
876
1029
 
877
- const exists = isValid && (await pool.hasBlockProposal(block));
878
- const canAdd = isValid && (await pool.canAddProposal(block));
1030
+ // Try to add the attestation: this handles existence check, cap check, and adding in one call
1031
+ // count is the number of attestations by this signer for this slot (for duplicate detection)
1032
+ const slot = attestation.payload.header.slotNumber;
1033
+ const { added, alreadyExists, count } =
1034
+ await this.mempools.attestationPool.tryAddCheckpointAttestation(attestation);
1035
+
1036
+ this.logger.trace(`Validate propagated checkpoint attestation`, {
1037
+ added,
1038
+ alreadyExists,
1039
+ count,
1040
+ [Attributes.SLOT_NUMBER]: slot.toString(),
1041
+ [Attributes.P2P_ID]: peerId.toString(),
1042
+ });
879
1043
 
880
- this.logger.trace(`Validate propagated block proposal`, {
881
- isValid,
882
- exists,
883
- canAdd,
884
- [Attributes.SLOT_NUMBER]: block.payload.header.slotNumber.toString(),
885
- [Attributes.P2P_ID]: source.toString(),
1044
+ // Exact same attestation received, no need to re-broadcast
1045
+ if (alreadyExists) {
1046
+ return { result: TopicValidatorResult.Ignore, obj: attestation };
1047
+ }
1048
+
1049
+ // Could not add (cap reached for signer), no need to re-broadcast
1050
+ if (!added) {
1051
+ this.logger.warn(`Dropping checkpoint attestation due to cap`, {
1052
+ slot: slot.toString(),
1053
+ archive: attestation.archive.toString(),
1054
+ source: peerId.toString(),
1055
+ attester: attestation.getSender()?.toString(),
1056
+ count,
886
1057
  });
1058
+ return { result: TopicValidatorResult.Ignore, obj: attestation };
1059
+ }
887
1060
 
888
- if (!isValid) {
889
- return { result: TopicValidatorResult.Reject };
890
- } else if (exists) {
891
- return { result: TopicValidatorResult.Ignore, obj: block };
892
- } else if (!canAdd) {
893
- this.peerManager.penalizePeer(source, PeerErrorSeverity.MidToleranceError);
894
- this.logger.warn(`Penalizing peer for block proposal exceeding per-slot cap`, {
895
- slot: block.slotNumber.toString(),
896
- archive: block.archive.toString(),
897
- source: source.toString(),
1061
+ // Check if this is a duplicate attestation (signer attested to a different proposal at the same slot)
1062
+ // count is the number of attestations by this signer for this slot
1063
+ if (count === 2) {
1064
+ const attester = attestation.getSender();
1065
+ if (attester) {
1066
+ this.logger.warn(`Detected duplicate attestation (equivocation) at slot ${slot}`, {
1067
+ slot: slot.toString(),
1068
+ archive: attestation.archive.toString(),
1069
+ source: peerId.toString(),
1070
+ attester: attester.toString(),
898
1071
  });
899
- return { result: TopicValidatorResult.Reject };
900
- } else {
901
- return { result: TopicValidatorResult.Accept, obj: block };
1072
+ this.duplicateAttestationCallback?.({ slot, attester });
902
1073
  }
903
- };
1074
+ }
1075
+
1076
+ // Attestation was added successfully - accept it so other nodes can also detect the equivocation
1077
+ return { result: TopicValidatorResult.Accept, obj: attestation };
1078
+ }
904
1079
 
905
- const { result, obj: block } = await this.validateReceivedMessage<BlockProposal>(
906
- validationFunc,
1080
+ protected async processBlockFromPeer(payloadData: Buffer, msgId: string, source: PeerId): Promise<void> {
1081
+ const {
1082
+ result,
1083
+ obj: block,
1084
+ metadata: { isEquivocated } = {},
1085
+ } = await this.validateReceivedMessage<BlockProposal, { isEquivocated: boolean }>(
1086
+ () => this.validateAndStoreBlockProposal(source, BlockProposal.fromBuffer(payloadData)),
907
1087
  msgId,
908
1088
  source,
909
1089
  TopicType.block_proposal,
910
1090
  );
911
1091
 
912
- if (!result || !block) {
1092
+ // If not accepted or equivocated, return
1093
+ if (result !== TopicValidatorResult.Accept || !block || isEquivocated) {
913
1094
  return;
914
1095
  }
915
1096
 
916
1097
  await this.processValidBlockProposal(block, source);
917
1098
  }
918
1099
 
919
- // REVIEW: callback pattern https://github.com/AztecProtocol/aztec-packages/issues/7963
1100
+ /** Validates a block proposal. Triggers a penalization to the peer that sent it if invalid. Adds to the mempool if valid. */
1101
+ @trackSpan('Libp2pService.validateAndStoreBlockProposal', (_peerId, block) => ({
1102
+ [Attributes.BLOCK_NUMBER]: block.blockNumber.toString(),
1103
+ [Attributes.SLOT_NUMBER]: block.slotNumber.toString(),
1104
+ }))
1105
+ protected async validateAndStoreBlockProposal(
1106
+ peerId: PeerId,
1107
+ block: BlockProposal,
1108
+ ): Promise<ReceivedMessageValidationResult<BlockProposal, { isEquivocated: boolean }>> {
1109
+ const validationResult = await this.blockProposalValidator.validate(block);
1110
+
1111
+ if (validationResult.result === 'reject') {
1112
+ this.logger.warn(`Penalizing peer ${peerId} for block proposal validation failure`);
1113
+ this.peerManager.penalizePeer(peerId, validationResult.severity);
1114
+ return { result: TopicValidatorResult.Reject };
1115
+ }
1116
+
1117
+ if (validationResult.result === 'ignore') {
1118
+ return { result: TopicValidatorResult.Ignore, obj: block };
1119
+ }
1120
+
1121
+ // Try to add the proposal: this handles existence check, cap check, and adding in one call
1122
+ const { added, alreadyExists, count } = await this.mempools.attestationPool.tryAddBlockProposal(block);
1123
+ const isEquivocated = count !== undefined && count > 1;
1124
+
1125
+ // Duplicate proposal received, no need to re-broadcast
1126
+ if (alreadyExists) {
1127
+ this.logger.debug(`Ignoring duplicate block proposal received`, {
1128
+ ...block.toBlockInfo(),
1129
+ indexWithinCheckpoint: block.indexWithinCheckpoint,
1130
+ proposer: block.getSender()?.toString(),
1131
+ source: peerId.toString(),
1132
+ });
1133
+ return { result: TopicValidatorResult.Ignore, obj: block, metadata: { isEquivocated } };
1134
+ }
1135
+
1136
+ // Too many blocks received for this slot and index, penalize peer and do not re-broadcast
1137
+ if (!added) {
1138
+ this.peerManager.penalizePeer(peerId, PeerErrorSeverity.HighToleranceError);
1139
+ this.logger.warn(`Penalizing peer for block proposal exceeding per-position cap`, {
1140
+ ...block.toBlockInfo(),
1141
+ indexWithinCheckpoint: block.indexWithinCheckpoint,
1142
+ count,
1143
+ proposer: block.getSender()?.toString(),
1144
+ source: peerId.toString(),
1145
+ });
1146
+ return { result: TopicValidatorResult.Reject, metadata: { isEquivocated } };
1147
+ }
1148
+
1149
+ // If this was a duplicate proposal, do not process it, but do invoke the duplicate callback,
1150
+ // and do re-broadcast it so other nodes in the network know to slash the proposer
1151
+ if (isEquivocated) {
1152
+ const proposer = block.getSender();
1153
+ this.logger.warn(`Detected duplicate block proposal (equivocation) at slot ${block.slotNumber}`, {
1154
+ ...block.toBlockInfo(),
1155
+ source: peerId.toString(),
1156
+ proposer: proposer?.toString(),
1157
+ });
1158
+ // Invoke the duplicate callback on the first duplicate spotted only
1159
+ if (proposer && count === 2) {
1160
+ this.duplicateProposalCallback?.({ slot: block.slotNumber, proposer, type: 'block' });
1161
+ }
1162
+ return { result: TopicValidatorResult.Accept, obj: block, metadata: { isEquivocated } };
1163
+ }
1164
+
1165
+ // Otherwise, we're good to go!
1166
+ return { result: TopicValidatorResult.Accept, obj: block };
1167
+ }
1168
+
1169
+ // REFACTOR(palla): This method should be moved to the p2p_client or to a separate component,
1170
+ // should not be here as it does not deal with p2p networking.
920
1171
  @trackSpan('Libp2pService.processValidBlockProposal', async block => ({
921
1172
  [Attributes.SLOT_NUMBER]: block.slotNumber,
922
1173
  [Attributes.BLOCK_ARCHIVE]: block.archive.toString(),
923
1174
  [Attributes.P2P_ID]: await block.p2pMessageLoggingIdentifier().then(i => i.toString()),
924
1175
  }))
925
- private async processValidBlockProposal(block: BlockProposal, sender: PeerId) {
1176
+ protected async processValidBlockProposal(block: BlockProposal, sender: PeerId) {
926
1177
  const slot = block.slotNumber;
927
- const previousSlot = SlotNumber(slot - 1);
928
1178
  this.logger.verbose(`Received block proposal for slot ${slot} from external peer ${sender.toString()}.`, {
929
1179
  p2pMessageIdentifier: await block.p2pMessageLoggingIdentifier(),
930
- slot: block.slotNumber,
931
- archive: block.archive.toString(),
932
1180
  source: sender.toString(),
1181
+ ...block.toBlockInfo(),
933
1182
  });
934
- const attestationsForPreviousSlot = await this.mempools.attestationPool.getAttestationsForSlot(previousSlot);
935
- this.logger.verbose(`Received ${attestationsForPreviousSlot.length} attestations for slot ${previousSlot}`);
936
1183
 
937
- // Attempt to add proposal, then mark the txs in this proposal as non-evictable
938
- try {
939
- await this.mempools.attestationPool.addBlockProposal(block);
940
- } catch (err: unknown) {
941
- // Drop proposals if we hit per-slot cap in the attestation pool; rethrow unknown errors
942
- if (err instanceof ProposalSlotCapExceededError) {
943
- this.logger.warn(`Dropping block proposal due to per-slot proposal cap`, {
944
- slot: String(slot),
945
- archive: block.archive.toString(),
946
- error: (err as Error).message,
1184
+ // Mark the txs in this proposal as protected
1185
+ await this.mempools.txPool.protectTxs(block.txHashes, block.blockHeader);
1186
+
1187
+ // Call the block received callback to validate the proposal.
1188
+ // Note: Validators do NOT attest to individual blocks, only to checkpoint proposals.
1189
+ const isValid = await this.blockReceivedCallback(block, sender);
1190
+ if (!isValid) {
1191
+ this.logger.warn(`Block proposal validation failed for block ${block.blockNumber}`, block.toBlockInfo());
1192
+ }
1193
+ }
1194
+
1195
+ /**
1196
+ * Handle a gossiped checkpoint proposal.
1197
+ * Validates and processes the checkpoint proposal, then triggers the callback for attestation.
1198
+ */
1199
+ protected async handleGossipedCheckpointProposal(payloadData: Buffer, msgId: string, source: PeerId): Promise<void> {
1200
+ const {
1201
+ result,
1202
+ obj: checkpoint,
1203
+ metadata: { isEquivocated, processBlock } = {},
1204
+ } = await this.validateReceivedMessage<CheckpointProposal, { isEquivocated: boolean; processBlock: boolean }>(
1205
+ () => this.validateAndStoreCheckpointProposal(source, CheckpointProposal.fromBuffer(payloadData)),
1206
+ msgId,
1207
+ source,
1208
+ TopicType.checkpoint_proposal,
1209
+ );
1210
+
1211
+ // If the checkpoint contained a valid last block, we process it even if the checkpoint itself is to be rejected
1212
+ // TODO(palla/mbps): Is this ok? Should we be considering a block from a checkpoint that was equivocated?
1213
+ if (processBlock && checkpoint?.getBlockProposal()) {
1214
+ await this.processValidBlockProposal(checkpoint.getBlockProposal()!, source);
1215
+ }
1216
+
1217
+ if (result !== TopicValidatorResult.Accept || !checkpoint || isEquivocated) {
1218
+ return;
1219
+ }
1220
+
1221
+ await this.processValidCheckpointProposal(checkpoint.toCore(), source);
1222
+ }
1223
+
1224
+ /**
1225
+ * Validates a checkpoint proposal. Penalizes peer if validation fails. Adds the checkpoint and
1226
+ * its last block (if present) to the mempool if valid. Triggers equivocation detection on both.
1227
+ */
1228
+ @trackSpan('Libp2pService.validateAndStoreCheckpointProposal', (_peerId, checkpoint) => ({
1229
+ [Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
1230
+ }))
1231
+ protected async validateAndStoreCheckpointProposal(
1232
+ peerId: PeerId,
1233
+ checkpoint: CheckpointProposal,
1234
+ ): Promise<ReceivedMessageValidationResult<CheckpointProposal, { isEquivocated: boolean; processBlock: boolean }>> {
1235
+ const validationResult = await this.checkpointProposalValidator.validate(checkpoint);
1236
+
1237
+ if (validationResult.result === 'reject') {
1238
+ this.logger.warn(`Penalizing peer ${peerId} for checkpoint proposal validation failure`);
1239
+ this.peerManager.penalizePeer(peerId, validationResult.severity);
1240
+ return { result: TopicValidatorResult.Reject };
1241
+ }
1242
+
1243
+ if (validationResult.result === 'ignore') {
1244
+ return { result: TopicValidatorResult.Ignore, obj: checkpoint };
1245
+ }
1246
+
1247
+ // Extract and try to add the block proposal first if present
1248
+ const blockProposal = checkpoint.getBlockProposal();
1249
+ let processBlock = false;
1250
+ if (blockProposal) {
1251
+ this.logger.debug(`Validating block proposal from propagated checkpoint`, {
1252
+ [Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
1253
+ [Attributes.P2P_ID]: peerId.toString(),
1254
+ });
1255
+ const {
1256
+ result,
1257
+ obj,
1258
+ metadata: { isEquivocated } = {},
1259
+ } = await this.validateAndStoreBlockProposal(peerId, blockProposal);
1260
+ if (result === TopicValidatorResult.Reject || !obj || isEquivocated) {
1261
+ this.logger.debug(`Rejecting checkpoint due to invalid last block proposal`, {
1262
+ [Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
1263
+ [Attributes.P2P_ID]: peerId.toString(),
1264
+ isEquivocated,
1265
+ result,
947
1266
  });
948
- return;
1267
+ return { result: TopicValidatorResult.Reject };
1268
+ } else if (result === TopicValidatorResult.Accept && obj && !isEquivocated) {
1269
+ processBlock = true;
949
1270
  }
950
- throw err;
951
1271
  }
952
- await this.mempools.txPool.markTxsAsNonEvictable(block.txHashes);
953
- const attestations = await this.blockReceivedCallback(block, sender);
954
1272
 
955
- // TODO: fix up this pattern - the abstraction is not nice
956
- // The attestation can be undefined if no handler is registered / the validator deems the block invalid / in fisherman mode
957
- if (attestations?.length) {
958
- for (const attestation of attestations) {
959
- this.logger.verbose(`Broadcasting attestation for slot ${attestation.slotNumber}`, {
960
- p2pMessageIdentifier: await attestation.p2pMessageLoggingIdentifier(),
961
- slot: attestation.slotNumber,
962
- archive: attestation.archive.toString(),
963
- });
964
- await this.broadcastAttestation(attestation);
1273
+ // Try to add the checkpoint proposal core: this handles existence check, cap check, and adding in one call
1274
+ const checkpointCore = checkpoint.toCore();
1275
+ const tryAddResult = await this.mempools.attestationPool.tryAddCheckpointProposal(checkpointCore);
1276
+ const { added, alreadyExists, count } = tryAddResult;
1277
+ const isEquivocated = count !== undefined && count > 1;
1278
+
1279
+ // Duplicate proposal received, do not re-broadcast
1280
+ if (alreadyExists) {
1281
+ this.logger.debug(`Ignoring duplicate checkpoint proposal received`, {
1282
+ ...checkpoint.toCheckpointInfo(),
1283
+ source: peerId.toString(),
1284
+ });
1285
+ return {
1286
+ result: TopicValidatorResult.Ignore,
1287
+ obj: checkpoint,
1288
+ metadata: { isEquivocated, processBlock },
1289
+ };
1290
+ }
1291
+
1292
+ // Too many checkpoint proposals received for this slot, penalize peer and do not re-broadcast
1293
+ // Note: We still return the checkpoint obj so the lastBlock can be processed if valid
1294
+ if (!added) {
1295
+ this.peerManager.penalizePeer(peerId, PeerErrorSeverity.HighToleranceError);
1296
+ this.logger.warn(`Penalizing peer for checkpoint proposal exceeding per-slot cap`, {
1297
+ ...checkpoint.toCheckpointInfo(),
1298
+ count,
1299
+ source: peerId.toString(),
1300
+ });
1301
+ return { result: TopicValidatorResult.Reject, obj: checkpoint, metadata: { isEquivocated, processBlock } };
1302
+ }
1303
+
1304
+ // If this was a duplicate proposal, do not process it, but do invoke the duplicate callback,
1305
+ // and do re-broadcast it so other nodes in the network know to slash the proposer
1306
+ if (isEquivocated) {
1307
+ const proposer = checkpoint.getSender();
1308
+ this.logger.warn(`Detected duplicate checkpoint proposal (equivocation) at slot ${checkpoint.slotNumber}`, {
1309
+ ...checkpoint.toCheckpointInfo(),
1310
+ source: peerId.toString(),
1311
+ proposer: proposer?.toString(),
1312
+ });
1313
+ // Invoke the duplicate callback on the first duplicate spotted only
1314
+ if (proposer && count === 2) {
1315
+ this.duplicateProposalCallback?.({ slot: checkpoint.slotNumber, proposer, type: 'checkpoint' });
965
1316
  }
1317
+ return {
1318
+ result: TopicValidatorResult.Accept,
1319
+ obj: checkpoint,
1320
+ metadata: { isEquivocated, processBlock },
1321
+ };
966
1322
  }
1323
+
1324
+ // Otherwise, we're good to go!
1325
+ return { result: TopicValidatorResult.Accept, obj: checkpoint, metadata: { processBlock, isEquivocated } };
967
1326
  }
968
1327
 
969
1328
  /**
970
- * Broadcast an attestation to all peers.
971
- * @param attestation - The attestation to broadcast.
1329
+ * Process a validated checkpoint proposal.
1330
+ * Note: The proposal was already added to the pool by tryAddCheckpointProposal in handleGossipedCheckpointProposal.
972
1331
  */
973
- @trackSpan('Libp2pService.broadcastAttestation', async attestation => ({
974
- [Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber,
975
- [Attributes.BLOCK_ARCHIVE]: attestation.archive.toString(),
976
- [Attributes.P2P_ID]: await attestation.p2pMessageLoggingIdentifier().then(i => i.toString()),
1332
+ @trackSpan('Libp2pService.processValidCheckpointProposal', async checkpoint => ({
1333
+ [Attributes.SLOT_NUMBER]: checkpoint.slotNumber,
1334
+ [Attributes.BLOCK_ARCHIVE]: checkpoint.archive.toString(),
1335
+ [Attributes.P2P_ID]: await checkpoint.p2pMessageLoggingIdentifier().then(i => i.toString()),
977
1336
  }))
978
- private async broadcastAttestation(attestation: BlockAttestation) {
979
- await this.propagate(attestation);
1337
+ protected async processValidCheckpointProposal(checkpoint: CheckpointProposalCore, sender: PeerId) {
1338
+ const slot = checkpoint.slotNumber;
1339
+ this.logger.verbose(`Received checkpoint proposal for slot ${slot} from external peer ${sender.toString()}.`, {
1340
+ p2pMessageIdentifier: await checkpoint.p2pMessageLoggingIdentifier(),
1341
+ slot: checkpoint.slotNumber,
1342
+ archive: checkpoint.archive.toString(),
1343
+ source: sender.toString(),
1344
+ });
1345
+
1346
+ // Call the checkpoint received callback with the core version (without lastBlock)
1347
+ // to validate and potentially generate attestations
1348
+ const attestations = await this.checkpointReceivedCallback(checkpoint, sender);
1349
+ if (attestations && attestations.length > 0) {
1350
+ // If the callback returned attestations, add them to the pool and propagate them
1351
+ await this.mempools.attestationPool.addOwnCheckpointAttestations(attestations);
1352
+ for (const attestation of attestations) {
1353
+ await this.propagate(attestation);
1354
+ }
1355
+ }
980
1356
  }
981
1357
 
982
1358
  /**
@@ -999,9 +1375,9 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
999
1375
  * @returns True if the requested block transactions are valid, false otherwise.
1000
1376
  */
1001
1377
  @trackSpan('Libp2pService.validateRequestedBlockTxs', request => ({
1002
- [Attributes.BLOCK_HASH]: request.blockHash.toString(),
1378
+ [Attributes.BLOCK_ARCHIVE]: request.archiveRoot.toString(),
1003
1379
  }))
1004
- private async validateRequestedBlockTxs(
1380
+ protected async validateRequestedBlockTxs(
1005
1381
  request: BlockTxsRequest,
1006
1382
  response: BlockTxsResponse,
1007
1383
  peerId: PeerId,
@@ -1009,10 +1385,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1009
1385
  const requestedTxValidator = this.createRequestedTxValidator();
1010
1386
 
1011
1387
  try {
1012
- if (!response.blockHash.equals(request.blockHash)) {
1388
+ if (!response.archiveRoot.equals(request.archiveRoot)) {
1013
1389
  this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
1014
1390
  throw new ValidationError(
1015
- `Received block txs for unexpected block: expected ${request.blockHash.toString()}, got ${response.blockHash.toString()}`,
1391
+ `Received block txs for unexpected archive root: expected ${request.archiveRoot.toString()}, got ${response.archiveRoot.toString()}`,
1016
1392
  );
1017
1393
  }
1018
1394
 
@@ -1042,7 +1418,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1042
1418
  }
1043
1419
 
1044
1420
  // Given proposal (should have locally), ensure returned txs are valid subset and match request indices
1045
- const proposal = await this.mempools.attestationPool.getBlockProposal(request.blockHash.toString());
1421
+ const proposal = await this.mempools.attestationPool.getBlockProposal(request.archiveRoot.toString());
1046
1422
  if (proposal) {
1047
1423
  // Build intersected indices
1048
1424
  const intersectIdx = request.txIndices.getTrueIndices().filter(i => response.txIndices.isSet(i));
@@ -1062,7 +1438,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1062
1438
  } else {
1063
1439
  // No local proposal, cannot check the membership/order of the returned txs
1064
1440
  this.logger.warn(
1065
- `Block proposal not found for block hash ${request.blockHash.toString()}; cannot validate membership/order of returned txs`,
1441
+ `Block proposal not found for archive root ${request.archiveRoot.toString()}; cannot validate membership/order of returned txs`,
1066
1442
  );
1067
1443
  return false;
1068
1444
  }
@@ -1101,7 +1477,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1101
1477
  const requested = new Set(requestedTxHash.map(h => h.toString()));
1102
1478
  const requestedTxValidator = this.createRequestedTxValidator();
1103
1479
 
1104
- //TODO: (mralj) - this is somewhat naive implementation, if single tx is invlid we consider the whole response invalid.
1480
+ //TODO: (mralj) - this is somewhat naive implementation, if single tx is invalid we consider the whole response invalid.
1105
1481
  // I think we should still extract the valid txs and return them, so that we can still use the response.
1106
1482
  try {
1107
1483
  await Promise.all(responseTx.map(tx => this.validateRequestedTx(tx, peerId, requestedTxValidator, requested)));
@@ -1131,7 +1507,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1131
1507
  @trackSpan('Libp2pService.validateRequestedBlock', (requestedBlockNumber, _responseBlock) => ({
1132
1508
  [Attributes.BLOCK_NUMBER]: requestedBlockNumber.toString(),
1133
1509
  }))
1134
- private async validateRequestedBlock(
1510
+ protected async validateRequestedBlock(
1135
1511
  requestedBlockNumber: Fr,
1136
1512
  responseBlock: L2Block,
1137
1513
  peerId: PeerId,
@@ -1164,27 +1540,13 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1164
1540
  }
1165
1541
  }
1166
1542
 
1167
- private createRequestedTxValidator(): TxValidator {
1168
- return new AggregateTxValidator(
1169
- new DataTxValidator(),
1170
- new MetadataTxValidator({
1171
- l1ChainId: new Fr(this.config.l1ChainId),
1172
- rollupVersion: new Fr(this.config.rollupVersion),
1173
- protocolContractsHash,
1174
- vkTreeRoot: getVKTreeRoot(),
1175
- }),
1176
- new TxProofValidator(this.proofVerifier),
1177
- );
1178
- }
1179
-
1180
- private async validateRequestedTx(tx: Tx, peerId: PeerId, txValidator: TxValidator, requested?: Set<`0x${string}`>) {
1543
+ protected async validateRequestedTx(
1544
+ tx: Tx,
1545
+ peerId: PeerId,
1546
+ txValidator: TxValidator,
1547
+ requested?: Set<`0x${string}`>,
1548
+ ) {
1181
1549
  const penalize = (severity: PeerErrorSeverity) => this.peerManager.penalizePeer(peerId, severity);
1182
-
1183
- if (!(await tx.validateTxHash())) {
1184
- penalize(PeerErrorSeverity.MidToleranceError);
1185
- throw new ValidationError(`Received tx with invalid hash ${tx.getTxHash().toString()}.`);
1186
- }
1187
-
1188
1550
  if (requested && !requested.has(tx.getTxHash().toString())) {
1189
1551
  penalize(PeerErrorSeverity.MidToleranceError);
1190
1552
  throw new ValidationError(`Received tx with hash ${tx.getTxHash().toString()} that was not requested.`);
@@ -1197,35 +1559,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1197
1559
  }
1198
1560
  }
1199
1561
 
1200
- @trackSpan('Libp2pService.validatePropagatedTx', tx => ({
1201
- [Attributes.TX_HASH]: tx.getTxHash().toString(),
1202
- }))
1203
- private async validatePropagatedTx(tx: Tx, peerId: PeerId): Promise<boolean> {
1204
- const currentBlockNumber = await this.archiver.getBlockNumber();
1205
-
1206
- // We accept transactions if they are not expired by the next slot (checked based on the IncludeByTimestamp field)
1207
- const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
1208
- const messageValidators = await this.createMessageValidators(currentBlockNumber, nextSlotTimestamp);
1209
-
1210
- for (const validator of messageValidators) {
1211
- const outcome = await this.runValidations(tx, validator);
1212
-
1213
- if (outcome.allPassed) {
1214
- continue;
1215
- }
1216
- const { name } = outcome.failure;
1217
- let { severity } = outcome.failure;
1218
-
1219
- // Double spend validator has a special case handler
1220
- if (name === 'doubleSpendValidator') {
1221
- const txBlockNumber = BlockNumber(currentBlockNumber + 1); // tx is expected to be in the next block
1222
- severity = await this.handleDoubleSpendFailure(tx, txBlockNumber);
1223
- }
1224
-
1225
- this.peerManager.penalizePeer(peerId, severity);
1226
- return false;
1227
- }
1228
- return true;
1562
+ protected createRequestedTxValidator(): TxValidator {
1563
+ return createTxValidatorForReqResponseReceivedTxs(this.proofVerifier, {
1564
+ l1ChainId: this.config.l1ChainId,
1565
+ rollupVersion: this.config.rollupVersion,
1566
+ });
1229
1567
  }
1230
1568
 
1231
1569
  private async getGasFees(blockNumber: BlockNumber): Promise<GasFees> {
@@ -1239,59 +1577,69 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1239
1577
  return gasFees;
1240
1578
  }
1241
1579
 
1242
- public async validate(txs: Tx[]): Promise<void> {
1243
- const currentBlockNumber = await this.archiver.getBlockNumber();
1580
+ /**
1581
+ * Get the BatchTxRequesterLibP2PService dependencies for creating BatchTxRequester instances
1582
+ */
1583
+ public getBatchTxRequesterService(): BatchTxRequesterLibP2PService {
1584
+ return {
1585
+ reqResp: this.reqresp,
1586
+ connectionSampler: this.reqresp.getConnectionSampler(),
1587
+ txValidatorConfig: {
1588
+ l1ChainId: this.config.l1ChainId,
1589
+ rollupVersion: this.config.rollupVersion,
1590
+ proofVerifier: this.proofVerifier,
1591
+ },
1592
+ peerScoring: this.peerManager,
1593
+ };
1594
+ }
1244
1595
 
1245
- // We accept transactions if they are not expired by the next slot (checked based on the IncludeByTimestamp field)
1246
- const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
1247
- const messageValidators = await this.createMessageValidators(currentBlockNumber, nextSlotTimestamp);
1596
+ public async validateTxsReceivedInBlockProposal(txs: Tx[]): Promise<void> {
1597
+ const validator = createTxValidatorForBlockProposalReceivedTxs(
1598
+ this.proofVerifier,
1599
+ { l1ChainId: this.config.l1ChainId, rollupVersion: this.config.rollupVersion },
1600
+ this.logger.getBindings(),
1601
+ );
1248
1602
 
1249
- await Promise.all(
1603
+ const results = await Promise.all(
1250
1604
  txs.map(async tx => {
1251
- for (const validator of messageValidators) {
1252
- const outcome = await this.runValidations(tx, validator);
1253
- if (!outcome.allPassed) {
1254
- throw new Error('Invalid tx detected', { cause: { outcome } });
1255
- }
1256
- }
1605
+ const result = await validator.validateTx(tx);
1606
+ return result.result !== 'invalid';
1257
1607
  }),
1258
1608
  );
1609
+ if (results.some(value => value === false)) {
1610
+ throw new Error('Invalid tx detected');
1611
+ }
1259
1612
  }
1260
1613
 
1261
- /**
1262
- * Create message validators for the given block number and timestamp.
1263
- *
1264
- * Each validator is a pair of a validator and a severity.
1265
- * If a validator fails, the peer is penalized with the severity of the validator.
1266
- *
1267
- * @param currentBlockNumber - The current synced block number.
1268
- * @param nextSlotTimestamp - The timestamp of the next slot (used to validate txs are not expired).
1269
- * @returns The message validators.
1270
- */
1271
- private async createMessageValidators(
1614
+ /** Creates the first stage (fast) validators for gossiped transactions. */
1615
+ protected async createFirstStageMessageValidators(
1272
1616
  currentBlockNumber: BlockNumber,
1273
1617
  nextSlotTimestamp: UInt64,
1274
- ): Promise<Record<string, MessageValidator>[]> {
1618
+ ): Promise<Record<string, TransactionValidator>> {
1275
1619
  const gasFees = await this.getGasFees(currentBlockNumber);
1276
1620
  const allowedInSetup = this.config.txPublicSetupAllowList ?? (await getDefaultAllowedSetupFunctions());
1621
+ const blockNumber = BlockNumber(currentBlockNumber + 1);
1277
1622
 
1278
- const blockNumberInWhichTheTxIsConsideredToBeIncluded = BlockNumber(currentBlockNumber + 1);
1279
-
1280
- return createTxMessageValidators(
1623
+ return createFirstStageTxValidationsForGossipedTransactions(
1281
1624
  nextSlotTimestamp,
1282
- blockNumberInWhichTheTxIsConsideredToBeIncluded,
1625
+ blockNumber,
1283
1626
  this.worldStateSynchronizer,
1284
1627
  gasFees,
1285
1628
  this.config.l1ChainId,
1286
1629
  this.config.rollupVersion,
1287
1630
  protocolContractsHash,
1288
1631
  this.archiver,
1289
- this.proofVerifier,
1290
1632
  !this.config.disableTransactions,
1291
1633
  allowedInSetup,
1634
+ this.logger.getBindings(),
1292
1635
  );
1293
1636
  }
1294
1637
 
1638
+ /** Creates the second stage (expensive proof verification) validators for gossiped transactions. */
1639
+ protected createSecondStageMessageValidators(): Record<string, TransactionValidator> {
1640
+ return createSecondStageTxValidationsForGossipedTransactions(this.proofVerifier, this.logger.getBindings());
1641
+ }
1642
+
1295
1643
  /**
1296
1644
  * Run validations on a tx.
1297
1645
  * @param tx - The tx to validate.
@@ -1300,7 +1648,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1300
1648
  */
1301
1649
  private async runValidations(
1302
1650
  tx: Tx,
1303
- messageValidators: Record<string, MessageValidator>,
1651
+ messageValidators: Record<string, TransactionValidator>,
1304
1652
  ): Promise<ValidationOutcome> {
1305
1653
  const validationPromises = Object.entries(messageValidators).map(async ([name, { validator, severity }]) => {
1306
1654
  const { result } = await validator.validateTx(tx);
@@ -1342,15 +1690,18 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1342
1690
  return PeerErrorSeverity.HighToleranceError;
1343
1691
  }
1344
1692
 
1345
- const snapshotValidator = new DoubleSpendTxValidator({
1346
- nullifiersExist: async (nullifiers: Buffer[]) => {
1347
- const merkleTree = this.worldStateSynchronizer.getSnapshot(
1348
- BlockNumber(blockNumber - this.config.doubleSpendSeverePeerPenaltyWindow),
1349
- );
1350
- const indices = await merkleTree.findLeafIndices(MerkleTreeId.NULLIFIER_TREE, nullifiers);
1351
- return indices.map(index => index !== undefined);
1693
+ const snapshotValidator = new DoubleSpendTxValidator(
1694
+ {
1695
+ nullifiersExist: async (nullifiers: Buffer[]) => {
1696
+ const merkleTree = this.worldStateSynchronizer.getSnapshot(
1697
+ BlockNumber(blockNumber - this.config.doubleSpendSeverePeerPenaltyWindow),
1698
+ );
1699
+ const indices = await merkleTree.findLeafIndices(MerkleTreeId.NULLIFIER_TREE, nullifiers);
1700
+ return indices.map(index => index !== undefined);
1701
+ },
1352
1702
  },
1353
- });
1703
+ this.logger.getBindings(),
1704
+ );
1354
1705
 
1355
1706
  const validSnapshot = await snapshotValidator.validateTx(tx);
1356
1707
  if (validSnapshot.result !== 'valid') {
@@ -1361,44 +1712,28 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
1361
1712
  }
1362
1713
 
1363
1714
  /**
1364
- * Validate an attestation.
1715
+ * Validate a checkpoint attestation.
1365
1716
  *
1366
- * @param attestation - The attestation to validate.
1367
- * @returns True if the attestation is valid, false otherwise.
1717
+ * @param attestation - The checkpoint attestation to validate.
1718
+ * @returns True if the checkpoint attestation is valid, false otherwise.
1368
1719
  */
1369
- @trackSpan('Libp2pService.validateAttestation', async (_, attestation) => ({
1720
+ @trackSpan('Libp2pService.validateCheckpointAttestation', async (_, attestation) => ({
1370
1721
  [Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber,
1371
1722
  [Attributes.BLOCK_ARCHIVE]: attestation.archive.toString(),
1372
1723
  [Attributes.P2P_ID]: await attestation.p2pMessageLoggingIdentifier().then(i => i.toString()),
1373
1724
  }))
1374
- public async validateAttestation(peerId: PeerId, attestation: BlockAttestation): Promise<boolean> {
1375
- const severity = await this.attestationValidator.validate(attestation);
1376
- if (severity) {
1377
- this.peerManager.penalizePeer(peerId, severity);
1378
- return false;
1379
- }
1380
-
1381
- return true;
1382
- }
1725
+ public async validateCheckpointAttestation(
1726
+ peerId: PeerId,
1727
+ attestation: CheckpointAttestation,
1728
+ ): Promise<P2PValidationResult> {
1729
+ const result = await this.checkpointAttestationValidator.validate(attestation);
1383
1730
 
1384
- /**
1385
- * Validate a block proposal.
1386
- *
1387
- * @param block - The block proposal to validate.
1388
- * @returns True if the block proposal is valid, false otherwise.
1389
- */
1390
- @trackSpan('Libp2pService.validateBlockProposal', (_peerId, block) => ({
1391
- [Attributes.SLOT_NUMBER]: block.payload.header.slotNumber.toString(),
1392
- }))
1393
- public async validateBlockProposal(peerId: PeerId, block: BlockProposal): Promise<boolean> {
1394
- const severity = await this.blockProposalValidator.validate(block);
1395
- if (severity) {
1396
- this.logger.debug(`Penalizing peer ${peerId} for block proposal validation failure`);
1397
- this.peerManager.penalizePeer(peerId, severity);
1398
- return false;
1731
+ if (result.result === 'reject') {
1732
+ this.logger.warn(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
1733
+ this.peerManager.penalizePeer(peerId, result.severity);
1399
1734
  }
1400
1735
 
1401
- return true;
1736
+ return result;
1402
1737
  }
1403
1738
 
1404
1739
  public getPeerScore(peerId: PeerId): number {