@aztec/p2p 0.0.1-commit.6d63667d → 0.0.1-commit.733c4a3

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 (315) hide show
  1. package/dest/client/factory.d.ts +7 -6
  2. package/dest/client/factory.d.ts.map +1 -1
  3. package/dest/client/factory.js +35 -11
  4. package/dest/client/interface.d.ts +45 -32
  5. package/dest/client/interface.d.ts.map +1 -1
  6. package/dest/client/p2p_client.d.ts +38 -50
  7. package/dest/client/p2p_client.d.ts.map +1 -1
  8. package/dest/client/p2p_client.js +145 -200
  9. package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.js +6 -6
  10. package/dest/config.d.ts +24 -8
  11. package/dest/config.d.ts.map +1 -1
  12. package/dest/config.js +16 -6
  13. package/dest/errors/tx-pool.error.d.ts +8 -0
  14. package/dest/errors/tx-pool.error.d.ts.map +1 -0
  15. package/dest/errors/tx-pool.error.js +9 -0
  16. package/dest/index.d.ts +2 -1
  17. package/dest/index.d.ts.map +1 -1
  18. package/dest/index.js +1 -0
  19. package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +104 -88
  20. package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -1
  21. package/dest/mem_pools/attestation_pool/attestation_pool.js +441 -3
  22. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.d.ts +2 -2
  23. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.d.ts.map +1 -1
  24. package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.js +353 -87
  25. package/dest/mem_pools/attestation_pool/index.d.ts +2 -3
  26. package/dest/mem_pools/attestation_pool/index.d.ts.map +1 -1
  27. package/dest/mem_pools/attestation_pool/index.js +1 -2
  28. package/dest/mem_pools/attestation_pool/mocks.d.ts +2 -2
  29. package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -1
  30. package/dest/mem_pools/attestation_pool/mocks.js +2 -2
  31. package/dest/mem_pools/index.d.ts +3 -2
  32. package/dest/mem_pools/index.d.ts.map +1 -1
  33. package/dest/mem_pools/index.js +1 -1
  34. package/dest/mem_pools/interface.d.ts +5 -5
  35. package/dest/mem_pools/interface.d.ts.map +1 -1
  36. package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.js +3 -3
  37. package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts +104 -0
  38. package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts.map +1 -0
  39. package/dest/mem_pools/tx_pool_v2/deleted_pool.js +251 -0
  40. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts +3 -3
  41. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts.map +1 -1
  42. package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.js +18 -9
  43. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts +1 -1
  44. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -1
  45. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.js +5 -2
  46. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts +3 -3
  47. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts.map +1 -1
  48. package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.js +10 -4
  49. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts +2 -2
  50. package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts.map +1 -1
  51. package/dest/mem_pools/tx_pool_v2/eviction/index.js +1 -1
  52. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts +48 -5
  53. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts.map +1 -1
  54. package/dest/mem_pools/tx_pool_v2/eviction/interfaces.js +8 -0
  55. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.js +7 -5
  56. package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.js +7 -5
  57. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts +2 -2
  58. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts.map +1 -1
  59. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.js +14 -6
  60. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts +4 -4
  61. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts.map +1 -1
  62. package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.js +14 -4
  63. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts +3 -3
  64. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts.map +1 -1
  65. package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.js +2 -2
  66. package/dest/mem_pools/tx_pool_v2/index.d.ts +2 -1
  67. package/dest/mem_pools/tx_pool_v2/index.d.ts.map +1 -1
  68. package/dest/mem_pools/tx_pool_v2/index.js +1 -0
  69. package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts +15 -0
  70. package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts.map +1 -0
  71. package/dest/mem_pools/tx_pool_v2/instrumentation.js +43 -0
  72. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts +22 -8
  73. package/dest/mem_pools/tx_pool_v2/interfaces.d.ts.map +1 -1
  74. package/dest/mem_pools/tx_pool_v2/interfaces.js +4 -1
  75. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts +39 -5
  76. package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts.map +1 -1
  77. package/dest/mem_pools/tx_pool_v2/tx_metadata.js +74 -5
  78. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts +12 -3
  79. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts.map +1 -1
  80. package/dest/mem_pools/tx_pool_v2/tx_pool_indices.js +27 -4
  81. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts +9 -4
  82. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts.map +1 -1
  83. package/dest/mem_pools/tx_pool_v2/tx_pool_v2.js +11 -6
  84. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts +13 -5
  85. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts.map +1 -1
  86. package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.js +312 -169
  87. package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts +3 -3
  88. package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts.map +1 -1
  89. package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts +4 -4
  90. package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts.map +1 -1
  91. package/dest/msg_validators/tx_validator/aggregate_tx_validator.js +3 -3
  92. package/dest/msg_validators/tx_validator/block_header_validator.d.ts +16 -3
  93. package/dest/msg_validators/tx_validator/block_header_validator.d.ts.map +1 -1
  94. package/dest/msg_validators/tx_validator/block_header_validator.js +1 -1
  95. package/dest/msg_validators/tx_validator/double_spend_validator.d.ts +13 -3
  96. package/dest/msg_validators/tx_validator/double_spend_validator.d.ts.map +1 -1
  97. package/dest/msg_validators/tx_validator/double_spend_validator.js +4 -4
  98. package/dest/msg_validators/tx_validator/factory.d.ts +114 -6
  99. package/dest/msg_validators/tx_validator/factory.d.ts.map +1 -1
  100. package/dest/msg_validators/tx_validator/factory.js +219 -58
  101. package/dest/msg_validators/tx_validator/gas_validator.d.ts +58 -3
  102. package/dest/msg_validators/tx_validator/gas_validator.d.ts.map +1 -1
  103. package/dest/msg_validators/tx_validator/gas_validator.js +73 -36
  104. package/dest/msg_validators/tx_validator/index.d.ts +2 -1
  105. package/dest/msg_validators/tx_validator/index.d.ts.map +1 -1
  106. package/dest/msg_validators/tx_validator/index.js +1 -0
  107. package/dest/msg_validators/tx_validator/nullifier_cache.d.ts +14 -0
  108. package/dest/msg_validators/tx_validator/nullifier_cache.d.ts.map +1 -0
  109. package/dest/msg_validators/tx_validator/nullifier_cache.js +24 -0
  110. package/dest/msg_validators/tx_validator/timestamp_validator.d.ts +20 -4
  111. package/dest/msg_validators/tx_validator/timestamp_validator.d.ts.map +1 -1
  112. package/dest/msg_validators/tx_validator/timestamp_validator.js +6 -6
  113. package/dest/services/dummy_service.d.ts +13 -5
  114. package/dest/services/dummy_service.d.ts.map +1 -1
  115. package/dest/services/dummy_service.js +10 -4
  116. package/dest/services/encoding.d.ts +2 -2
  117. package/dest/services/encoding.d.ts.map +1 -1
  118. package/dest/services/encoding.js +4 -3
  119. package/dest/services/gossipsub/index.d.ts +3 -0
  120. package/dest/services/gossipsub/index.d.ts.map +1 -0
  121. package/dest/services/gossipsub/index.js +2 -0
  122. package/dest/services/gossipsub/scoring.d.ts +21 -3
  123. package/dest/services/gossipsub/scoring.d.ts.map +1 -1
  124. package/dest/services/gossipsub/scoring.js +24 -7
  125. package/dest/services/gossipsub/topic_score_params.d.ts +173 -0
  126. package/dest/services/gossipsub/topic_score_params.d.ts.map +1 -0
  127. package/dest/services/gossipsub/topic_score_params.js +346 -0
  128. package/dest/services/libp2p/libp2p_service.d.ts +91 -38
  129. package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
  130. package/dest/services/libp2p/libp2p_service.js +411 -321
  131. package/dest/services/peer-manager/peer_scoring.d.ts +1 -1
  132. package/dest/services/peer-manager/peer_scoring.d.ts.map +1 -1
  133. package/dest/services/peer-manager/peer_scoring.js +25 -2
  134. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts +4 -3
  135. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts.map +1 -1
  136. package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.js +19 -46
  137. package/dest/services/reqresp/batch-tx-requester/interface.d.ts +2 -6
  138. package/dest/services/reqresp/batch-tx-requester/interface.d.ts.map +1 -1
  139. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts +10 -13
  140. package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts.map +1 -1
  141. package/dest/services/reqresp/batch-tx-requester/missing_txs.js +25 -46
  142. package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts +17 -11
  143. package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts.map +1 -1
  144. package/dest/services/reqresp/batch-tx-requester/peer_collection.js +49 -15
  145. package/dest/services/reqresp/batch-tx-requester/tx_validator.js +2 -2
  146. package/dest/services/reqresp/interface.d.ts +10 -1
  147. package/dest/services/reqresp/interface.d.ts.map +1 -1
  148. package/dest/services/reqresp/interface.js +15 -1
  149. package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts +4 -3
  150. package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts.map +1 -1
  151. package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts +7 -1
  152. package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts.map +1 -1
  153. package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.js +15 -0
  154. package/dest/services/reqresp/protocols/tx.d.ts +7 -1
  155. package/dest/services/reqresp/protocols/tx.d.ts.map +1 -1
  156. package/dest/services/reqresp/protocols/tx.js +20 -0
  157. package/dest/services/reqresp/reqresp.d.ts +1 -1
  158. package/dest/services/reqresp/reqresp.d.ts.map +1 -1
  159. package/dest/services/reqresp/reqresp.js +11 -4
  160. package/dest/services/service.d.ts +39 -3
  161. package/dest/services/service.d.ts.map +1 -1
  162. package/dest/services/tx_collection/config.d.ts +19 -1
  163. package/dest/services/tx_collection/config.d.ts.map +1 -1
  164. package/dest/services/tx_collection/config.js +46 -0
  165. package/dest/services/tx_collection/fast_tx_collection.d.ts +3 -1
  166. package/dest/services/tx_collection/fast_tx_collection.d.ts.map +1 -1
  167. package/dest/services/tx_collection/fast_tx_collection.js +56 -36
  168. package/dest/services/tx_collection/file_store_tx_collection.d.ts +53 -0
  169. package/dest/services/tx_collection/file_store_tx_collection.d.ts.map +1 -0
  170. package/dest/services/tx_collection/file_store_tx_collection.js +167 -0
  171. package/dest/services/tx_collection/file_store_tx_source.d.ts +37 -0
  172. package/dest/services/tx_collection/file_store_tx_source.d.ts.map +1 -0
  173. package/dest/services/tx_collection/file_store_tx_source.js +90 -0
  174. package/dest/services/tx_collection/index.d.ts +2 -1
  175. package/dest/services/tx_collection/index.d.ts.map +1 -1
  176. package/dest/services/tx_collection/index.js +1 -0
  177. package/dest/services/tx_collection/instrumentation.d.ts +1 -1
  178. package/dest/services/tx_collection/instrumentation.d.ts.map +1 -1
  179. package/dest/services/tx_collection/instrumentation.js +2 -1
  180. package/dest/services/tx_collection/missing_txs_tracker.d.ts +32 -0
  181. package/dest/services/tx_collection/missing_txs_tracker.d.ts.map +1 -0
  182. package/dest/services/tx_collection/missing_txs_tracker.js +27 -0
  183. package/dest/services/tx_collection/proposal_tx_collector.d.ts +7 -6
  184. package/dest/services/tx_collection/proposal_tx_collector.d.ts.map +1 -1
  185. package/dest/services/tx_collection/proposal_tx_collector.js +5 -4
  186. package/dest/services/tx_collection/slow_tx_collection.d.ts +7 -3
  187. package/dest/services/tx_collection/slow_tx_collection.d.ts.map +1 -1
  188. package/dest/services/tx_collection/slow_tx_collection.js +60 -26
  189. package/dest/services/tx_collection/tx_collection.d.ts +23 -10
  190. package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
  191. package/dest/services/tx_collection/tx_collection.js +75 -3
  192. package/dest/services/tx_collection/tx_collection_sink.d.ts +18 -8
  193. package/dest/services/tx_collection/tx_collection_sink.d.ts.map +1 -1
  194. package/dest/services/tx_collection/tx_collection_sink.js +26 -29
  195. package/dest/services/tx_collection/tx_source.d.ts +8 -3
  196. package/dest/services/tx_collection/tx_source.d.ts.map +1 -1
  197. package/dest/services/tx_collection/tx_source.js +19 -2
  198. package/dest/services/tx_file_store/config.d.ts +1 -3
  199. package/dest/services/tx_file_store/config.d.ts.map +1 -1
  200. package/dest/services/tx_file_store/config.js +0 -4
  201. package/dest/services/tx_file_store/tx_file_store.d.ts +4 -3
  202. package/dest/services/tx_file_store/tx_file_store.d.ts.map +1 -1
  203. package/dest/services/tx_file_store/tx_file_store.js +9 -6
  204. package/dest/services/tx_provider.d.ts +4 -4
  205. package/dest/services/tx_provider.d.ts.map +1 -1
  206. package/dest/services/tx_provider.js +9 -8
  207. package/dest/test-helpers/make-test-p2p-clients.d.ts +3 -3
  208. package/dest/test-helpers/make-test-p2p-clients.d.ts.map +1 -1
  209. package/dest/test-helpers/mock-pubsub.d.ts +29 -2
  210. package/dest/test-helpers/mock-pubsub.d.ts.map +1 -1
  211. package/dest/test-helpers/mock-pubsub.js +103 -2
  212. package/dest/test-helpers/reqresp-nodes.d.ts +1 -1
  213. package/dest/test-helpers/reqresp-nodes.d.ts.map +1 -1
  214. package/dest/test-helpers/reqresp-nodes.js +2 -1
  215. package/dest/test-helpers/testbench-utils.d.ts +43 -38
  216. package/dest/test-helpers/testbench-utils.d.ts.map +1 -1
  217. package/dest/test-helpers/testbench-utils.js +128 -59
  218. package/dest/testbench/p2p_client_testbench_worker.d.ts +2 -2
  219. package/dest/testbench/p2p_client_testbench_worker.d.ts.map +1 -1
  220. package/dest/testbench/p2p_client_testbench_worker.js +10 -10
  221. package/dest/util.d.ts +2 -2
  222. package/dest/util.d.ts.map +1 -1
  223. package/package.json +14 -14
  224. package/src/client/factory.ts +66 -14
  225. package/src/client/interface.ts +61 -33
  226. package/src/client/p2p_client.ts +179 -236
  227. package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker.ts +19 -9
  228. package/src/config.ts +35 -11
  229. package/src/errors/tx-pool.error.ts +12 -0
  230. package/src/index.ts +1 -0
  231. package/src/mem_pools/attestation_pool/attestation_pool.ts +496 -91
  232. package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +442 -102
  233. package/src/mem_pools/attestation_pool/index.ts +9 -2
  234. package/src/mem_pools/attestation_pool/mocks.ts +2 -1
  235. package/src/mem_pools/index.ts +4 -1
  236. package/src/mem_pools/interface.ts +4 -4
  237. package/src/mem_pools/tx_pool/README.md +1 -1
  238. package/src/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.ts +3 -3
  239. package/src/mem_pools/tx_pool_v2/README.md +76 -10
  240. package/src/mem_pools/tx_pool_v2/deleted_pool.ts +321 -0
  241. package/src/mem_pools/tx_pool_v2/eviction/eviction_manager.ts +21 -8
  242. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.ts +5 -2
  243. package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.ts +15 -4
  244. package/src/mem_pools/tx_pool_v2/eviction/index.ts +4 -0
  245. package/src/mem_pools/tx_pool_v2/eviction/interfaces.ts +49 -4
  246. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.ts +5 -5
  247. package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.ts +5 -5
  248. package/src/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.ts +14 -9
  249. package/src/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.ts +24 -6
  250. package/src/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.ts +3 -3
  251. package/src/mem_pools/tx_pool_v2/index.ts +1 -0
  252. package/src/mem_pools/tx_pool_v2/instrumentation.ts +69 -0
  253. package/src/mem_pools/tx_pool_v2/interfaces.ts +23 -8
  254. package/src/mem_pools/tx_pool_v2/tx_metadata.ts +103 -9
  255. package/src/mem_pools/tx_pool_v2/tx_pool_indices.ts +32 -5
  256. package/src/mem_pools/tx_pool_v2/tx_pool_v2.ts +18 -7
  257. package/src/mem_pools/tx_pool_v2/tx_pool_v2_impl.ts +347 -166
  258. package/src/msg_validators/attestation_validator/fisherman_attestation_validator.ts +2 -2
  259. package/src/msg_validators/tx_validator/README.md +115 -0
  260. package/src/msg_validators/tx_validator/aggregate_tx_validator.ts +5 -5
  261. package/src/msg_validators/tx_validator/block_header_validator.ts +15 -3
  262. package/src/msg_validators/tx_validator/double_spend_validator.ts +11 -6
  263. package/src/msg_validators/tx_validator/factory.ts +353 -77
  264. package/src/msg_validators/tx_validator/gas_validator.ts +90 -27
  265. package/src/msg_validators/tx_validator/index.ts +1 -0
  266. package/src/msg_validators/tx_validator/nullifier_cache.ts +30 -0
  267. package/src/msg_validators/tx_validator/timestamp_validator.ts +23 -18
  268. package/src/services/dummy_service.ts +18 -6
  269. package/src/services/encoding.ts +4 -3
  270. package/src/services/gossipsub/README.md +641 -0
  271. package/src/services/gossipsub/index.ts +2 -0
  272. package/src/services/gossipsub/scoring.ts +29 -5
  273. package/src/services/gossipsub/topic_score_params.ts +487 -0
  274. package/src/services/libp2p/libp2p_service.ts +431 -344
  275. package/src/services/peer-manager/peer_scoring.ts +25 -0
  276. package/src/services/reqresp/batch-tx-requester/batch_tx_requester.ts +20 -48
  277. package/src/services/reqresp/batch-tx-requester/interface.ts +1 -5
  278. package/src/services/reqresp/batch-tx-requester/missing_txs.ts +23 -71
  279. package/src/services/reqresp/batch-tx-requester/peer_collection.ts +63 -24
  280. package/src/services/reqresp/batch-tx-requester/tx_validator.ts +2 -2
  281. package/src/services/reqresp/interface.ts +26 -1
  282. package/src/services/reqresp/protocols/block_txs/block_txs_handler.ts +4 -3
  283. package/src/services/reqresp/protocols/block_txs/block_txs_reqresp.ts +17 -0
  284. package/src/services/reqresp/protocols/tx.ts +22 -0
  285. package/src/services/reqresp/reqresp.ts +13 -3
  286. package/src/services/service.ts +51 -2
  287. package/src/services/tx_collection/config.ts +68 -0
  288. package/src/services/tx_collection/fast_tx_collection.ts +65 -32
  289. package/src/services/tx_collection/file_store_tx_collection.ts +202 -0
  290. package/src/services/tx_collection/file_store_tx_source.ts +117 -0
  291. package/src/services/tx_collection/index.ts +1 -0
  292. package/src/services/tx_collection/instrumentation.ts +7 -1
  293. package/src/services/tx_collection/missing_txs_tracker.ts +52 -0
  294. package/src/services/tx_collection/proposal_tx_collector.ts +8 -7
  295. package/src/services/tx_collection/slow_tx_collection.ts +66 -33
  296. package/src/services/tx_collection/tx_collection.ts +113 -16
  297. package/src/services/tx_collection/tx_collection_sink.ts +30 -34
  298. package/src/services/tx_collection/tx_source.ts +22 -3
  299. package/src/services/tx_file_store/config.ts +0 -6
  300. package/src/services/tx_file_store/tx_file_store.ts +10 -8
  301. package/src/services/tx_provider.ts +10 -9
  302. package/src/test-helpers/make-test-p2p-clients.ts +3 -3
  303. package/src/test-helpers/mock-pubsub.ts +143 -3
  304. package/src/test-helpers/reqresp-nodes.ts +2 -1
  305. package/src/test-helpers/testbench-utils.ts +127 -71
  306. package/src/testbench/p2p_client_testbench_worker.ts +22 -15
  307. package/src/util.ts +7 -1
  308. package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts +0 -40
  309. package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts.map +0 -1
  310. package/dest/mem_pools/attestation_pool/kv_attestation_pool.js +0 -218
  311. package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts +0 -31
  312. package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts.map +0 -1
  313. package/dest/mem_pools/attestation_pool/memory_attestation_pool.js +0 -180
  314. package/src/mem_pools/attestation_pool/kv_attestation_pool.ts +0 -320
  315. package/src/mem_pools/attestation_pool/memory_attestation_pool.ts +0 -264
@@ -372,7 +372,6 @@ function _apply_decs_2203_r(targetClass, memberDecs, classDecs, parentClass) {
372
372
  }
373
373
  var _dec, _dec1, _dec2, _dec3, _dec4, _dec5, _dec6, _dec7, _dec8, _initProto;
374
374
  import { BlockNumber } from '@aztec/foundation/branded-types';
375
- import { randomInt } from '@aztec/foundation/crypto/random';
376
375
  import { createLibp2pComponentLogger, createLogger } from '@aztec/foundation/log';
377
376
  import { RunningPromise } from '@aztec/foundation/running-promise';
378
377
  import { Timer } from '@aztec/foundation/timer';
@@ -384,7 +383,7 @@ import { Tx } from '@aztec/stdlib/tx';
384
383
  import { compressComponentVersions } from '@aztec/stdlib/versioning';
385
384
  import { Attributes, OtelMetricsAdapter, SpanStatusCode, WithTracer, trackSpan } from '@aztec/telemetry-client';
386
385
  import { gossipsub } from '@chainsafe/libp2p-gossipsub';
387
- import { createPeerScoreParams, createTopicScoreParams } from '@chainsafe/libp2p-gossipsub/score';
386
+ import { createPeerScoreParams } from '@chainsafe/libp2p-gossipsub/score';
388
387
  import { SignaturePolicy } from '@chainsafe/libp2p-gossipsub/types';
389
388
  import { noise } from '@chainsafe/libp2p-noise';
390
389
  import { yamux } from '@chainsafe/libp2p-yamux';
@@ -395,47 +394,47 @@ import { mplex } from '@libp2p/mplex';
395
394
  import { tcp } from '@libp2p/tcp';
396
395
  import { ENR } from '@nethermindeth/enr';
397
396
  import { createLibp2p } from 'libp2p';
398
- import { ProposalSlotCapExceededError } from '../../errors/attestation-pool.error.js';
399
397
  import { BlockProposalValidator, CheckpointAttestationValidator, CheckpointProposalValidator, DoubleSpendTxValidator, FishermanAttestationValidator, getDefaultAllowedSetupFunctions } from '../../msg_validators/index.js';
400
398
  import { MessageSeenValidator } from '../../msg_validators/msg_seen_validator/msg_seen_validator.js';
401
- import { createTxMessageValidators, createTxReqRespValidator } from '../../msg_validators/tx_validator/factory.js';
399
+ import { createFirstStageTxValidationsForGossipedTransactions, createSecondStageTxValidationsForGossipedTransactions, createTxValidatorForBlockProposalReceivedTxs, createTxValidatorForReqResponseReceivedTxs } from '../../msg_validators/tx_validator/factory.js';
402
400
  import { GossipSubEvent } from '../../types/index.js';
403
401
  import { convertToMultiaddr } from '../../util.js';
404
402
  import { getVersions } from '../../versioning.js';
405
403
  import { AztecDatastore } from '../data_store.js';
406
404
  import { DiscV5Service } from '../discv5/discV5_service.js';
407
405
  import { SnappyTransform, fastMsgIdFn, getMsgIdFn, msgIdToStrFn } from '../encoding.js';
408
- import { gossipScoreThresholds } from '../gossipsub/scoring.js';
406
+ import { APP_SPECIFIC_WEIGHT, gossipScoreThresholds } from '../gossipsub/scoring.js';
407
+ import { createAllTopicScoreParams } from '../gossipsub/topic_score_params.js';
409
408
  import { PeerManager } from '../peer-manager/peer_manager.js';
410
409
  import { PeerScoring } from '../peer-manager/peer_scoring.js';
411
- import { DEFAULT_SUB_PROTOCOL_VALIDATORS, ReqRespSubProtocol, ValidationError } from '../reqresp/index.js';
412
- import { pingHandler, reqGoodbyeHandler, reqRespBlockHandler, reqRespBlockTxsHandler, reqRespStatusHandler, reqRespTxHandler } from '../reqresp/index.js';
410
+ import { DEFAULT_SUB_PROTOCOL_VALIDATORS, ReqRespSubProtocol, ValidationError, pingHandler, reqGoodbyeHandler, reqRespBlockHandler, reqRespBlockTxsHandler, reqRespStatusHandler, reqRespTxHandler } from '../reqresp/index.js';
413
411
  import { ReqResp } from '../reqresp/reqresp.js';
414
412
  import { P2PInstrumentation } from './instrumentation.js';
415
- _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
413
+ _dec = trackSpan('Libp2pService.validateAndStoreCheckpointAttestation', (_peerId, attestation)=>({
414
+ [Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber.toString()
415
+ })), _dec1 = trackSpan('Libp2pService.validateAndStoreBlockProposal', (_peerId, block)=>({
416
+ [Attributes.BLOCK_NUMBER]: block.blockNumber.toString(),
417
+ [Attributes.SLOT_NUMBER]: block.slotNumber.toString()
418
+ })), _dec2 = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
416
419
  [Attributes.SLOT_NUMBER]: block.slotNumber,
417
420
  [Attributes.BLOCK_ARCHIVE]: block.archive.toString(),
418
421
  [Attributes.P2P_ID]: await block.p2pMessageLoggingIdentifier().then((i)=>i.toString())
419
- })), _dec1 = trackSpan('Libp2pService.processValidCheckpointProposal', async (checkpoint)=>({
422
+ })), _dec3 = trackSpan('Libp2pService.validateAndStoreCheckpointProposal', (_peerId, checkpoint)=>({
423
+ [Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString()
424
+ })), _dec4 = trackSpan('Libp2pService.processValidCheckpointProposal', async (checkpoint)=>({
420
425
  [Attributes.SLOT_NUMBER]: checkpoint.slotNumber,
421
426
  [Attributes.BLOCK_ARCHIVE]: checkpoint.archive.toString(),
422
427
  [Attributes.P2P_ID]: await checkpoint.p2pMessageLoggingIdentifier().then((i)=>i.toString())
423
- })), _dec2 = trackSpan('Libp2pService.validateRequestedBlockTxs', (request)=>({
428
+ })), _dec5 = trackSpan('Libp2pService.validateRequestedBlockTxs', (request)=>({
424
429
  [Attributes.BLOCK_ARCHIVE]: request.archiveRoot.toString()
425
- })), _dec3 = trackSpan('Libp2pService.validateRequestedTx', (requestedTxHash, _responseTx)=>({
430
+ })), _dec6 = trackSpan('Libp2pService.validateRequestedTx', (requestedTxHash, _responseTx)=>({
426
431
  [Attributes.TX_HASH]: requestedTxHash.toString()
427
- })), _dec4 = trackSpan('Libp2pService.validateRequestedBlock', (requestedBlockNumber, _responseBlock)=>({
432
+ })), _dec7 = trackSpan('Libp2pService.validateRequestedBlock', (requestedBlockNumber, _responseBlock)=>({
428
433
  [Attributes.BLOCK_NUMBER]: requestedBlockNumber.toString()
429
- })), _dec5 = trackSpan('Libp2pService.validatePropagatedTx', (tx)=>({
430
- [Attributes.TX_HASH]: tx.getTxHash().toString()
431
- })), _dec6 = trackSpan('Libp2pService.validateCheckpointAttestation', async (_, attestation)=>({
434
+ })), _dec8 = trackSpan('Libp2pService.validateCheckpointAttestation', async (_, attestation)=>({
432
435
  [Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber,
433
436
  [Attributes.BLOCK_ARCHIVE]: attestation.archive.toString(),
434
437
  [Attributes.P2P_ID]: await attestation.p2pMessageLoggingIdentifier().then((i)=>i.toString())
435
- })), _dec7 = trackSpan('Libp2pService.validateBlockProposal', (_peerId, block)=>({
436
- [Attributes.SLOT_NUMBER]: block.slotNumber.toString()
437
- })), _dec8 = trackSpan('Libp2pService.validateCheckpointProposal', (_peerId, checkpoint)=>({
438
- [Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString()
439
438
  }));
440
439
  /**
441
440
  * Lib P2P implementation of the P2PService interface.
@@ -456,47 +455,47 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
456
455
  [
457
456
  _dec,
458
457
  2,
459
- "processValidBlockProposal"
458
+ "validateAndStoreCheckpointAttestation"
460
459
  ],
461
460
  [
462
461
  _dec1,
463
462
  2,
464
- "processValidCheckpointProposal"
463
+ "validateAndStoreBlockProposal"
465
464
  ],
466
465
  [
467
466
  _dec2,
468
467
  2,
469
- "validateRequestedBlockTxs"
468
+ "processValidBlockProposal"
470
469
  ],
471
470
  [
472
471
  _dec3,
473
472
  2,
474
- "validateRequestedTxs"
473
+ "validateAndStoreCheckpointProposal"
475
474
  ],
476
475
  [
477
476
  _dec4,
478
477
  2,
479
- "validateRequestedBlock"
478
+ "processValidCheckpointProposal"
480
479
  ],
481
480
  [
482
481
  _dec5,
483
482
  2,
484
- "validatePropagatedTx"
483
+ "validateRequestedBlockTxs"
485
484
  ],
486
485
  [
487
486
  _dec6,
488
487
  2,
489
- "validateCheckpointAttestation"
488
+ "validateRequestedTxs"
490
489
  ],
491
490
  [
492
491
  _dec7,
493
492
  2,
494
- "validateBlockProposal"
493
+ "validateRequestedBlock"
495
494
  ],
496
495
  [
497
496
  _dec8,
498
497
  2,
499
- "validateCheckpointProposal"
498
+ "validateCheckpointAttestation"
500
499
  ]
501
500
  ], []));
502
501
  }
@@ -509,6 +508,8 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
509
508
  protocolVersion;
510
509
  topicStrings;
511
510
  feesCache;
511
+ /** Callback invoked when a duplicate proposal is detected (triggers slashing). */ duplicateProposalCallback;
512
+ /** Callback invoked when a duplicate attestation is detected (triggers slashing). */ duplicateAttestationCallback;
512
513
  /**
513
514
  * Callback for when a block is received from a peer.
514
515
  * @param block - The block received from the peer.
@@ -587,10 +588,6 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
587
588
  }
588
589
  const versions = getVersions(config);
589
590
  const protocolVersion = compressComponentVersions(versions);
590
- const txTopic = createTopicString(TopicType.tx, protocolVersion);
591
- const blockProposalTopic = createTopicString(TopicType.block_proposal, protocolVersion);
592
- const checkpointProposalTopic = createTopicString(TopicType.checkpoint_proposal, protocolVersion);
593
- const checkpointAttestationTopic = createTopicString(TopicType.checkpoint_attestation, protocolVersion);
594
591
  const preferredPeersEnrs = config.preferredPeers.map((enr)=>ENR.decodeTxt(enr));
595
592
  const directPeers = (await Promise.all(preferredPeersEnrs.map(async (enr)=>{
596
593
  const peerId = await enr.peerId();
@@ -608,6 +605,15 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
608
605
  const announceTcpMultiaddr = config.p2pIp ? [
609
606
  convertToMultiaddr(config.p2pIp, p2pPort, 'tcp')
610
607
  ] : [];
608
+ // Create dynamic topic score params based on network configuration
609
+ const l1Constants = epochCache.getL1Constants();
610
+ const topicScoreParams = createAllTopicScoreParams(protocolVersion, {
611
+ slotDurationMs: l1Constants.slotDuration * 1000,
612
+ heartbeatIntervalMs: config.gossipsubInterval,
613
+ targetCommitteeSize: l1Constants.targetCommitteeSize,
614
+ blockDurationMs: config.blockDurationMs,
615
+ expectedBlockProposalsPerSlot: config.expectedBlockProposalsPerSlot
616
+ });
611
617
  const node = await createLibp2p({
612
618
  start: false,
613
619
  peerId,
@@ -708,28 +714,7 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
708
714
  scoreParams: createPeerScoreParams({
709
715
  // IPColocation factor can be disabled for local testing - default to -5
710
716
  IPColocationFactorWeight: config.debugDisableColocationPenalty ? 0 : -5.0,
711
- topics: {
712
- [txTopic]: createTopicScoreParams({
713
- topicWeight: 1,
714
- invalidMessageDeliveriesWeight: -20,
715
- invalidMessageDeliveriesDecay: 0.5
716
- }),
717
- [blockProposalTopic]: createTopicScoreParams({
718
- topicWeight: 1,
719
- invalidMessageDeliveriesWeight: -20,
720
- invalidMessageDeliveriesDecay: 0.5
721
- }),
722
- [checkpointProposalTopic]: createTopicScoreParams({
723
- topicWeight: 1,
724
- invalidMessageDeliveriesWeight: -20,
725
- invalidMessageDeliveriesDecay: 0.5
726
- }),
727
- [checkpointAttestationTopic]: createTopicScoreParams({
728
- topicWeight: 1,
729
- invalidMessageDeliveriesWeight: -20,
730
- invalidMessageDeliveriesDecay: 0.5
731
- })
732
- }
717
+ topics: topicScoreParams
733
718
  })
734
719
  }),
735
720
  components: (components)=>({
@@ -741,8 +726,12 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
741
726
  const peerScoring = new PeerScoring(config, telemetry);
742
727
  const reqresp = new ReqResp(config, node, peerScoring, createLogger(`${logger.module}:reqresp`));
743
728
  const peerManager = new PeerManager(node, peerDiscoveryService, config, telemetry, createLogger(`${logger.module}:peer_manager`), peerScoring, reqresp, worldStateSynchronizer, protocolVersion, epochCache);
744
- // Update gossipsub score params
745
- node.services.pubsub.score.params.appSpecificWeight = 10;
729
+ // Configure application-specific scoring for gossipsub.
730
+ // The weight scales app score to align with gossipsub thresholds:
731
+ // - Disconnect (-50) × 10 = -500 = gossipThreshold (stops receiving gossip)
732
+ // - Ban (-100) × 10 = -1000 = publishThreshold (cannot publish)
733
+ // Note: positive topic scores can offset penalties, so alignment is best-effort.
734
+ node.services.pubsub.score.params.appSpecificWeight = APP_SPECIFIC_WEIGHT;
746
735
  node.services.pubsub.score.params.appSpecificScore = (peerId)=>peerManager.shouldDisableP2PGossip(peerId) ? -Infinity : peerManager.getPeerScore(peerId);
747
736
  return new LibP2PService(clientType, config, node, peerDiscoveryService, reqresp, peerManager, mempools, l2BlockSource, epochCache, proofVerifier, worldStateSynchronizer, telemetry, logger);
748
737
  }
@@ -837,6 +826,9 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
837
826
  getPeers(includePending) {
838
827
  return this.peerManager.getPeers(includePending);
839
828
  }
829
+ getGossipMeshPeerCount(topicType) {
830
+ return this.node.services.pubsub.getMeshPeers(this.topicStrings[topicType]).length;
831
+ }
840
832
  handleGossipSubEvent(e) {
841
833
  this.logger.trace(`Received PUBSUB message.`);
842
834
  const safeJob = async ()=>{
@@ -872,6 +864,19 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
872
864
  this.checkpointReceivedCallback = callback;
873
865
  }
874
866
  /**
867
+ * Registers a callback to be invoked when a duplicate proposal is detected.
868
+ * This callback is triggered on the first duplicate (when count goes from 1 to 2).
869
+ */ registerDuplicateProposalCallback(callback) {
870
+ this.duplicateProposalCallback = callback;
871
+ }
872
+ /**
873
+ * Registers a callback to be invoked when a duplicate attestation is detected.
874
+ * A validator signing attestations for different proposals at the same slot.
875
+ * This callback is triggered on the first duplicate (when count goes from 1 to 2).
876
+ */ registerDuplicateAttestationCallback(callback) {
877
+ this.duplicateAttestationCallback = callback;
878
+ }
879
+ /**
875
880
  * Subscribes to a topic.
876
881
  * @param topic - The topic to subscribe to.
877
882
  */ subscribeToTopic(topic) {
@@ -1052,26 +1057,71 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
1052
1057
  async handleGossipedTx(payloadData, msgId, source) {
1053
1058
  const validationFunc = async ()=>{
1054
1059
  const tx = Tx.fromBuffer(payloadData);
1055
- const isValid = await this.validatePropagatedTx(tx, source);
1056
- const exists = isValid && await this.mempools.txPool.hasTx(tx.getTxHash());
1060
+ const currentBlockNumber = await this.archiver.getBlockNumber();
1061
+ const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
1062
+ // Stage 1: fast validators (metadata, data, timestamps, double-spend, gas, phases, block header)
1063
+ const firstStageValidators = await this.createFirstStageMessageValidators(currentBlockNumber, nextSlotTimestamp);
1064
+ const firstStageOutcome = await this.runValidations(tx, firstStageValidators);
1065
+ if (!firstStageOutcome.allPassed) {
1066
+ const { name } = firstStageOutcome.failure;
1067
+ let { severity } = firstStageOutcome.failure;
1068
+ // Double spend validator has a special case handler. We perform more detailed examination
1069
+ // as to how recently the nullifier was entered into the tree and if the transaction should
1070
+ // have 'known' the nullifier existed. This determines the severity of the penalty applied to the peer.
1071
+ if (name === 'doubleSpendValidator') {
1072
+ const txBlockNumber = BlockNumber(currentBlockNumber + 1);
1073
+ severity = await this.handleDoubleSpendFailure(tx, txBlockNumber);
1074
+ }
1075
+ this.peerManager.penalizePeer(source, severity);
1076
+ return {
1077
+ result: TopicValidatorResult.Reject
1078
+ };
1079
+ }
1080
+ // Pool pre-check: see if the pool would accept this tx before doing expensive proof verification
1081
+ const canAdd = await this.mempools.txPool.canAddPendingTx(tx);
1082
+ if (canAdd === 'ignored') {
1083
+ return {
1084
+ result: TopicValidatorResult.Ignore,
1085
+ obj: tx
1086
+ };
1087
+ }
1088
+ // Stage 2: expensive proof verification
1089
+ const secondStageValidators = this.createSecondStageMessageValidators();
1090
+ const secondStageOutcome = await this.runValidations(tx, secondStageValidators);
1091
+ if (!secondStageOutcome.allPassed) {
1092
+ const { severity } = secondStageOutcome.failure;
1093
+ this.peerManager.penalizePeer(source, severity);
1094
+ return {
1095
+ result: TopicValidatorResult.Reject
1096
+ };
1097
+ }
1098
+ // Pool add: persist the tx
1099
+ const txHash = tx.getTxHash();
1100
+ const addResult = await this.mempools.txPool.addPendingTxs([
1101
+ tx
1102
+ ], {
1103
+ source: 'gossip'
1104
+ });
1105
+ const wasAccepted = addResult.accepted.some((h)=>h.equals(txHash));
1106
+ const wasIgnored = addResult.ignored.some((h)=>h.equals(txHash));
1057
1107
  this.logger.trace(`Validate propagated tx`, {
1058
- isValid,
1059
- exists,
1108
+ wasAccepted,
1109
+ wasIgnored,
1060
1110
  [Attributes.P2P_ID]: source.toString()
1061
1111
  });
1062
- if (!isValid) {
1112
+ if (wasAccepted) {
1063
1113
  return {
1064
- result: TopicValidatorResult.Reject
1114
+ result: TopicValidatorResult.Accept,
1115
+ obj: tx
1065
1116
  };
1066
- } else if (exists) {
1117
+ } else if (wasIgnored) {
1067
1118
  return {
1068
1119
  result: TopicValidatorResult.Ignore,
1069
1120
  obj: tx
1070
1121
  };
1071
1122
  } else {
1072
1123
  return {
1073
- result: TopicValidatorResult.Accept,
1074
- obj: tx
1124
+ result: TopicValidatorResult.Reject
1075
1125
  };
1076
1126
  }
1077
1127
  };
@@ -1079,133 +1129,190 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
1079
1129
  if (result !== TopicValidatorResult.Accept || !tx) {
1080
1130
  return;
1081
1131
  }
1132
+ // Tx was accepted into pool and will be propagated - just log and record metrics
1082
1133
  const txHash = tx.getTxHash();
1083
1134
  const txHashString = txHash.toString();
1084
1135
  this.logger.verbose(`Received tx ${txHashString} from external peer ${source.toString()} via gossip`, {
1085
1136
  source: source.toString(),
1086
1137
  txHash: txHashString
1087
1138
  });
1088
- if (this.config.dropTransactions && randomInt(1000) < this.config.dropTransactionsProbability * 1000) {
1089
- this.logger.warn(`Intentionally dropping tx ${txHashString} (probability rule)`);
1090
- return;
1091
- }
1092
1139
  this.instrumentation.incrementTxReceived(1);
1093
- await this.mempools.txPool.addTxs([
1094
- tx
1095
- ]);
1096
1140
  }
1097
1141
  /**
1098
1142
  * Process a checkpoint attestation from a peer.
1099
1143
  * Validates the attestation and adds it to the pool.
1100
1144
  */ async processCheckpointAttestationFromPeer(payloadData, msgId, source) {
1101
- const validationFunc = async ()=>{
1102
- const attestation = CheckpointAttestation.fromBuffer(payloadData);
1103
- const pool = this.mempools.attestationPool;
1104
- const validationResult = await this.validateCheckpointAttestation(source, attestation);
1105
- const isValid = validationResult.result === 'accept';
1106
- const exists = isValid && await pool.hasCheckpointAttestation(attestation);
1107
- let canAdd = true;
1108
- if (isValid && !exists) {
1109
- const slot = attestation.payload.header.slotNumber;
1110
- const { committee } = await this.epochCache.getCommittee(slot);
1111
- const committeeSize = committee?.length ?? 0;
1112
- canAdd = await pool.canAddCheckpointAttestation(attestation, committeeSize);
1113
- }
1114
- this.logger.trace(`Validate propagated checkpoint attestation`, {
1115
- isValid,
1116
- exists,
1117
- canAdd,
1118
- [Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber.toString(),
1119
- [Attributes.P2P_ID]: source.toString()
1120
- });
1121
- if (validationResult.result === 'reject') {
1122
- return {
1123
- result: TopicValidatorResult.Reject
1124
- };
1125
- } else if (validationResult.result === 'ignore' || exists) {
1126
- return {
1127
- result: TopicValidatorResult.Ignore,
1128
- obj: attestation
1129
- };
1130
- } else if (!canAdd) {
1131
- this.logger.warn(`Dropping checkpoint attestation due to per-(slot, proposalId) attestation cap`, {
1132
- slot: attestation.payload.header.slotNumber.toString(),
1133
- archive: attestation.archive.toString(),
1134
- source: source.toString()
1135
- });
1136
- return {
1137
- result: TopicValidatorResult.Ignore,
1138
- obj: attestation
1139
- };
1140
- } else {
1141
- return {
1142
- result: TopicValidatorResult.Accept,
1143
- obj: attestation
1144
- };
1145
- }
1146
- };
1147
- const { result, obj: attestation } = await this.validateReceivedMessage(validationFunc, msgId, source, TopicType.checkpoint_attestation);
1145
+ const { result, obj: attestation } = await this.validateReceivedMessage(()=>this.validateAndStoreCheckpointAttestation(source, CheckpointAttestation.fromBuffer(payloadData)), msgId, source, TopicType.checkpoint_attestation);
1148
1146
  if (result !== TopicValidatorResult.Accept || !attestation) {
1149
1147
  return;
1150
1148
  }
1151
- this.logger.debug(`Received checkpoint attestation for slot ${attestation.slotNumber} from external peer ${source.toString()}`, {
1149
+ this.logger.verbose(`Received valid checkpoint attestation for slot ${attestation.slotNumber} from external peer ${source.toString()}`, {
1152
1150
  p2pMessageIdentifier: await attestation.p2pMessageLoggingIdentifier(),
1153
1151
  slot: attestation.slotNumber,
1154
1152
  archive: attestation.archive.toString(),
1155
1153
  source: source.toString()
1156
1154
  });
1157
- await this.mempools.attestationPool.addCheckpointAttestations([
1158
- attestation
1159
- ]);
1160
1155
  }
1161
- async processBlockFromPeer(payloadData, msgId, source) {
1162
- const validationFunc = async ()=>{
1163
- const block = BlockProposal.fromBuffer(payloadData);
1164
- const validationResult = await this.validateBlockProposal(source, block);
1165
- const isValid = validationResult.result === 'accept';
1166
- const pool = this.mempools.attestationPool;
1167
- const exists = isValid && await pool.hasBlockProposal(block);
1168
- const canAdd = isValid && await pool.canAddProposal(block);
1169
- this.logger.trace(`Validate propagated block proposal`, {
1170
- isValid,
1171
- exists,
1172
- canAdd,
1173
- [Attributes.SLOT_NUMBER]: block.slotNumber.toString(),
1174
- [Attributes.P2P_ID]: source.toString()
1156
+ /** Validates a checkpoint attestation and adds it to the pool. Penalizes the peer if validation fails. */ async validateAndStoreCheckpointAttestation(peerId, attestation) {
1157
+ const validationResult = await this.checkpointAttestationValidator.validate(attestation);
1158
+ if (validationResult.result === 'reject') {
1159
+ this.logger.warn(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
1160
+ this.peerManager.penalizePeer(peerId, validationResult.severity);
1161
+ return {
1162
+ result: TopicValidatorResult.Reject
1163
+ };
1164
+ }
1165
+ if (validationResult.result === 'ignore') {
1166
+ return {
1167
+ result: TopicValidatorResult.Ignore,
1168
+ obj: attestation
1169
+ };
1170
+ }
1171
+ // Try to add the attestation: this handles existence check, cap check, and adding in one call
1172
+ // count is the number of attestations by this signer for this slot (for duplicate detection)
1173
+ const slot = attestation.payload.header.slotNumber;
1174
+ const { added, alreadyExists, count } = await this.mempools.attestationPool.tryAddCheckpointAttestation(attestation);
1175
+ this.logger.trace(`Validate propagated checkpoint attestation`, {
1176
+ added,
1177
+ alreadyExists,
1178
+ count,
1179
+ [Attributes.SLOT_NUMBER]: slot.toString(),
1180
+ [Attributes.P2P_ID]: peerId.toString()
1181
+ });
1182
+ // Exact same attestation received, no need to re-broadcast
1183
+ if (alreadyExists) {
1184
+ return {
1185
+ result: TopicValidatorResult.Ignore,
1186
+ obj: attestation
1187
+ };
1188
+ }
1189
+ // Could not add (cap reached for signer), no need to re-broadcast
1190
+ if (!added) {
1191
+ this.logger.warn(`Dropping checkpoint attestation due to cap`, {
1192
+ slot: slot.toString(),
1193
+ archive: attestation.archive.toString(),
1194
+ source: peerId.toString(),
1195
+ attester: attestation.getSender()?.toString(),
1196
+ count
1175
1197
  });
1176
- if (validationResult.result === 'reject') {
1177
- return {
1178
- result: TopicValidatorResult.Reject
1179
- };
1180
- } else if (validationResult.result === 'ignore' || exists) {
1181
- return {
1182
- result: TopicValidatorResult.Ignore,
1183
- obj: block
1184
- };
1185
- } else if (!canAdd) {
1186
- this.peerManager.penalizePeer(source, PeerErrorSeverity.MidToleranceError);
1187
- this.logger.warn(`Penalizing peer for block proposal exceeding per-slot cap`, {
1188
- slot: block.slotNumber.toString(),
1189
- archive: block.archive.toString(),
1190
- source: source.toString()
1198
+ return {
1199
+ result: TopicValidatorResult.Ignore,
1200
+ obj: attestation
1201
+ };
1202
+ }
1203
+ // Check if this is a duplicate attestation (signer attested to a different proposal at the same slot)
1204
+ // count is the number of attestations by this signer for this slot
1205
+ if (count === 2) {
1206
+ const attester = attestation.getSender();
1207
+ if (attester) {
1208
+ this.logger.warn(`Detected duplicate attestation (equivocation) at slot ${slot}`, {
1209
+ slot: slot.toString(),
1210
+ archive: attestation.archive.toString(),
1211
+ source: peerId.toString(),
1212
+ attester: attester.toString()
1213
+ });
1214
+ this.duplicateAttestationCallback?.({
1215
+ slot,
1216
+ attester
1191
1217
  });
1192
- return {
1193
- result: TopicValidatorResult.Reject
1194
- };
1195
- } else {
1196
- return {
1197
- result: TopicValidatorResult.Accept,
1198
- obj: block
1199
- };
1200
1218
  }
1219
+ }
1220
+ // Attestation was added successfully - accept it so other nodes can also detect the equivocation
1221
+ return {
1222
+ result: TopicValidatorResult.Accept,
1223
+ obj: attestation
1201
1224
  };
1202
- const { result, obj: block } = await this.validateReceivedMessage(validationFunc, msgId, source, TopicType.block_proposal);
1203
- if (!result || !block) {
1225
+ }
1226
+ async processBlockFromPeer(payloadData, msgId, source) {
1227
+ const { result, obj: block, metadata: { isEquivocated } = {} } = await this.validateReceivedMessage(()=>this.validateAndStoreBlockProposal(source, BlockProposal.fromBuffer(payloadData)), msgId, source, TopicType.block_proposal);
1228
+ // If not accepted or equivocated, return
1229
+ if (result !== TopicValidatorResult.Accept || !block || isEquivocated) {
1204
1230
  return;
1205
1231
  }
1206
1232
  await this.processValidBlockProposal(block, source);
1207
1233
  }
1208
- // REVIEW: callback pattern https://github.com/AztecProtocol/aztec-packages/issues/7963
1234
+ /** Validates a block proposal. Triggers a penalization to the peer that sent it if invalid. Adds to the mempool if valid. */ async validateAndStoreBlockProposal(peerId, block) {
1235
+ const validationResult = await this.blockProposalValidator.validate(block);
1236
+ if (validationResult.result === 'reject') {
1237
+ this.logger.warn(`Penalizing peer ${peerId} for block proposal validation failure`);
1238
+ this.peerManager.penalizePeer(peerId, validationResult.severity);
1239
+ return {
1240
+ result: TopicValidatorResult.Reject
1241
+ };
1242
+ }
1243
+ if (validationResult.result === 'ignore') {
1244
+ return {
1245
+ result: TopicValidatorResult.Ignore,
1246
+ obj: block
1247
+ };
1248
+ }
1249
+ // Try to add the proposal: this handles existence check, cap check, and adding in one call
1250
+ const { added, alreadyExists, count } = await this.mempools.attestationPool.tryAddBlockProposal(block);
1251
+ const isEquivocated = count !== undefined && count > 1;
1252
+ // Duplicate proposal received, no need to re-broadcast
1253
+ if (alreadyExists) {
1254
+ this.logger.debug(`Ignoring duplicate block proposal received`, {
1255
+ ...block.toBlockInfo(),
1256
+ indexWithinCheckpoint: block.indexWithinCheckpoint,
1257
+ proposer: block.getSender()?.toString(),
1258
+ source: peerId.toString()
1259
+ });
1260
+ return {
1261
+ result: TopicValidatorResult.Ignore,
1262
+ obj: block,
1263
+ metadata: {
1264
+ isEquivocated
1265
+ }
1266
+ };
1267
+ }
1268
+ // Too many blocks received for this slot and index, penalize peer and do not re-broadcast
1269
+ if (!added) {
1270
+ this.peerManager.penalizePeer(peerId, PeerErrorSeverity.HighToleranceError);
1271
+ this.logger.warn(`Penalizing peer for block proposal exceeding per-position cap`, {
1272
+ ...block.toBlockInfo(),
1273
+ indexWithinCheckpoint: block.indexWithinCheckpoint,
1274
+ count,
1275
+ proposer: block.getSender()?.toString(),
1276
+ source: peerId.toString()
1277
+ });
1278
+ return {
1279
+ result: TopicValidatorResult.Reject,
1280
+ metadata: {
1281
+ isEquivocated
1282
+ }
1283
+ };
1284
+ }
1285
+ // If this was a duplicate proposal, do not process it, but do invoke the duplicate callback,
1286
+ // and do re-broadcast it so other nodes in the network know to slash the proposer
1287
+ if (isEquivocated) {
1288
+ const proposer = block.getSender();
1289
+ this.logger.warn(`Detected duplicate block proposal (equivocation) at slot ${block.slotNumber}`, {
1290
+ ...block.toBlockInfo(),
1291
+ source: peerId.toString(),
1292
+ proposer: proposer?.toString()
1293
+ });
1294
+ // Invoke the duplicate callback on the first duplicate spotted only
1295
+ if (proposer && count === 2) {
1296
+ this.duplicateProposalCallback?.({
1297
+ slot: block.slotNumber,
1298
+ proposer,
1299
+ type: 'block'
1300
+ });
1301
+ }
1302
+ return {
1303
+ result: TopicValidatorResult.Accept,
1304
+ obj: block,
1305
+ metadata: {
1306
+ isEquivocated
1307
+ }
1308
+ };
1309
+ }
1310
+ // Otherwise, we're good to go!
1311
+ return {
1312
+ result: TopicValidatorResult.Accept,
1313
+ obj: block
1314
+ };
1315
+ }
1209
1316
  // REFACTOR(palla): This method should be moved to the p2p_client or to a separate component,
1210
1317
  // should not be here as it does not deal with p2p networking.
1211
1318
  async processValidBlockProposal(block, sender) {
@@ -1215,23 +1322,8 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
1215
1322
  source: sender.toString(),
1216
1323
  ...block.toBlockInfo()
1217
1324
  });
1218
- // Attempt to add proposal
1219
- try {
1220
- await this.mempools.attestationPool.addBlockProposal(block);
1221
- } catch (err) {
1222
- // Drop proposals if we hit per-slot cap in the attestation pool; rethrow unknown errors
1223
- if (err instanceof ProposalSlotCapExceededError) {
1224
- this.logger.warn(`Dropping block proposal due to per-slot proposal cap`, {
1225
- slot: String(slot),
1226
- archive: block.archive.toString(),
1227
- error: err.message
1228
- });
1229
- return;
1230
- }
1231
- throw err;
1232
- }
1233
- // Mark the txs in this proposal as non-evictable
1234
- await this.mempools.txPool.markTxsAsNonEvictable(block.txHashes);
1325
+ // Mark the txs in this proposal as protected
1326
+ await this.mempools.txPool.protectTxs(block.txHashes, block.blockHeader);
1235
1327
  // Call the block received callback to validate the proposal.
1236
1328
  // Note: Validators do NOT attest to individual blocks, only to checkpoint proposals.
1237
1329
  const isValid = await this.blockReceivedCallback(block, sender);
@@ -1243,57 +1335,135 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
1243
1335
  * Handle a gossiped checkpoint proposal.
1244
1336
  * Validates and processes the checkpoint proposal, then triggers the callback for attestation.
1245
1337
  */ async handleGossipedCheckpointProposal(payloadData, msgId, source) {
1246
- // TODO(palla/mbps): This pattern is repeated across multiple message handlers, consider abstracting it.
1247
- const validationFunc = async ()=>{
1248
- const checkpoint = CheckpointProposal.fromBuffer(payloadData);
1249
- const validationResult = await this.validateCheckpointProposal(source, checkpoint);
1250
- const isValid = validationResult.result === 'accept';
1251
- const pool = this.mempools.attestationPool;
1252
- const exists = isValid && await pool.hasCheckpointProposal(checkpoint);
1253
- const canAdd = isValid && await pool.canAddCheckpointProposal(checkpoint);
1254
- this.logger.trace(`Validate propagated checkpoint proposal`, {
1255
- isValid,
1256
- exists,
1257
- canAdd,
1338
+ const { result, obj: checkpoint, metadata: { isEquivocated, processBlock } = {} } = await this.validateReceivedMessage(()=>this.validateAndStoreCheckpointProposal(source, CheckpointProposal.fromBuffer(payloadData)), msgId, source, TopicType.checkpoint_proposal);
1339
+ // If the checkpoint contained a valid last block, we process it even if the checkpoint itself is to be rejected
1340
+ // TODO(palla/mbps): Is this ok? Should we be considering a block from a checkpoint that was equivocated?
1341
+ if (processBlock && checkpoint?.getBlockProposal()) {
1342
+ await this.processValidBlockProposal(checkpoint.getBlockProposal(), source);
1343
+ }
1344
+ if (result !== TopicValidatorResult.Accept || !checkpoint || isEquivocated) {
1345
+ return;
1346
+ }
1347
+ await this.processValidCheckpointProposal(checkpoint.toCore(), source);
1348
+ }
1349
+ /**
1350
+ * Validates a checkpoint proposal. Penalizes peer if validation fails. Adds the checkpoint and
1351
+ * its last block (if present) to the mempool if valid. Triggers equivocation detection on both.
1352
+ */ async validateAndStoreCheckpointProposal(peerId, checkpoint) {
1353
+ const validationResult = await this.checkpointProposalValidator.validate(checkpoint);
1354
+ if (validationResult.result === 'reject') {
1355
+ this.logger.warn(`Penalizing peer ${peerId} for checkpoint proposal validation failure`);
1356
+ this.peerManager.penalizePeer(peerId, validationResult.severity);
1357
+ return {
1358
+ result: TopicValidatorResult.Reject
1359
+ };
1360
+ }
1361
+ if (validationResult.result === 'ignore') {
1362
+ return {
1363
+ result: TopicValidatorResult.Ignore,
1364
+ obj: checkpoint
1365
+ };
1366
+ }
1367
+ // Extract and try to add the block proposal first if present
1368
+ const blockProposal = checkpoint.getBlockProposal();
1369
+ let processBlock = false;
1370
+ if (blockProposal) {
1371
+ this.logger.debug(`Validating block proposal from propagated checkpoint`, {
1258
1372
  [Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
1259
- [Attributes.P2P_ID]: source.toString()
1373
+ [Attributes.P2P_ID]: peerId.toString()
1260
1374
  });
1261
- if (validationResult.result === 'reject') {
1262
- return {
1263
- result: TopicValidatorResult.Reject
1264
- };
1265
- } else if (validationResult.result === 'ignore' || exists) {
1266
- return {
1267
- result: TopicValidatorResult.Ignore,
1268
- obj: checkpoint
1269
- };
1270
- } else if (!canAdd) {
1271
- this.peerManager.penalizePeer(source, PeerErrorSeverity.MidToleranceError);
1272
- this.logger.warn(`Penalizing peer for checkpoint proposal exceeding per-slot cap`, {
1273
- slot: checkpoint.slotNumber.toString(),
1274
- archive: checkpoint.archive.toString(),
1275
- source: source.toString()
1375
+ const { result, obj, metadata: { isEquivocated } = {} } = await this.validateAndStoreBlockProposal(peerId, blockProposal);
1376
+ if (result === TopicValidatorResult.Reject || !obj || isEquivocated) {
1377
+ this.logger.debug(`Rejecting checkpoint due to invalid last block proposal`, {
1378
+ [Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
1379
+ [Attributes.P2P_ID]: peerId.toString(),
1380
+ isEquivocated,
1381
+ result
1276
1382
  });
1277
1383
  return {
1278
1384
  result: TopicValidatorResult.Reject
1279
1385
  };
1280
- } else {
1281
- return {
1282
- result: TopicValidatorResult.Accept,
1283
- obj: checkpoint
1284
- };
1386
+ } else if (result === TopicValidatorResult.Accept && obj && !isEquivocated) {
1387
+ processBlock = true;
1285
1388
  }
1286
- };
1287
- const { result, obj: checkpoint } = await this.validateReceivedMessage(validationFunc, msgId, source, TopicType.checkpoint_proposal);
1288
- if (result !== TopicValidatorResult.Accept || !checkpoint) {
1289
- return;
1290
1389
  }
1291
- await this.processValidCheckpointProposal(checkpoint, source);
1390
+ // Try to add the checkpoint proposal core: this handles existence check, cap check, and adding in one call
1391
+ const checkpointCore = checkpoint.toCore();
1392
+ const tryAddResult = await this.mempools.attestationPool.tryAddCheckpointProposal(checkpointCore);
1393
+ const { added, alreadyExists, count } = tryAddResult;
1394
+ const isEquivocated = count !== undefined && count > 1;
1395
+ // Duplicate proposal received, do not re-broadcast
1396
+ if (alreadyExists) {
1397
+ this.logger.debug(`Ignoring duplicate checkpoint proposal received`, {
1398
+ ...checkpoint.toCheckpointInfo(),
1399
+ source: peerId.toString()
1400
+ });
1401
+ return {
1402
+ result: TopicValidatorResult.Ignore,
1403
+ obj: checkpoint,
1404
+ metadata: {
1405
+ isEquivocated,
1406
+ processBlock
1407
+ }
1408
+ };
1409
+ }
1410
+ // Too many checkpoint proposals received for this slot, penalize peer and do not re-broadcast
1411
+ // Note: We still return the checkpoint obj so the lastBlock can be processed if valid
1412
+ if (!added) {
1413
+ this.peerManager.penalizePeer(peerId, PeerErrorSeverity.HighToleranceError);
1414
+ this.logger.warn(`Penalizing peer for checkpoint proposal exceeding per-slot cap`, {
1415
+ ...checkpoint.toCheckpointInfo(),
1416
+ count,
1417
+ source: peerId.toString()
1418
+ });
1419
+ return {
1420
+ result: TopicValidatorResult.Reject,
1421
+ obj: checkpoint,
1422
+ metadata: {
1423
+ isEquivocated,
1424
+ processBlock
1425
+ }
1426
+ };
1427
+ }
1428
+ // If this was a duplicate proposal, do not process it, but do invoke the duplicate callback,
1429
+ // and do re-broadcast it so other nodes in the network know to slash the proposer
1430
+ if (isEquivocated) {
1431
+ const proposer = checkpoint.getSender();
1432
+ this.logger.warn(`Detected duplicate checkpoint proposal (equivocation) at slot ${checkpoint.slotNumber}`, {
1433
+ ...checkpoint.toCheckpointInfo(),
1434
+ source: peerId.toString(),
1435
+ proposer: proposer?.toString()
1436
+ });
1437
+ // Invoke the duplicate callback on the first duplicate spotted only
1438
+ if (proposer && count === 2) {
1439
+ this.duplicateProposalCallback?.({
1440
+ slot: checkpoint.slotNumber,
1441
+ proposer,
1442
+ type: 'checkpoint'
1443
+ });
1444
+ }
1445
+ return {
1446
+ result: TopicValidatorResult.Accept,
1447
+ obj: checkpoint,
1448
+ metadata: {
1449
+ isEquivocated,
1450
+ processBlock
1451
+ }
1452
+ };
1453
+ }
1454
+ // Otherwise, we're good to go!
1455
+ return {
1456
+ result: TopicValidatorResult.Accept,
1457
+ obj: checkpoint,
1458
+ metadata: {
1459
+ processBlock,
1460
+ isEquivocated
1461
+ }
1462
+ };
1292
1463
  }
1293
1464
  /**
1294
1465
  * Process a validated checkpoint proposal.
1295
- * Extracts and processes the last block proposal (if present) first, then processes the checkpoint.
1296
- * The block callback is invoked before the checkpoint callback.
1466
+ * Note: The proposal was already added to the pool by tryAddCheckpointProposal in handleGossipedCheckpointProposal.
1297
1467
  */ async processValidCheckpointProposal(checkpoint, sender) {
1298
1468
  const slot = checkpoint.slotNumber;
1299
1469
  this.logger.verbose(`Received checkpoint proposal for slot ${slot} from external peer ${sender.toString()}.`, {
@@ -1302,33 +1472,12 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
1302
1472
  archive: checkpoint.archive.toString(),
1303
1473
  source: sender.toString()
1304
1474
  });
1305
- // Extract block proposal before adding to pool (pool stores them separately)
1306
- const blockProposal = checkpoint.getBlockProposal();
1307
- // Add proposal to the pool (this extracts and stores block proposal separately)
1308
- await this.mempools.attestationPool.addCheckpointProposal(checkpoint);
1309
- // Mark txs as non-evictable if present (from the last block)
1310
- if (checkpoint.txHashes.length > 0) {
1311
- await this.mempools.txPool.markTxsAsNonEvictable(checkpoint.txHashes);
1312
- }
1313
- // If there was a last block proposal, invoke the block callback first for validation.
1314
- // Note: The block proposal is already stored in the pool by addCheckpointProposal.
1315
- if (blockProposal) {
1316
- const isValid = await this.blockReceivedCallback(blockProposal, sender);
1317
- if (!isValid) {
1318
- this.logger.warn(`Block proposal from checkpoint failed validation`, {
1319
- slot: slot.toString(),
1320
- archive: checkpoint.archive.toString(),
1321
- blockNumber: blockProposal.blockNumber.toString()
1322
- });
1323
- return;
1324
- }
1325
- }
1326
1475
  // Call the checkpoint received callback with the core version (without lastBlock)
1327
1476
  // to validate and potentially generate attestations
1328
- const attestations = await this.checkpointReceivedCallback(checkpoint.toCore(), sender);
1477
+ const attestations = await this.checkpointReceivedCallback(checkpoint, sender);
1329
1478
  if (attestations && attestations.length > 0) {
1330
1479
  // If the callback returned attestations, add them to the pool and propagate them
1331
- await this.mempools.attestationPool.addCheckpointAttestations(attestations);
1480
+ await this.mempools.attestationPool.addOwnCheckpointAttestations(attestations);
1332
1481
  for (const attestation of attestations){
1333
1482
  await this.propagate(attestation);
1334
1483
  }
@@ -1497,33 +1646,11 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
1497
1646
  }
1498
1647
  }
1499
1648
  createRequestedTxValidator() {
1500
- return createTxReqRespValidator(this.proofVerifier, {
1649
+ return createTxValidatorForReqResponseReceivedTxs(this.proofVerifier, {
1501
1650
  l1ChainId: this.config.l1ChainId,
1502
1651
  rollupVersion: this.config.rollupVersion
1503
1652
  });
1504
1653
  }
1505
- async validatePropagatedTx(tx, peerId) {
1506
- const currentBlockNumber = await this.archiver.getBlockNumber();
1507
- // We accept transactions if they are not expired by the next slot (checked based on the IncludeByTimestamp field)
1508
- const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
1509
- const messageValidators = await this.createMessageValidators(currentBlockNumber, nextSlotTimestamp);
1510
- for (const validator of messageValidators){
1511
- const outcome = await this.runValidations(tx, validator);
1512
- if (outcome.allPassed) {
1513
- continue;
1514
- }
1515
- const { name } = outcome.failure;
1516
- let { severity } = outcome.failure;
1517
- // Double spend validator has a special case handler
1518
- if (name === 'doubleSpendValidator') {
1519
- const txBlockNumber = BlockNumber(currentBlockNumber + 1); // tx is expected to be in the next block
1520
- severity = await this.handleDoubleSpendFailure(tx, txBlockNumber);
1521
- }
1522
- this.peerManager.penalizePeer(peerId, severity);
1523
- return false;
1524
- }
1525
- return true;
1526
- }
1527
1654
  async getGasFees(blockNumber) {
1528
1655
  if (blockNumber === this.feesCache?.blockNumber) {
1529
1656
  return this.feesCache.gasFees;
@@ -1550,38 +1677,27 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
1550
1677
  peerScoring: this.peerManager
1551
1678
  };
1552
1679
  }
1553
- async validate(txs) {
1554
- const currentBlockNumber = await this.archiver.getBlockNumber();
1555
- // We accept transactions if they are not expired by the next slot (checked based on the IncludeByTimestamp field)
1556
- const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
1557
- const messageValidators = await this.createMessageValidators(currentBlockNumber, nextSlotTimestamp);
1558
- await Promise.all(txs.map(async (tx)=>{
1559
- for (const validator of messageValidators){
1560
- const outcome = await this.runValidations(tx, validator);
1561
- if (!outcome.allPassed) {
1562
- throw new Error('Invalid tx detected', {
1563
- cause: {
1564
- outcome
1565
- }
1566
- });
1567
- }
1568
- }
1680
+ async validateTxsReceivedInBlockProposal(txs) {
1681
+ const validator = createTxValidatorForBlockProposalReceivedTxs(this.proofVerifier, {
1682
+ l1ChainId: this.config.l1ChainId,
1683
+ rollupVersion: this.config.rollupVersion
1684
+ }, this.logger.getBindings());
1685
+ const results = await Promise.all(txs.map(async (tx)=>{
1686
+ const result = await validator.validateTx(tx);
1687
+ return result.result !== 'invalid';
1569
1688
  }));
1689
+ if (results.some((value)=>value === false)) {
1690
+ throw new Error('Invalid tx detected');
1691
+ }
1570
1692
  }
1571
- /**
1572
- * Create message validators for the given block number and timestamp.
1573
- *
1574
- * Each validator is a pair of a validator and a severity.
1575
- * If a validator fails, the peer is penalized with the severity of the validator.
1576
- *
1577
- * @param currentBlockNumber - The current synced block number.
1578
- * @param nextSlotTimestamp - The timestamp of the next slot (used to validate txs are not expired).
1579
- * @returns The message validators.
1580
- */ async createMessageValidators(currentBlockNumber, nextSlotTimestamp) {
1693
+ /** Creates the first stage (fast) validators for gossiped transactions. */ async createFirstStageMessageValidators(currentBlockNumber, nextSlotTimestamp) {
1581
1694
  const gasFees = await this.getGasFees(currentBlockNumber);
1582
1695
  const allowedInSetup = this.config.txPublicSetupAllowList ?? await getDefaultAllowedSetupFunctions();
1583
- const blockNumberInWhichTheTxIsConsideredToBeIncluded = BlockNumber(currentBlockNumber + 1);
1584
- return createTxMessageValidators(nextSlotTimestamp, blockNumberInWhichTheTxIsConsideredToBeIncluded, this.worldStateSynchronizer, gasFees, this.config.l1ChainId, this.config.rollupVersion, protocolContractsHash, this.archiver, this.proofVerifier, !this.config.disableTransactions, allowedInSetup, this.logger.getBindings());
1696
+ const blockNumber = BlockNumber(currentBlockNumber + 1);
1697
+ return createFirstStageTxValidationsForGossipedTransactions(nextSlotTimestamp, blockNumber, this.worldStateSynchronizer, gasFees, this.config.l1ChainId, this.config.rollupVersion, protocolContractsHash, this.archiver, !this.config.disableTransactions, allowedInSetup, this.logger.getBindings());
1698
+ }
1699
+ /** Creates the second stage (expensive proof verification) validators for gossiped transactions. */ createSecondStageMessageValidators() {
1700
+ return createSecondStageTxValidationsForGossipedTransactions(this.proofVerifier, this.logger.getBindings());
1585
1701
  }
1586
1702
  /**
1587
1703
  * Run validations on a tx.
@@ -1655,33 +1771,7 @@ _dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
1655
1771
  */ async validateCheckpointAttestation(peerId, attestation) {
1656
1772
  const result = await this.checkpointAttestationValidator.validate(attestation);
1657
1773
  if (result.result === 'reject') {
1658
- this.logger.debug(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
1659
- this.peerManager.penalizePeer(peerId, result.severity);
1660
- }
1661
- return result;
1662
- }
1663
- /**
1664
- * Validate a block proposal.
1665
- *
1666
- * @param block - The block proposal to validate.
1667
- * @returns True if the block proposal is valid, false otherwise.
1668
- */ async validateBlockProposal(peerId, block) {
1669
- const result = await this.blockProposalValidator.validate(block);
1670
- if (result.result === 'reject') {
1671
- this.logger.debug(`Penalizing peer ${peerId} for block proposal validation failure`);
1672
- this.peerManager.penalizePeer(peerId, result.severity);
1673
- }
1674
- return result;
1675
- }
1676
- /**
1677
- * Validate a checkpoint proposal.
1678
- *
1679
- * @param checkpoint - The checkpoint proposal to validate.
1680
- * @returns True if the checkpoint proposal is valid, false otherwise.
1681
- */ async validateCheckpointProposal(peerId, checkpoint) {
1682
- const result = await this.checkpointProposalValidator.validate(checkpoint);
1683
- if (result.result === 'reject') {
1684
- this.logger.debug(`Penalizing peer ${peerId} for checkpoint proposal validation failure`);
1774
+ this.logger.warn(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
1685
1775
  this.peerManager.penalizePeer(peerId, result.severity);
1686
1776
  }
1687
1777
  return result;