@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.
- package/dest/bootstrap/bootstrap.d.ts +4 -3
- package/dest/bootstrap/bootstrap.d.ts.map +1 -1
- package/dest/bootstrap/bootstrap.js +4 -4
- package/dest/client/factory.d.ts +10 -10
- package/dest/client/factory.d.ts.map +1 -1
- package/dest/client/factory.js +45 -18
- package/dest/client/interface.d.ts +61 -35
- package/dest/client/interface.d.ts.map +1 -1
- package/dest/client/p2p_client.d.ts +47 -60
- package/dest/client/p2p_client.d.ts.map +1 -1
- package/dest/client/p2p_client.js +600 -313
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.d.ts +2 -0
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.d.ts.map +1 -0
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.js +304 -0
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.d.ts +73 -0
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.d.ts.map +1 -0
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.js +8 -0
- package/dest/config.d.ts +35 -7
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +23 -9
- package/dest/errors/tx-pool.error.d.ts +8 -0
- package/dest/errors/tx-pool.error.d.ts.map +1 -0
- package/dest/errors/tx-pool.error.js +9 -0
- package/dest/index.d.ts +2 -1
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +1 -0
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +111 -76
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.js +441 -3
- package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.d.ts +2 -2
- package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.js +527 -287
- package/dest/mem_pools/attestation_pool/index.d.ts +2 -3
- package/dest/mem_pools/attestation_pool/index.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/index.js +1 -2
- package/dest/mem_pools/attestation_pool/mocks.d.ts +9 -6
- package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/mocks.js +16 -12
- package/dest/mem_pools/index.d.ts +3 -2
- package/dest/mem_pools/index.d.ts.map +1 -1
- package/dest/mem_pools/index.js +1 -1
- package/dest/mem_pools/instrumentation.d.ts +1 -1
- package/dest/mem_pools/instrumentation.d.ts.map +1 -1
- package/dest/mem_pools/instrumentation.js +5 -14
- package/dest/mem_pools/interface.d.ts +5 -5
- package/dest/mem_pools/interface.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts +15 -10
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.js +91 -50
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts +19 -5
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.js +59 -3
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts +79 -5
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.js +47 -0
- package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts +16 -0
- package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.js +122 -0
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts +2 -2
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.js +3 -3
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts +4 -4
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.js +2 -0
- package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts +2 -2
- package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.d.ts +25 -0
- package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.js +57 -0
- package/dest/mem_pools/tx_pool_v2/archive/index.d.ts +2 -0
- package/dest/mem_pools/tx_pool_v2/archive/index.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/archive/index.js +1 -0
- package/dest/mem_pools/tx_pool_v2/archive/tx_archive.d.ts +43 -0
- package/dest/mem_pools/tx_pool_v2/archive/tx_archive.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/archive/tx_archive.js +103 -0
- package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts +104 -0
- package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/deleted_pool.js +251 -0
- package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts +47 -0
- package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.js +128 -0
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts +17 -0
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.js +93 -0
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts +19 -0
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.js +97 -0
- package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts +10 -0
- package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/index.js +11 -0
- package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts +174 -0
- package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/interfaces.js +25 -0
- package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.d.ts +15 -0
- package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.js +65 -0
- package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.d.ts +17 -0
- package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.js +93 -0
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts +16 -0
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.js +78 -0
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts +20 -0
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.js +73 -0
- package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts +15 -0
- package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.js +19 -0
- package/dest/mem_pools/tx_pool_v2/index.d.ts +6 -0
- package/dest/mem_pools/tx_pool_v2/index.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/index.js +5 -0
- package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts +15 -0
- package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/instrumentation.js +43 -0
- package/dest/mem_pools/tx_pool_v2/interfaces.d.ts +211 -0
- package/dest/mem_pools/tx_pool_v2/interfaces.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/interfaces.js +9 -0
- package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts +119 -0
- package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/tx_metadata.js +193 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.d.ts +26 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_bench_metrics.js +70 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts +108 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_indices.js +354 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts +60 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.js +161 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts +77 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.js +905 -0
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts +4 -4
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts.map +1 -1
- package/dest/msg_validators/attestation_validator/attestation_validator.js +51 -18
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts +7 -7
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts.map +1 -1
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.js +22 -13
- package/dest/msg_validators/clock_tolerance.d.ts +21 -0
- package/dest/msg_validators/clock_tolerance.d.ts.map +1 -0
- package/dest/msg_validators/clock_tolerance.js +37 -0
- package/dest/msg_validators/index.d.ts +2 -2
- package/dest/msg_validators/index.d.ts.map +1 -1
- package/dest/msg_validators/index.js +1 -1
- package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts +9 -0
- package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/block_proposal_validator.js +6 -0
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts +9 -0
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.js +6 -0
- package/dest/msg_validators/proposal_validator/index.d.ts +4 -0
- package/dest/msg_validators/proposal_validator/index.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/index.js +3 -0
- package/dest/msg_validators/proposal_validator/proposal_validator.d.ts +13 -0
- package/dest/msg_validators/proposal_validator/proposal_validator.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/proposal_validator.js +104 -0
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts +23 -0
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.js +212 -0
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts +4 -4
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.js +3 -3
- package/dest/msg_validators/tx_validator/archive_cache.d.ts +3 -3
- package/dest/msg_validators/tx_validator/archive_cache.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/archive_cache.js +1 -1
- package/dest/msg_validators/tx_validator/block_header_validator.d.ts +20 -6
- package/dest/msg_validators/tx_validator/block_header_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/block_header_validator.js +4 -3
- package/dest/msg_validators/tx_validator/data_validator.d.ts +3 -1
- package/dest/msg_validators/tx_validator/data_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/data_validator.js +4 -1
- package/dest/msg_validators/tx_validator/double_spend_validator.d.ts +15 -4
- package/dest/msg_validators/tx_validator/double_spend_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/double_spend_validator.js +7 -6
- package/dest/msg_validators/tx_validator/factory.d.ts +118 -5
- package/dest/msg_validators/tx_validator/factory.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/factory.js +228 -57
- package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts +10 -0
- package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/fee_payer_balance.js +20 -0
- package/dest/msg_validators/tx_validator/gas_validator.d.ts +59 -3
- package/dest/msg_validators/tx_validator/gas_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/gas_validator.js +84 -52
- package/dest/msg_validators/tx_validator/index.d.ts +3 -1
- package/dest/msg_validators/tx_validator/index.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/index.js +2 -0
- package/dest/msg_validators/tx_validator/metadata_validator.d.ts +3 -2
- package/dest/msg_validators/tx_validator/metadata_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/metadata_validator.js +2 -2
- package/dest/msg_validators/tx_validator/nullifier_cache.d.ts +14 -0
- package/dest/msg_validators/tx_validator/nullifier_cache.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/nullifier_cache.js +24 -0
- package/dest/msg_validators/tx_validator/phases_validator.d.ts +3 -2
- package/dest/msg_validators/tx_validator/phases_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/phases_validator.js +3 -3
- package/dest/msg_validators/tx_validator/size_validator.d.ts +8 -0
- package/dest/msg_validators/tx_validator/size_validator.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/size_validator.js +23 -0
- package/dest/msg_validators/tx_validator/timestamp_validator.d.ts +22 -5
- package/dest/msg_validators/tx_validator/timestamp_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/timestamp_validator.js +8 -8
- package/dest/msg_validators/tx_validator/tx_permitted_validator.d.ts +3 -2
- package/dest/msg_validators/tx_validator/tx_permitted_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/tx_permitted_validator.js +2 -2
- package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts +3 -2
- package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/tx_proof_validator.js +2 -2
- package/dest/services/data_store.d.ts +1 -1
- package/dest/services/data_store.d.ts.map +1 -1
- package/dest/services/data_store.js +10 -6
- package/dest/services/discv5/discV5_service.js +1 -1
- package/dest/services/dummy_service.d.ts +28 -4
- package/dest/services/dummy_service.d.ts.map +1 -1
- package/dest/services/dummy_service.js +49 -1
- package/dest/services/encoding.d.ts +3 -3
- package/dest/services/encoding.d.ts.map +1 -1
- package/dest/services/encoding.js +16 -14
- package/dest/services/gossipsub/index.d.ts +3 -0
- package/dest/services/gossipsub/index.d.ts.map +1 -0
- package/dest/services/gossipsub/index.js +2 -0
- package/dest/services/gossipsub/scoring.d.ts +21 -3
- package/dest/services/gossipsub/scoring.d.ts.map +1 -1
- package/dest/services/gossipsub/scoring.js +24 -7
- package/dest/services/gossipsub/topic_score_params.d.ts +173 -0
- package/dest/services/gossipsub/topic_score_params.d.ts.map +1 -0
- package/dest/services/gossipsub/topic_score_params.js +346 -0
- package/dest/services/index.d.ts +2 -1
- package/dest/services/index.d.ts.map +1 -1
- package/dest/services/index.js +1 -0
- package/dest/services/libp2p/instrumentation.d.ts +1 -1
- package/dest/services/libp2p/instrumentation.d.ts.map +1 -1
- package/dest/services/libp2p/instrumentation.js +30 -72
- package/dest/services/libp2p/libp2p_service.d.ts +115 -40
- package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
- package/dest/services/libp2p/libp2p_service.js +1020 -369
- package/dest/services/peer-manager/metrics.d.ts +2 -2
- package/dest/services/peer-manager/metrics.d.ts.map +1 -1
- package/dest/services/peer-manager/metrics.js +21 -26
- package/dest/services/peer-manager/peer_manager.d.ts +2 -2
- package/dest/services/peer-manager/peer_manager.d.ts.map +1 -1
- package/dest/services/peer-manager/peer_manager.js +0 -10
- package/dest/services/peer-manager/peer_scoring.d.ts +1 -1
- package/dest/services/peer-manager/peer_scoring.d.ts.map +1 -1
- package/dest/services/peer-manager/peer_scoring.js +32 -6
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts +48 -0
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts.map +1 -0
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.js +539 -0
- package/dest/services/reqresp/batch-tx-requester/config.d.ts +17 -0
- package/dest/services/reqresp/batch-tx-requester/config.d.ts.map +1 -0
- package/dest/services/reqresp/batch-tx-requester/config.js +27 -0
- package/dest/services/reqresp/batch-tx-requester/interface.d.ts +46 -0
- package/dest/services/reqresp/batch-tx-requester/interface.d.ts.map +1 -0
- package/dest/services/reqresp/batch-tx-requester/interface.js +1 -0
- package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts +34 -0
- package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts.map +1 -0
- package/dest/services/reqresp/batch-tx-requester/missing_txs.js +130 -0
- package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts +60 -0
- package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts.map +1 -0
- package/dest/services/reqresp/batch-tx-requester/peer_collection.js +173 -0
- package/dest/services/reqresp/batch-tx-requester/tx_validator.d.ts +20 -0
- package/dest/services/reqresp/batch-tx-requester/tx_validator.d.ts.map +1 -0
- package/dest/services/reqresp/batch-tx-requester/tx_validator.js +21 -0
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts +22 -3
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts.map +1 -1
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.js +63 -4
- package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts +2 -1
- package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts.map +1 -1
- package/dest/services/reqresp/connection-sampler/connection_sampler.js +12 -0
- package/dest/services/reqresp/constants.d.ts +12 -0
- package/dest/services/reqresp/constants.d.ts.map +1 -0
- package/dest/services/reqresp/constants.js +7 -0
- package/dest/services/reqresp/interface.d.ts +12 -1
- package/dest/services/reqresp/interface.d.ts.map +1 -1
- package/dest/services/reqresp/interface.js +15 -1
- package/dest/services/reqresp/metrics.d.ts +6 -5
- package/dest/services/reqresp/metrics.d.ts.map +1 -1
- package/dest/services/reqresp/metrics.js +17 -21
- package/dest/services/reqresp/protocols/block_txs/bitvector.d.ts +5 -1
- package/dest/services/reqresp/protocols/block_txs/bitvector.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/block_txs/bitvector.js +12 -0
- package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts +7 -5
- package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/block_txs/block_txs_handler.js +27 -9
- package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts +29 -6
- package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.js +59 -13
- package/dest/services/reqresp/protocols/status.d.ts +1 -1
- package/dest/services/reqresp/protocols/status.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/status.js +4 -1
- package/dest/services/reqresp/protocols/tx.d.ts +7 -1
- package/dest/services/reqresp/protocols/tx.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/tx.js +20 -0
- package/dest/services/reqresp/reqresp.d.ts +6 -1
- package/dest/services/reqresp/reqresp.d.ts.map +1 -1
- package/dest/services/reqresp/reqresp.js +473 -51
- package/dest/services/service.d.ts +56 -4
- package/dest/services/service.d.ts.map +1 -1
- package/dest/services/tx_collection/config.d.ts +22 -1
- package/dest/services/tx_collection/config.d.ts.map +1 -1
- package/dest/services/tx_collection/config.js +55 -1
- package/dest/services/tx_collection/fast_tx_collection.d.ts +7 -4
- package/dest/services/tx_collection/fast_tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/fast_tx_collection.js +71 -44
- package/dest/services/tx_collection/file_store_tx_collection.d.ts +53 -0
- package/dest/services/tx_collection/file_store_tx_collection.d.ts.map +1 -0
- package/dest/services/tx_collection/file_store_tx_collection.js +167 -0
- package/dest/services/tx_collection/file_store_tx_source.d.ts +37 -0
- package/dest/services/tx_collection/file_store_tx_source.d.ts.map +1 -0
- package/dest/services/tx_collection/file_store_tx_source.js +90 -0
- package/dest/services/tx_collection/index.d.ts +3 -1
- package/dest/services/tx_collection/index.d.ts.map +1 -1
- package/dest/services/tx_collection/index.js +2 -0
- package/dest/services/tx_collection/instrumentation.d.ts +1 -1
- package/dest/services/tx_collection/instrumentation.d.ts.map +1 -1
- package/dest/services/tx_collection/instrumentation.js +11 -13
- package/dest/services/tx_collection/missing_txs_tracker.d.ts +32 -0
- package/dest/services/tx_collection/missing_txs_tracker.d.ts.map +1 -0
- package/dest/services/tx_collection/missing_txs_tracker.js +27 -0
- package/dest/services/tx_collection/proposal_tx_collector.d.ts +49 -0
- package/dest/services/tx_collection/proposal_tx_collector.d.ts.map +1 -0
- package/dest/services/tx_collection/proposal_tx_collector.js +50 -0
- package/dest/services/tx_collection/slow_tx_collection.d.ts +9 -5
- package/dest/services/tx_collection/slow_tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/slow_tx_collection.js +60 -26
- package/dest/services/tx_collection/tx_collection.d.ts +29 -16
- package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_collection.js +79 -7
- package/dest/services/tx_collection/tx_collection_sink.d.ts +18 -8
- package/dest/services/tx_collection/tx_collection_sink.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_collection_sink.js +26 -29
- package/dest/services/tx_collection/tx_source.d.ts +8 -3
- package/dest/services/tx_collection/tx_source.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_source.js +19 -2
- package/dest/services/tx_file_store/config.d.ts +16 -0
- package/dest/services/tx_file_store/config.d.ts.map +1 -0
- package/dest/services/tx_file_store/config.js +22 -0
- package/dest/services/tx_file_store/index.d.ts +4 -0
- package/dest/services/tx_file_store/index.d.ts.map +1 -0
- package/dest/services/tx_file_store/index.js +3 -0
- package/dest/services/tx_file_store/instrumentation.d.ts +15 -0
- package/dest/services/tx_file_store/instrumentation.d.ts.map +1 -0
- package/dest/services/tx_file_store/instrumentation.js +29 -0
- package/dest/services/tx_file_store/tx_file_store.d.ts +48 -0
- package/dest/services/tx_file_store/tx_file_store.d.ts.map +1 -0
- package/dest/services/tx_file_store/tx_file_store.js +152 -0
- package/dest/services/tx_provider.d.ts +6 -6
- package/dest/services/tx_provider.d.ts.map +1 -1
- package/dest/services/tx_provider.js +9 -8
- package/dest/services/tx_provider_instrumentation.d.ts +1 -1
- package/dest/services/tx_provider_instrumentation.d.ts.map +1 -1
- package/dest/services/tx_provider_instrumentation.js +7 -20
- package/dest/test-helpers/index.d.ts +3 -1
- package/dest/test-helpers/index.d.ts.map +1 -1
- package/dest/test-helpers/index.js +2 -0
- package/dest/test-helpers/make-test-p2p-clients.d.ts +7 -8
- package/dest/test-helpers/make-test-p2p-clients.d.ts.map +1 -1
- package/dest/test-helpers/make-test-p2p-clients.js +1 -2
- package/dest/test-helpers/mock-pubsub.d.ts +30 -4
- package/dest/test-helpers/mock-pubsub.d.ts.map +1 -1
- package/dest/test-helpers/mock-pubsub.js +105 -4
- package/dest/test-helpers/reqresp-nodes.d.ts +2 -3
- package/dest/test-helpers/reqresp-nodes.d.ts.map +1 -1
- package/dest/test-helpers/reqresp-nodes.js +4 -3
- package/dest/test-helpers/test_tx_provider.d.ts +40 -0
- package/dest/test-helpers/test_tx_provider.d.ts.map +1 -0
- package/dest/test-helpers/test_tx_provider.js +41 -0
- package/dest/test-helpers/testbench-utils.d.ts +163 -0
- package/dest/test-helpers/testbench-utils.d.ts.map +1 -0
- package/dest/test-helpers/testbench-utils.js +366 -0
- package/dest/testbench/p2p_client_testbench_worker.d.ts +28 -2
- package/dest/testbench/p2p_client_testbench_worker.d.ts.map +1 -1
- package/dest/testbench/p2p_client_testbench_worker.js +225 -131
- package/dest/testbench/worker_client_manager.d.ts +51 -6
- package/dest/testbench/worker_client_manager.d.ts.map +1 -1
- package/dest/testbench/worker_client_manager.js +226 -39
- package/dest/util.d.ts +2 -2
- package/dest/util.d.ts.map +1 -1
- package/package.json +16 -16
- package/src/bootstrap/bootstrap.ts +7 -4
- package/src/client/factory.ts +83 -36
- package/src/client/interface.ts +73 -36
- package/src/client/p2p_client.ts +259 -364
- package/src/client/test/tx_proposal_collector/README.md +227 -0
- package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker.ts +345 -0
- package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.ts +43 -0
- package/src/config.ts +49 -13
- package/src/errors/tx-pool.error.ts +12 -0
- package/src/index.ts +1 -0
- package/src/mem_pools/attestation_pool/attestation_pool.ts +510 -78
- package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +612 -320
- package/src/mem_pools/attestation_pool/index.ts +9 -2
- package/src/mem_pools/attestation_pool/mocks.ts +20 -13
- package/src/mem_pools/index.ts +4 -1
- package/src/mem_pools/instrumentation.ts +10 -18
- package/src/mem_pools/interface.ts +4 -4
- package/src/mem_pools/tx_pool/README.md +29 -14
- package/src/mem_pools/tx_pool/aztec_kv_tx_pool.ts +130 -75
- package/src/mem_pools/tx_pool/eviction/eviction_manager.ts +66 -5
- package/src/mem_pools/tx_pool/eviction/eviction_strategy.ts +119 -4
- package/src/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.ts +162 -0
- package/src/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.ts +3 -3
- package/src/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.ts +4 -2
- package/src/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.ts +75 -0
- package/src/mem_pools/tx_pool_v2/README.md +275 -0
- package/src/mem_pools/tx_pool_v2/archive/index.ts +1 -0
- package/src/mem_pools/tx_pool_v2/archive/tx_archive.ts +120 -0
- package/src/mem_pools/tx_pool_v2/deleted_pool.ts +321 -0
- package/src/mem_pools/tx_pool_v2/eviction/eviction_manager.ts +160 -0
- package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.ts +121 -0
- package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.ts +125 -0
- package/src/mem_pools/tx_pool_v2/eviction/index.ts +27 -0
- package/src/mem_pools/tx_pool_v2/eviction/interfaces.ts +209 -0
- package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.ts +74 -0
- package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.ts +101 -0
- package/src/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.ts +91 -0
- package/src/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.ts +90 -0
- package/src/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.ts +31 -0
- package/src/mem_pools/tx_pool_v2/index.ts +12 -0
- package/src/mem_pools/tx_pool_v2/instrumentation.ts +69 -0
- package/src/mem_pools/tx_pool_v2/interfaces.ts +242 -0
- package/src/mem_pools/tx_pool_v2/tx_metadata.ts +297 -0
- package/src/mem_pools/tx_pool_v2/tx_pool_bench_metrics.ts +77 -0
- package/src/mem_pools/tx_pool_v2/tx_pool_indices.ts +444 -0
- package/src/mem_pools/tx_pool_v2/tx_pool_v2.ts +223 -0
- package/src/mem_pools/tx_pool_v2/tx_pool_v2_impl.ts +1083 -0
- package/src/msg_validators/attestation_validator/attestation_validator.ts +36 -21
- package/src/msg_validators/attestation_validator/fisherman_attestation_validator.ts +21 -18
- package/src/msg_validators/clock_tolerance.ts +51 -0
- package/src/msg_validators/index.ts +1 -1
- package/src/msg_validators/proposal_validator/block_proposal_validator.ts +10 -0
- package/src/msg_validators/proposal_validator/checkpoint_proposal_validator.ts +13 -0
- package/src/msg_validators/proposal_validator/index.ts +3 -0
- package/src/msg_validators/proposal_validator/proposal_validator.ts +92 -0
- package/src/msg_validators/proposal_validator/proposal_validator_test_suite.ts +230 -0
- package/src/msg_validators/tx_validator/README.md +115 -0
- package/src/msg_validators/tx_validator/aggregate_tx_validator.ts +5 -5
- package/src/msg_validators/tx_validator/archive_cache.ts +3 -3
- package/src/msg_validators/tx_validator/block_header_validator.ts +21 -8
- package/src/msg_validators/tx_validator/data_validator.ts +18 -6
- package/src/msg_validators/tx_validator/double_spend_validator.ts +15 -9
- package/src/msg_validators/tx_validator/factory.ts +372 -55
- package/src/msg_validators/tx_validator/fee_payer_balance.ts +40 -0
- package/src/msg_validators/tx_validator/gas_validator.ts +106 -54
- package/src/msg_validators/tx_validator/index.ts +2 -0
- package/src/msg_validators/tx_validator/metadata_validator.ts +18 -7
- package/src/msg_validators/tx_validator/nullifier_cache.ts +30 -0
- package/src/msg_validators/tx_validator/phases_validator.ts +5 -3
- package/src/msg_validators/tx_validator/size_validator.ts +22 -0
- package/src/msg_validators/tx_validator/timestamp_validator.ts +29 -19
- package/src/msg_validators/tx_validator/tx_permitted_validator.ts +8 -3
- package/src/msg_validators/tx_validator/tx_proof_validator.ts +8 -3
- package/src/services/data_store.ts +10 -7
- package/src/services/discv5/discV5_service.ts +1 -1
- package/src/services/dummy_service.ts +65 -2
- package/src/services/encoding.ts +13 -12
- package/src/services/gossipsub/README.md +641 -0
- package/src/services/gossipsub/index.ts +2 -0
- package/src/services/gossipsub/scoring.ts +29 -5
- package/src/services/gossipsub/topic_score_params.ts +487 -0
- package/src/services/index.ts +1 -0
- package/src/services/libp2p/instrumentation.ts +32 -73
- package/src/services/libp2p/libp2p_service.ts +706 -371
- package/src/services/peer-manager/metrics.ts +22 -26
- package/src/services/peer-manager/peer_manager.ts +1 -2
- package/src/services/peer-manager/peer_scoring.ts +28 -4
- package/src/services/reqresp/batch-tx-requester/README.md +305 -0
- package/src/services/reqresp/batch-tx-requester/batch_tx_requester.ts +678 -0
- package/src/services/reqresp/batch-tx-requester/config.ts +40 -0
- package/src/services/reqresp/batch-tx-requester/interface.ts +53 -0
- package/src/services/reqresp/batch-tx-requester/missing_txs.ts +161 -0
- package/src/services/reqresp/batch-tx-requester/peer_collection.ts +244 -0
- package/src/services/reqresp/batch-tx-requester/tx_validator.ts +37 -0
- package/src/services/reqresp/connection-sampler/batch_connection_sampler.ts +65 -4
- package/src/services/reqresp/connection-sampler/connection_sampler.ts +19 -1
- package/src/services/reqresp/constants.ts +14 -0
- package/src/services/reqresp/interface.ts +29 -1
- package/src/services/reqresp/metrics.ts +36 -27
- package/src/services/reqresp/protocols/block_txs/bitvector.ts +16 -0
- package/src/services/reqresp/protocols/block_txs/block_txs_handler.ts +35 -12
- package/src/services/reqresp/protocols/block_txs/block_txs_reqresp.ts +74 -9
- package/src/services/reqresp/protocols/status.ts +7 -4
- package/src/services/reqresp/protocols/tx.ts +22 -0
- package/src/services/reqresp/reqresp.ts +82 -23
- package/src/services/service.ts +73 -5
- package/src/services/tx_collection/config.ts +83 -1
- package/src/services/tx_collection/fast_tx_collection.ts +93 -47
- package/src/services/tx_collection/file_store_tx_collection.ts +202 -0
- package/src/services/tx_collection/file_store_tx_source.ts +117 -0
- package/src/services/tx_collection/index.ts +6 -0
- package/src/services/tx_collection/instrumentation.ts +11 -13
- package/src/services/tx_collection/missing_txs_tracker.ts +52 -0
- package/src/services/tx_collection/proposal_tx_collector.ts +113 -0
- package/src/services/tx_collection/slow_tx_collection.ts +68 -35
- package/src/services/tx_collection/tx_collection.ts +121 -24
- package/src/services/tx_collection/tx_collection_sink.ts +30 -34
- package/src/services/tx_collection/tx_source.ts +22 -3
- package/src/services/tx_file_store/config.ts +37 -0
- package/src/services/tx_file_store/index.ts +3 -0
- package/src/services/tx_file_store/instrumentation.ts +36 -0
- package/src/services/tx_file_store/tx_file_store.ts +175 -0
- package/src/services/tx_provider.ts +12 -11
- package/src/services/tx_provider_instrumentation.ts +13 -20
- package/src/test-helpers/index.ts +2 -0
- package/src/test-helpers/make-test-p2p-clients.ts +3 -5
- package/src/test-helpers/mock-pubsub.ts +146 -9
- package/src/test-helpers/reqresp-nodes.ts +4 -6
- package/src/test-helpers/test_tx_provider.ts +64 -0
- package/src/test-helpers/testbench-utils.ts +430 -0
- package/src/testbench/p2p_client_testbench_worker.ts +349 -128
- package/src/testbench/worker_client_manager.ts +304 -42
- package/src/util.ts +7 -1
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts +0 -37
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts.map +0 -1
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.js +0 -213
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts +0 -30
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts.map +0 -1
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.js +0 -219
- package/dest/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.d.ts +0 -15
- package/dest/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.js +0 -88
- package/dest/msg_validators/block_proposal_validator/block_proposal_validator.d.ts +0 -12
- package/dest/msg_validators/block_proposal_validator/block_proposal_validator.d.ts.map +0 -1
- package/dest/msg_validators/block_proposal_validator/block_proposal_validator.js +0 -82
- package/dest/msg_validators/block_proposal_validator/index.d.ts +0 -2
- package/dest/msg_validators/block_proposal_validator/index.d.ts.map +0 -1
- package/dest/msg_validators/block_proposal_validator/index.js +0 -1
- package/src/mem_pools/attestation_pool/kv_attestation_pool.ts +0 -298
- package/src/mem_pools/attestation_pool/memory_attestation_pool.ts +0 -287
- package/src/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.ts +0 -108
- package/src/msg_validators/block_proposal_validator/block_proposal_validator.ts +0 -97
- 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
|
-
|
|
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 {
|
|
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
|
|
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
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
} from '../../msg_validators/tx_validator/
|
|
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/
|
|
110
|
+
} from '../reqresp/index.js';
|
|
103
111
|
import { ReqResp } from '../reqresp/reqresp.js';
|
|
104
|
-
import type {
|
|
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?:
|
|
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
|
|
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
|
-
|
|
188
|
+
protected peerManager: PeerManagerInterface,
|
|
156
189
|
protected mempools: MemPools,
|
|
157
|
-
|
|
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.
|
|
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.
|
|
182
|
-
TopicType.
|
|
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<
|
|
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
|
|
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
|
|
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
|
-
//
|
|
427
|
-
|
|
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(
|
|
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
|
|
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
|
-
|
|
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
|
-
|
|
715
|
-
|
|
716
|
-
|
|
717
|
-
|
|
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
|
-
|
|
721
|
-
|
|
722
|
-
|
|
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 (
|
|
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
|
-
|
|
764
|
-
const
|
|
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
|
-
|
|
768
|
-
|
|
949
|
+
wasAccepted,
|
|
950
|
+
wasIgnored,
|
|
769
951
|
[Attributes.P2P_ID]: source.toString(),
|
|
770
952
|
});
|
|
771
953
|
|
|
772
|
-
if (
|
|
773
|
-
return { result: TopicValidatorResult.
|
|
774
|
-
} else if (
|
|
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.
|
|
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
|
|
804
|
-
*
|
|
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
|
|
809
|
-
|
|
810
|
-
|
|
811
|
-
|
|
812
|
-
|
|
813
|
-
|
|
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.
|
|
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.
|
|
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
|
-
|
|
872
|
-
|
|
873
|
-
|
|
874
|
-
|
|
875
|
-
|
|
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
|
-
|
|
878
|
-
|
|
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
|
-
|
|
881
|
-
|
|
882
|
-
|
|
883
|
-
|
|
884
|
-
|
|
885
|
-
|
|
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
|
-
|
|
889
|
-
|
|
890
|
-
|
|
891
|
-
|
|
892
|
-
|
|
893
|
-
this.
|
|
894
|
-
|
|
895
|
-
|
|
896
|
-
|
|
897
|
-
|
|
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
|
-
|
|
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
|
-
|
|
906
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
//
|
|
938
|
-
|
|
939
|
-
|
|
940
|
-
|
|
941
|
-
|
|
942
|
-
|
|
943
|
-
|
|
944
|
-
|
|
945
|
-
|
|
946
|
-
|
|
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
|
-
//
|
|
956
|
-
|
|
957
|
-
|
|
958
|
-
|
|
959
|
-
|
|
960
|
-
|
|
961
|
-
|
|
962
|
-
|
|
963
|
-
|
|
964
|
-
|
|
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
|
-
*
|
|
971
|
-
*
|
|
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.
|
|
974
|
-
[Attributes.SLOT_NUMBER]:
|
|
975
|
-
[Attributes.BLOCK_ARCHIVE]:
|
|
976
|
-
[Attributes.P2P_ID]: await
|
|
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
|
-
|
|
979
|
-
|
|
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.
|
|
1378
|
+
[Attributes.BLOCK_ARCHIVE]: request.archiveRoot.toString(),
|
|
1003
1379
|
}))
|
|
1004
|
-
|
|
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.
|
|
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
|
|
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.
|
|
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
|
|
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
|
|
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
|
-
|
|
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
|
-
|
|
1168
|
-
|
|
1169
|
-
|
|
1170
|
-
|
|
1171
|
-
|
|
1172
|
-
|
|
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
|
-
|
|
1201
|
-
|
|
1202
|
-
|
|
1203
|
-
|
|
1204
|
-
|
|
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
|
-
|
|
1243
|
-
|
|
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
|
-
|
|
1246
|
-
const
|
|
1247
|
-
|
|
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
|
-
|
|
1252
|
-
|
|
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
|
-
|
|
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,
|
|
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
|
-
|
|
1279
|
-
|
|
1280
|
-
return createTxMessageValidators(
|
|
1623
|
+
return createFirstStageTxValidationsForGossipedTransactions(
|
|
1281
1624
|
nextSlotTimestamp,
|
|
1282
|
-
|
|
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,
|
|
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
|
-
|
|
1347
|
-
|
|
1348
|
-
|
|
1349
|
-
|
|
1350
|
-
|
|
1351
|
-
|
|
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
|
|
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.
|
|
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
|
|
1375
|
-
|
|
1376
|
-
|
|
1377
|
-
|
|
1378
|
-
|
|
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
|
-
|
|
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
|
|
1736
|
+
return result;
|
|
1402
1737
|
}
|
|
1403
1738
|
|
|
1404
1739
|
public getPeerScore(peerId: PeerId): number {
|