@aztec/p2p 0.0.1-commit.96bb3f7 → 0.0.1-commit.993d240
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/README.md +129 -3
- package/dest/bootstrap/bootstrap.d.ts +4 -3
- package/dest/bootstrap/bootstrap.d.ts.map +1 -1
- package/dest/bootstrap/bootstrap.js +13 -5
- package/dest/client/factory.d.ts +13 -12
- package/dest/client/factory.d.ts.map +1 -1
- package/dest/client/factory.js +65 -20
- package/dest/client/interface.d.ts +58 -36
- package/dest/client/interface.d.ts.map +1 -1
- package/dest/client/p2p_client.d.ts +52 -58
- package/dest/client/p2p_client.d.ts.map +1 -1
- package/dest/client/p2p_client.js +236 -236
- package/dest/config.d.ts +163 -84
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +156 -43
- package/dest/errors/p2p-service.error.d.ts +9 -0
- package/dest/errors/p2p-service.error.d.ts.map +1 -0
- package/dest/errors/p2p-service.error.js +10 -0
- package/dest/errors/reqresp.error.d.ts +1 -20
- package/dest/errors/reqresp.error.d.ts.map +1 -1
- package/dest/errors/reqresp.error.js +0 -21
- 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 -2
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +1 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +162 -106
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.js +511 -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 +525 -132
- 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 +4 -2
- package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/mocks.js +13 -8
- package/dest/mem_pools/index.d.ts +3 -3
- 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 +4 -2
- package/dest/mem_pools/instrumentation.d.ts.map +1 -1
- package/dest/mem_pools/instrumentation.js +35 -17
- 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_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 +94 -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 +11 -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 +12 -0
- package/dest/mem_pools/tx_pool_v2/eviction/insufficient_fee_per_gas_eviction_rule.d.ts +16 -0
- package/dest/mem_pools/tx_pool_v2/eviction/insufficient_fee_per_gas_eviction_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/eviction/insufficient_fee_per_gas_eviction_rule.js +62 -0
- package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts +180 -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 → tx_pool_v2}/eviction/invalid_txs_after_mining_rule.js +16 -35
- 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 +75 -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 +218 -0
- package/dest/mem_pools/tx_pool_v2/interfaces.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/interfaces.js +10 -0
- package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts +137 -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 +223 -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 +337 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts +62 -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 +167 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts +78 -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 +914 -0
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts +10 -4
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts.map +1 -1
- package/dest/msg_validators/attestation_validator/attestation_validator.js +69 -13
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts +9 -5
- 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 -11
- package/dest/msg_validators/clock_tolerance.d.ts +32 -0
- package/dest/msg_validators/clock_tolerance.d.ts.map +1 -0
- package/dest/msg_validators/clock_tolerance.js +95 -0
- package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts +10 -4
- package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts.map +1 -1
- package/dest/msg_validators/proposal_validator/block_proposal_validator.js +10 -2
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts +10 -4
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts.map +1 -1
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.js +16 -2
- package/dest/msg_validators/proposal_validator/proposal_validator.d.ts +21 -8
- package/dest/msg_validators/proposal_validator/proposal_validator.d.ts.map +1 -1
- package/dest/msg_validators/proposal_validator/proposal_validator.js +123 -53
- 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 +11 -18
- package/dest/msg_validators/tx_validator/allowed_public_setup.d.ts +2 -1
- package/dest/msg_validators/tx_validator/allowed_public_setup.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/allowed_public_setup.js +25 -21
- package/dest/msg_validators/tx_validator/allowed_setup_helpers.d.ts +17 -0
- package/dest/msg_validators/tx_validator/allowed_setup_helpers.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/allowed_setup_helpers.js +24 -0
- 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/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/cached_tx_validator.d.ts +15 -0
- package/dest/msg_validators/tx_validator/cached_tx_validator.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/cached_tx_validator.js +19 -0
- package/dest/msg_validators/tx_validator/contract_instance_validator.d.ts +9 -0
- package/dest/msg_validators/tx_validator/contract_instance_validator.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/contract_instance_validator.js +48 -0
- package/dest/msg_validators/tx_validator/data_validator.d.ts +4 -1
- package/dest/msg_validators/tx_validator/data_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/data_validator.js +40 -3
- 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 +138 -5
- package/dest/msg_validators/tx_validator/factory.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/factory.js +259 -58
- 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 +24 -0
- package/dest/msg_validators/tx_validator/gas_validator.d.ts +100 -3
- package/dest/msg_validators/tx_validator/gas_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/gas_validator.js +146 -67
- package/dest/msg_validators/tx_validator/index.d.ts +6 -1
- package/dest/msg_validators/tx_validator/index.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/index.js +5 -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 +6 -6
- 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 +24 -3
- package/dest/msg_validators/tx_validator/phases_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/phases_validator.js +75 -27
- 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 +4 -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 +4 -2
- package/dest/msg_validators/tx_validator/tx_validation_cache.d.ts +48 -0
- package/dest/msg_validators/tx_validator/tx_validation_cache.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/tx_validation_cache.js +69 -0
- 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 +14 -10
- package/dest/services/discv5/discV5_service.d.ts +2 -1
- package/dest/services/discv5/discV5_service.d.ts.map +1 -1
- package/dest/services/discv5/discV5_service.js +36 -9
- package/dest/services/dummy_service.d.ts +33 -17
- package/dest/services/dummy_service.d.ts.map +1 -1
- package/dest/services/dummy_service.js +56 -15
- package/dest/services/encoding.d.ts +7 -3
- package/dest/services/encoding.d.ts.map +1 -1
- package/dest/services/encoding.js +20 -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 +184 -0
- package/dest/services/gossipsub/topic_score_params.d.ts.map +1 -0
- package/dest/services/gossipsub/topic_score_params.js +363 -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 +3 -1
- package/dest/services/libp2p/instrumentation.d.ts.map +1 -1
- package/dest/services/libp2p/instrumentation.js +33 -8
- package/dest/services/libp2p/libp2p_service.d.ts +107 -59
- package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
- package/dest/services/libp2p/libp2p_service.js +683 -539
- package/dest/services/peer-manager/metrics.d.ts +4 -2
- package/dest/services/peer-manager/metrics.d.ts.map +1 -1
- package/dest/services/peer-manager/metrics.js +26 -5
- package/dest/services/peer-manager/peer_manager.d.ts +6 -2
- package/dest/services/peer-manager/peer_manager.d.ts.map +1 -1
- package/dest/services/peer-manager/peer_manager.js +40 -11
- package/dest/services/peer-manager/peer_scoring.d.ts +7 -2
- package/dest/services/peer-manager/peer_scoring.d.ts.map +1 -1
- package/dest/services/peer-manager/peer_scoring.js +65 -14
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts +51 -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 +552 -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 +50 -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 +35 -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 +136 -0
- package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts +62 -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 +176 -0
- package/dest/services/reqresp/batch-tx-requester/tx_validator.d.ts +11 -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 +7 -0
- package/dest/services/reqresp/config.d.ts +3 -3
- package/dest/services/reqresp/config.d.ts.map +1 -1
- 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 +27 -18
- package/dest/services/reqresp/interface.d.ts.map +1 -1
- package/dest/services/reqresp/interface.js +23 -19
- 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 +16 -5
- 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 +29 -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/index.d.ts +1 -2
- package/dest/services/reqresp/protocols/index.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/index.js +0 -1
- 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 +2 -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 +21 -3
- package/dest/services/reqresp/rate-limiter/rate_limiter.d.ts +5 -4
- package/dest/services/reqresp/rate-limiter/rate_limiter.d.ts.map +1 -1
- package/dest/services/reqresp/rate-limiter/rate_limiter.js +10 -8
- package/dest/services/reqresp/rate-limiter/rate_limits.d.ts +1 -1
- package/dest/services/reqresp/rate-limiter/rate_limits.d.ts.map +1 -1
- package/dest/services/reqresp/rate-limiter/rate_limits.js +0 -10
- package/dest/services/reqresp/reqresp.d.ts +12 -29
- package/dest/services/reqresp/reqresp.d.ts.map +1 -1
- package/dest/services/reqresp/reqresp.js +58 -187
- package/dest/services/service.d.ts +49 -13
- package/dest/services/service.d.ts.map +1 -1
- package/dest/services/tx_collection/config.d.ts +11 -11
- package/dest/services/tx_collection/config.d.ts.map +1 -1
- package/dest/services/tx_collection/config.js +27 -26
- package/dest/services/tx_collection/file_store_tx_collection.d.ts +37 -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 +127 -0
- package/dest/services/tx_collection/file_store_tx_source.d.ts +38 -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 +100 -0
- package/dest/services/tx_collection/index.d.ts +3 -2
- package/dest/services/tx_collection/index.d.ts.map +1 -1
- package/dest/services/tx_collection/index.js +1 -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 +8 -2
- package/dest/services/tx_collection/request_tracker.d.ts +53 -0
- package/dest/services/tx_collection/request_tracker.d.ts.map +1 -0
- package/dest/services/tx_collection/request_tracker.js +84 -0
- package/dest/services/tx_collection/tx_collection.d.ts +50 -56
- package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_collection.js +298 -70
- package/dest/services/tx_collection/tx_collection_sink.d.ts +19 -9
- package/dest/services/tx_collection/tx_collection_sink.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_collection_sink.js +28 -31
- package/dest/services/tx_collection/tx_source.d.ts +13 -7
- package/dest/services/tx_collection/tx_source.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_source.js +26 -7
- 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 +46 -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 +142 -0
- package/dest/services/tx_provider.d.ts +8 -6
- package/dest/services/tx_provider.d.ts.map +1 -1
- package/dest/services/tx_provider.js +12 -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 +5 -5
- 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 +5 -3
- package/dest/test-helpers/mock-pubsub.d.ts +46 -6
- package/dest/test-helpers/mock-pubsub.d.ts.map +1 -1
- package/dest/test-helpers/mock-pubsub.js +115 -14
- package/dest/test-helpers/reqresp-nodes.d.ts +5 -7
- package/dest/test-helpers/reqresp-nodes.d.ts.map +1 -1
- package/dest/test-helpers/reqresp-nodes.js +19 -20
- package/dest/test-helpers/test_tx_provider.d.ts +42 -0
- package/dest/test-helpers/test_tx_provider.d.ts.map +1 -0
- package/dest/test-helpers/test_tx_provider.js +44 -0
- package/dest/test-helpers/testbench-utils.d.ts +161 -0
- package/dest/test-helpers/testbench-utils.d.ts.map +1 -0
- package/dest/test-helpers/testbench-utils.js +393 -0
- package/dest/testbench/p2p_client_testbench_worker.d.ts +26 -2
- package/dest/testbench/p2p_client_testbench_worker.d.ts.map +1 -1
- package/dest/testbench/p2p_client_testbench_worker.js +270 -144
- package/dest/testbench/worker_client_manager.d.ts +57 -6
- package/dest/testbench/worker_client_manager.d.ts.map +1 -1
- package/dest/testbench/worker_client_manager.js +277 -49
- package/dest/util.d.ts +13 -8
- package/dest/util.d.ts.map +1 -1
- package/dest/util.js +35 -14
- package/dest/versioning.d.ts +3 -6
- package/dest/versioning.d.ts.map +1 -1
- package/dest/versioning.js +3 -24
- package/package.json +17 -16
- package/src/bootstrap/bootstrap.ts +16 -5
- package/src/client/factory.ts +135 -39
- package/src/client/interface.ts +70 -44
- package/src/client/p2p_client.ts +283 -302
- package/src/client/test/p2p_client.batch_tx_requester.bench.README.md +197 -0
- package/src/config.ts +274 -48
- package/src/errors/p2p-service.error.ts +11 -0
- package/src/errors/reqresp.error.ts +0 -25
- package/src/errors/tx-pool.error.ts +12 -0
- package/src/index.ts +1 -1
- package/src/mem_pools/attestation_pool/attestation_pool.ts +575 -94
- package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +633 -149
- package/src/mem_pools/attestation_pool/index.ts +9 -2
- package/src/mem_pools/attestation_pool/mocks.ts +19 -11
- package/src/mem_pools/index.ts +2 -2
- package/src/mem_pools/instrumentation.ts +24 -15
- package/src/mem_pools/interface.ts +4 -4
- package/src/mem_pools/tx_pool_v2/README.md +283 -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 +122 -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 +28 -0
- package/src/mem_pools/tx_pool_v2/eviction/insufficient_fee_per_gas_eviction_rule.ts +65 -0
- package/src/mem_pools/tx_pool_v2/eviction/interfaces.ts +219 -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 +99 -0
- package/src/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.ts +32 -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 +250 -0
- package/src/mem_pools/tx_pool_v2/tx_metadata.ts +349 -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 +430 -0
- package/src/mem_pools/tx_pool_v2/tx_pool_v2.ts +238 -0
- package/src/mem_pools/tx_pool_v2/tx_pool_v2_impl.ts +1090 -0
- package/src/msg_validators/attestation_validator/README.md +49 -0
- package/src/msg_validators/attestation_validator/attestation_validator.ts +60 -16
- package/src/msg_validators/attestation_validator/fisherman_attestation_validator.ts +29 -16
- package/src/msg_validators/clock_tolerance.ts +127 -0
- package/src/msg_validators/proposal_validator/README.md +123 -0
- package/src/msg_validators/proposal_validator/block_proposal_validator.ts +24 -4
- package/src/msg_validators/proposal_validator/checkpoint_proposal_validator.ts +35 -7
- package/src/msg_validators/proposal_validator/proposal_validator.ts +129 -62
- package/src/msg_validators/tx_validator/README.md +127 -0
- package/src/msg_validators/tx_validator/aggregate_tx_validator.ts +8 -17
- package/src/msg_validators/tx_validator/allowed_public_setup.ts +22 -27
- package/src/msg_validators/tx_validator/allowed_setup_helpers.ts +31 -0
- package/src/msg_validators/tx_validator/archive_cache.ts +2 -2
- package/src/msg_validators/tx_validator/block_header_validator.ts +21 -8
- package/src/msg_validators/tx_validator/cached_tx_validator.ts +31 -0
- package/src/msg_validators/tx_validator/contract_instance_validator.ts +56 -0
- package/src/msg_validators/tx_validator/data_validator.ts +50 -3
- package/src/msg_validators/tx_validator/double_spend_validator.ts +15 -9
- package/src/msg_validators/tx_validator/factory.ts +424 -56
- package/src/msg_validators/tx_validator/fee_payer_balance.ts +44 -0
- package/src/msg_validators/tx_validator/gas_validator.ts +211 -77
- package/src/msg_validators/tx_validator/index.ts +5 -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 +87 -30
- package/src/msg_validators/tx_validator/size_validator.ts +22 -0
- package/src/msg_validators/tx_validator/timestamp_validator.ts +29 -21
- package/src/msg_validators/tx_validator/tx_permitted_validator.ts +8 -3
- package/src/msg_validators/tx_validator/tx_proof_validator.ts +10 -3
- package/src/msg_validators/tx_validator/tx_validation_cache.ts +102 -0
- package/src/services/data_store.ts +14 -19
- package/src/services/discv5/discV5_service.ts +39 -6
- package/src/services/dummy_service.ts +71 -39
- package/src/services/encoding.ts +20 -13
- 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 +519 -0
- package/src/services/index.ts +1 -0
- package/src/services/libp2p/instrumentation.ts +34 -7
- package/src/services/libp2p/libp2p_service.ts +753 -597
- package/src/services/peer-manager/metrics.ts +28 -4
- package/src/services/peer-manager/peer_manager.ts +46 -11
- package/src/services/peer-manager/peer_scoring.ts +56 -6
- package/src/services/reqresp/README.md +215 -0
- package/src/services/reqresp/batch-tx-requester/README.md +344 -0
- package/src/services/reqresp/batch-tx-requester/batch_tx_requester.ts +684 -0
- package/src/services/reqresp/batch-tx-requester/config.ts +40 -0
- package/src/services/reqresp/batch-tx-requester/interface.ts +61 -0
- package/src/services/reqresp/batch-tx-requester/missing_txs.ts +168 -0
- package/src/services/reqresp/batch-tx-requester/peer_collection.ts +249 -0
- package/src/services/reqresp/batch-tx-requester/tx_validator.ts +24 -0
- package/src/services/reqresp/config.ts +2 -2
- package/src/services/reqresp/connection-sampler/connection_sampler.ts +16 -0
- package/src/services/reqresp/constants.ts +14 -0
- package/src/services/reqresp/interface.ts +48 -46
- package/src/services/reqresp/metrics.ts +33 -9
- package/src/services/reqresp/protocols/block_txs/bitvector.ts +16 -0
- package/src/services/reqresp/protocols/block_txs/block_txs_handler.ts +37 -12
- package/src/services/reqresp/protocols/block_txs/block_txs_reqresp.ts +74 -9
- package/src/services/reqresp/protocols/index.ts +0 -1
- package/src/services/reqresp/protocols/status.ts +5 -3
- package/src/services/reqresp/protocols/tx.ts +23 -3
- package/src/services/reqresp/rate-limiter/rate_limiter.ts +13 -9
- package/src/services/reqresp/rate-limiter/rate_limits.ts +0 -10
- package/src/services/reqresp/reqresp.ts +68 -224
- package/src/services/service.ts +66 -29
- package/src/services/tx_collection/config.ts +41 -36
- package/src/services/tx_collection/file_store_tx_collection.ts +153 -0
- package/src/services/tx_collection/file_store_tx_source.ts +129 -0
- package/src/services/tx_collection/index.ts +2 -1
- package/src/services/tx_collection/instrumentation.ts +11 -2
- package/src/services/tx_collection/request_tracker.ts +127 -0
- package/src/services/tx_collection/tx_collection.ts +367 -115
- package/src/services/tx_collection/tx_collection_sink.ts +32 -36
- package/src/services/tx_collection/tx_source.ts +28 -8
- 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 +163 -0
- package/src/services/tx_provider.ts +17 -11
- package/src/services/tx_provider_instrumentation.ts +11 -5
- package/src/test-helpers/index.ts +2 -0
- package/src/test-helpers/make-test-p2p-clients.ts +7 -6
- package/src/test-helpers/mock-pubsub.ts +137 -14
- package/src/test-helpers/reqresp-nodes.ts +17 -29
- package/src/test-helpers/test_tx_provider.ts +69 -0
- package/src/test-helpers/testbench-utils.ts +455 -0
- package/src/testbench/p2p_client_testbench_worker.ts +370 -138
- package/src/testbench/worker_client_manager.ts +355 -51
- package/src/util.ts +40 -19
- package/src/versioning.ts +3 -33
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts +0 -40
- 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 -218
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts +0 -31
- 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 -180
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts +0 -120
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.js +0 -555
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts +0 -18
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.js +0 -56
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts +0 -83
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.js +0 -5
- 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/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts +0 -17
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts +0 -19
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.js +0 -76
- package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts +0 -26
- package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.js +0 -84
- package/dest/mem_pools/tx_pool/index.d.ts +0 -3
- package/dest/mem_pools/tx_pool/index.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/index.js +0 -2
- package/dest/mem_pools/tx_pool/priority.d.ts +0 -12
- package/dest/mem_pools/tx_pool/priority.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/priority.js +0 -15
- package/dest/mem_pools/tx_pool/tx_pool.d.ts +0 -127
- package/dest/mem_pools/tx_pool/tx_pool.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/tx_pool.js +0 -3
- package/dest/mem_pools/tx_pool/tx_pool_test_suite.d.ts +0 -7
- package/dest/mem_pools/tx_pool/tx_pool_test_suite.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/tx_pool_test_suite.js +0 -400
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts +0 -23
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts.map +0 -1
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.js +0 -183
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts +0 -45
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts.map +0 -1
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.js +0 -92
- package/dest/services/reqresp/protocols/block.d.ts +0 -9
- package/dest/services/reqresp/protocols/block.d.ts.map +0 -1
- package/dest/services/reqresp/protocols/block.js +0 -32
- package/dest/services/tx_collection/fast_tx_collection.d.ts +0 -51
- package/dest/services/tx_collection/fast_tx_collection.d.ts.map +0 -1
- package/dest/services/tx_collection/fast_tx_collection.js +0 -300
- package/dest/services/tx_collection/slow_tx_collection.d.ts +0 -53
- package/dest/services/tx_collection/slow_tx_collection.d.ts.map +0 -1
- package/dest/services/tx_collection/slow_tx_collection.js +0 -177
- package/src/mem_pools/attestation_pool/kv_attestation_pool.ts +0 -320
- package/src/mem_pools/attestation_pool/memory_attestation_pool.ts +0 -264
- package/src/mem_pools/tx_pool/README.md +0 -255
- package/src/mem_pools/tx_pool/aztec_kv_tx_pool.ts +0 -691
- package/src/mem_pools/tx_pool/eviction/eviction_manager.ts +0 -71
- package/src/mem_pools/tx_pool/eviction/eviction_strategy.ts +0 -93
- package/src/mem_pools/tx_pool/eviction/insufficient_fee_payer_balance_rule.ts +0 -108
- package/src/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.ts +0 -104
- package/src/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.ts +0 -91
- package/src/mem_pools/tx_pool/eviction/low_priority_eviction_rule.ts +0 -106
- package/src/mem_pools/tx_pool/index.ts +0 -2
- package/src/mem_pools/tx_pool/priority.ts +0 -20
- package/src/mem_pools/tx_pool/tx_pool.ts +0 -141
- package/src/mem_pools/tx_pool/tx_pool_test_suite.ts +0 -319
- package/src/msg_validators/proposal_validator/proposal_validator_test_suite.ts +0 -206
- package/src/services/reqresp/connection-sampler/batch_connection_sampler.ts +0 -100
- package/src/services/reqresp/protocols/block.ts +0 -37
- package/src/services/tx_collection/fast_tx_collection.ts +0 -341
- package/src/services/tx_collection/slow_tx_collection.ts +0 -233
|
@@ -1,16 +1,14 @@
|
|
|
1
1
|
import type { EpochCacheInterface } from '@aztec/epoch-cache';
|
|
2
|
-
import { BlockNumber } from '@aztec/foundation/branded-types';
|
|
3
|
-
import {
|
|
4
|
-
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
2
|
+
import { BlockNumber, type SlotNumber } from '@aztec/foundation/branded-types';
|
|
3
|
+
import { maxBy, merge } from '@aztec/foundation/collection';
|
|
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
|
-
import type { EthAddress,
|
|
9
|
+
import type { EthAddress, L2BlockSource } from '@aztec/stdlib/block';
|
|
12
10
|
import type { ContractDataSource } from '@aztec/stdlib/contract';
|
|
13
|
-
import { GasFees } from '@aztec/stdlib/gas';
|
|
11
|
+
import { type BlockMinFeesProvider, GasFees } from '@aztec/stdlib/gas';
|
|
14
12
|
import type { ClientProtocolCircuitVerifier, PeerInfo, WorldStateSynchronizer } from '@aztec/stdlib/interfaces/server';
|
|
15
13
|
import {
|
|
16
14
|
BlockProposal,
|
|
@@ -18,16 +16,16 @@ import {
|
|
|
18
16
|
CheckpointProposal,
|
|
19
17
|
type CheckpointProposalCore,
|
|
20
18
|
type Gossipable,
|
|
21
|
-
P2PClientType,
|
|
22
19
|
P2PMessage,
|
|
23
20
|
PeerErrorSeverity,
|
|
21
|
+
PeerErrorSeverityByHarshness,
|
|
24
22
|
TopicType,
|
|
25
23
|
createTopicString,
|
|
26
|
-
|
|
24
|
+
getTopicsForConfig,
|
|
27
25
|
metricsTopicStrToLabels,
|
|
28
26
|
} from '@aztec/stdlib/p2p';
|
|
29
27
|
import { MerkleTreeId } from '@aztec/stdlib/trees';
|
|
30
|
-
import { Tx, type
|
|
28
|
+
import { Tx, type TxValidationResult } from '@aztec/stdlib/tx';
|
|
31
29
|
import type { UInt64 } from '@aztec/stdlib/types';
|
|
32
30
|
import { compressComponentVersions } from '@aztec/stdlib/versioning';
|
|
33
31
|
import {
|
|
@@ -45,75 +43,75 @@ import {
|
|
|
45
43
|
type GossipsubMessage,
|
|
46
44
|
gossipsub,
|
|
47
45
|
} from '@chainsafe/libp2p-gossipsub';
|
|
48
|
-
import { createPeerScoreParams
|
|
46
|
+
import { createPeerScoreParams } from '@chainsafe/libp2p-gossipsub/score';
|
|
49
47
|
import { SignaturePolicy } from '@chainsafe/libp2p-gossipsub/types';
|
|
50
48
|
import { noise } from '@chainsafe/libp2p-noise';
|
|
51
49
|
import { yamux } from '@chainsafe/libp2p-yamux';
|
|
52
50
|
import { bootstrap } from '@libp2p/bootstrap';
|
|
53
51
|
import { identify } from '@libp2p/identify';
|
|
54
52
|
import { type Message, type MultiaddrConnection, type PeerId, TopicValidatorResult } from '@libp2p/interface';
|
|
55
|
-
import type { ConnectionManager } from '@libp2p/interface-internal';
|
|
53
|
+
import type { AddressManager, ConnectionManager } from '@libp2p/interface-internal';
|
|
56
54
|
import { mplex } from '@libp2p/mplex';
|
|
57
55
|
import { tcp } from '@libp2p/tcp';
|
|
56
|
+
import { multiaddr } from '@multiformats/multiaddr';
|
|
58
57
|
import { ENR } from '@nethermindeth/enr';
|
|
59
58
|
import { createLibp2p } from 'libp2p';
|
|
60
59
|
|
|
61
60
|
import type { P2PConfig } from '../../config.js';
|
|
62
|
-
import {
|
|
61
|
+
import { CheckpointProposalReceivedCallbackNotRegisteredError } from '../../errors/p2p-service.error.js';
|
|
63
62
|
import type { MemPools } from '../../mem_pools/interface.js';
|
|
64
63
|
import {
|
|
65
64
|
BlockProposalValidator,
|
|
66
65
|
CheckpointAttestationValidator,
|
|
67
66
|
CheckpointProposalValidator,
|
|
67
|
+
DoubleSpendTxValidator,
|
|
68
68
|
FishermanAttestationValidator,
|
|
69
|
+
getDefaultAllowedSetupFunctions,
|
|
69
70
|
} from '../../msg_validators/index.js';
|
|
70
71
|
import { MessageSeenValidator } from '../../msg_validators/msg_seen_validator/msg_seen_validator.js';
|
|
71
|
-
import { getDefaultAllowedSetupFunctions } from '../../msg_validators/tx_validator/allowed_public_setup.js';
|
|
72
|
-
import { type MessageValidator, createTxMessageValidators } from '../../msg_validators/tx_validator/factory.js';
|
|
73
72
|
import {
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
} from '../../msg_validators/tx_validator/
|
|
73
|
+
type TransactionValidator,
|
|
74
|
+
createFirstStageTxValidationsForGossipedTransactions,
|
|
75
|
+
createSecondStageTxValidationsForGossipedTransactions,
|
|
76
|
+
createTxValidatorForBlockProposalReceivedTxs,
|
|
77
|
+
} from '../../msg_validators/tx_validator/factory.js';
|
|
78
|
+
import { TxValidationCache } from '../../msg_validators/tx_validator/tx_validation_cache.js';
|
|
80
79
|
import { GossipSubEvent } from '../../types/index.js';
|
|
81
80
|
import { type PubSubLibp2p, convertToMultiaddr } from '../../util.js';
|
|
82
81
|
import { getVersions } from '../../versioning.js';
|
|
83
82
|
import { AztecDatastore } from '../data_store.js';
|
|
84
83
|
import { DiscV5Service } from '../discv5/discV5_service.js';
|
|
85
84
|
import { SnappyTransform, fastMsgIdFn, getMsgIdFn, msgIdToStrFn } from '../encoding.js';
|
|
86
|
-
import { gossipScoreThresholds } from '../gossipsub/scoring.js';
|
|
85
|
+
import { APP_SPECIFIC_WEIGHT, gossipScoreThresholds } from '../gossipsub/scoring.js';
|
|
86
|
+
import { createAllTopicScoreParams } from '../gossipsub/topic_score_params.js';
|
|
87
87
|
import type { PeerManagerInterface } from '../peer-manager/interface.js';
|
|
88
88
|
import { PeerManager } from '../peer-manager/peer_manager.js';
|
|
89
89
|
import { PeerScoring } from '../peer-manager/peer_scoring.js';
|
|
90
|
+
import type { BatchTxRequesterLibP2PService } from '../reqresp/batch-tx-requester/interface.js';
|
|
90
91
|
import type { P2PReqRespConfig } from '../reqresp/config.js';
|
|
91
92
|
import {
|
|
92
|
-
|
|
93
|
+
AuthRequest,
|
|
94
|
+
BlockTxsRequest,
|
|
95
|
+
BlockTxsResponse,
|
|
93
96
|
type ReqRespInterface,
|
|
97
|
+
type ReqRespResponse,
|
|
94
98
|
ReqRespSubProtocol,
|
|
95
99
|
type ReqRespSubProtocolHandler,
|
|
96
100
|
type ReqRespSubProtocolHandlers,
|
|
97
|
-
type ReqRespSubProtocolValidators,
|
|
98
|
-
type SubProtocolMap,
|
|
99
|
-
ValidationError,
|
|
100
|
-
} from '../reqresp/interface.js';
|
|
101
|
-
import { reqRespBlockTxsHandler } from '../reqresp/protocols/block_txs/block_txs_handler.js';
|
|
102
|
-
import { reqGoodbyeHandler } from '../reqresp/protocols/goodbye.js';
|
|
103
|
-
import {
|
|
104
|
-
AuthRequest,
|
|
105
|
-
BlockTxsRequest,
|
|
106
|
-
BlockTxsResponse,
|
|
107
101
|
StatusMessage,
|
|
102
|
+
ValidationError,
|
|
108
103
|
pingHandler,
|
|
109
|
-
|
|
104
|
+
reqGoodbyeHandler,
|
|
105
|
+
reqRespBlockTxsHandler,
|
|
110
106
|
reqRespStatusHandler,
|
|
111
107
|
reqRespTxHandler,
|
|
112
|
-
} from '../reqresp/
|
|
108
|
+
} from '../reqresp/index.js';
|
|
113
109
|
import { ReqResp } from '../reqresp/reqresp.js';
|
|
114
110
|
import type {
|
|
115
111
|
P2PBlockReceivedCallback,
|
|
112
|
+
P2PCheckpointAttestationCallback,
|
|
116
113
|
P2PCheckpointReceivedCallback,
|
|
114
|
+
P2PDuplicateAttestationCallback,
|
|
117
115
|
P2PService,
|
|
118
116
|
PeerDiscoveryService,
|
|
119
117
|
} from '../service.js';
|
|
@@ -128,14 +126,14 @@ interface ValidationResult {
|
|
|
128
126
|
type ValidationOutcome = { allPassed: true } | { allPassed: false; failure: ValidationResult };
|
|
129
127
|
|
|
130
128
|
// REFACTOR: Unify with the type above
|
|
131
|
-
type ReceivedMessageValidationResult<T> =
|
|
132
|
-
| { obj: T; result: Exclude<TopicValidatorResult, TopicValidatorResult.Reject
|
|
133
|
-
| { obj?:
|
|
129
|
+
type ReceivedMessageValidationResult<T, M = undefined> =
|
|
130
|
+
| { obj: T; result: Exclude<TopicValidatorResult, TopicValidatorResult.Reject>; metadata?: M }
|
|
131
|
+
| { obj?: T; result: TopicValidatorResult.Reject; metadata?: M; severity: PeerErrorSeverity };
|
|
134
132
|
|
|
135
133
|
/**
|
|
136
134
|
* Lib P2P implementation of the P2PService interface.
|
|
137
135
|
*/
|
|
138
|
-
export class LibP2PService
|
|
136
|
+
export class LibP2PService extends WithTracer implements P2PService {
|
|
139
137
|
private discoveryRunningPromise?: RunningPromise;
|
|
140
138
|
private msgIdSeenValidators: Record<TopicType, MessageSeenValidator> = {} as Record<TopicType, MessageSeenValidator>;
|
|
141
139
|
|
|
@@ -147,7 +145,18 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
147
145
|
private protocolVersion = '';
|
|
148
146
|
private topicStrings: Record<TopicType, string> = {} as Record<TopicType, string>;
|
|
149
147
|
|
|
150
|
-
|
|
148
|
+
/** Callback invoked when a duplicate proposal is detected (triggers slashing). */
|
|
149
|
+
private duplicateProposalCallback?: (info: {
|
|
150
|
+
slot: SlotNumber;
|
|
151
|
+
proposer: EthAddress;
|
|
152
|
+
type: 'checkpoint' | 'block';
|
|
153
|
+
}) => void;
|
|
154
|
+
|
|
155
|
+
/** Callback invoked when a duplicate attestation is detected (triggers slashing). */
|
|
156
|
+
private duplicateAttestationCallback?: P2PDuplicateAttestationCallback;
|
|
157
|
+
|
|
158
|
+
/** Callback invoked when a valid checkpoint attestation is accepted into the pool. */
|
|
159
|
+
private checkpointAttestationCallback?: P2PCheckpointAttestationCallback;
|
|
151
160
|
|
|
152
161
|
/**
|
|
153
162
|
* Callback for when a block is received from a peer.
|
|
@@ -161,10 +170,19 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
161
170
|
* @param checkpoint - The checkpoint proposal received from the peer.
|
|
162
171
|
* @returns The attestations for the checkpoint, if any.
|
|
163
172
|
*/
|
|
164
|
-
private
|
|
173
|
+
private allNodesCheckpointReceivedCallback: P2PCheckpointReceivedCallback;
|
|
174
|
+
/**
|
|
175
|
+
* Callback for when a checkpoint proposal is received - specifically for validators - from a peer.
|
|
176
|
+
* @param checkpoint - The checkpoint proposal received from the peer.
|
|
177
|
+
* @returns The attestations for the checkpoint, if any.
|
|
178
|
+
*/
|
|
179
|
+
private validatorCheckpointReceivedCallback: P2PCheckpointReceivedCallback;
|
|
165
180
|
|
|
166
181
|
private gossipSubEventHandler: (e: CustomEvent<GossipsubMessage>) => void;
|
|
167
182
|
|
|
183
|
+
private ipChangedHandler?: (ip: string) => void;
|
|
184
|
+
private discoveredP2pIp?: string;
|
|
185
|
+
|
|
168
186
|
private instrumentation: P2PInstrumentation;
|
|
169
187
|
|
|
170
188
|
private telemetry: TelemetryClient;
|
|
@@ -172,19 +190,20 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
172
190
|
protected logger: Logger;
|
|
173
191
|
|
|
174
192
|
constructor(
|
|
175
|
-
private clientType: T,
|
|
176
193
|
private config: P2PConfig,
|
|
177
194
|
protected node: PubSubLibp2p,
|
|
178
195
|
private peerDiscoveryService: PeerDiscoveryService,
|
|
179
196
|
private reqresp: ReqRespInterface,
|
|
180
|
-
|
|
197
|
+
protected peerManager: PeerManagerInterface,
|
|
181
198
|
protected mempools: MemPools,
|
|
182
|
-
|
|
199
|
+
protected archiver: L2BlockSource & ContractDataSource,
|
|
183
200
|
private epochCache: EpochCacheInterface,
|
|
184
201
|
private proofVerifier: ClientProtocolCircuitVerifier,
|
|
185
202
|
private worldStateSynchronizer: WorldStateSynchronizer,
|
|
203
|
+
private blockMinFeesProvider: BlockMinFeesProvider,
|
|
186
204
|
telemetry: TelemetryClient,
|
|
187
205
|
logger: Logger = createLogger('p2p:libp2p_service'),
|
|
206
|
+
private txValidationCache?: TxValidationCache,
|
|
188
207
|
) {
|
|
189
208
|
super(telemetry, 'LibP2PService');
|
|
190
209
|
this.telemetry = telemetry;
|
|
@@ -214,36 +233,55 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
214
233
|
this.protocolVersion,
|
|
215
234
|
);
|
|
216
235
|
|
|
217
|
-
|
|
218
|
-
|
|
236
|
+
const p2pPropagationTime = config.attestationPropagationTime;
|
|
237
|
+
const proposalValidatorOpts = {
|
|
219
238
|
txsPermitted: !config.disableTransactions,
|
|
220
|
-
|
|
239
|
+
maxTxsPerBlock: config.validateMaxTxsPerBlock ?? config.validateMaxTxsPerCheckpoint,
|
|
240
|
+
maxBlocksPerCheckpoint: config.maxBlocksPerCheckpoint,
|
|
241
|
+
p2pPropagationTime,
|
|
242
|
+
skipSlotValidation: config.skipProposalSlotValidation,
|
|
243
|
+
signatureContext: {
|
|
244
|
+
chainId: config.l1ChainId,
|
|
245
|
+
rollupAddress: config.rollupAddress,
|
|
246
|
+
},
|
|
247
|
+
};
|
|
248
|
+
this.blockProposalValidator = new BlockProposalValidator(epochCache, proposalValidatorOpts);
|
|
249
|
+
this.checkpointProposalValidator = new CheckpointProposalValidator(epochCache, proposalValidatorOpts);
|
|
250
|
+
const attestationValidatorOpts = {
|
|
251
|
+
l1PublishingTime: config.l1PublishingTime,
|
|
252
|
+
p2pPropagationTime,
|
|
253
|
+
signatureContext: proposalValidatorOpts.signatureContext,
|
|
254
|
+
};
|
|
221
255
|
this.checkpointAttestationValidator = config.fishermanMode
|
|
222
|
-
? new FishermanAttestationValidator(epochCache, mempools.attestationPool, telemetry)
|
|
223
|
-
: new CheckpointAttestationValidator(epochCache);
|
|
256
|
+
? new FishermanAttestationValidator(epochCache, mempools.attestationPool, telemetry, attestationValidatorOpts)
|
|
257
|
+
: new CheckpointAttestationValidator(epochCache, attestationValidatorOpts);
|
|
224
258
|
|
|
225
259
|
this.gossipSubEventHandler = this.handleGossipSubEvent.bind(this);
|
|
226
260
|
|
|
227
261
|
this.blockReceivedCallback = async (block: BlockProposal): Promise<boolean> => {
|
|
228
|
-
this.logger.
|
|
229
|
-
`Handler not yet registered
|
|
262
|
+
this.logger.warn(
|
|
263
|
+
`Handler for block received not yet registered on P2P service. Received block ${block.blockNumber} for slot ${block.slotNumber} from peer.`,
|
|
230
264
|
{ p2pMessageIdentifier: await block.p2pMessageLoggingIdentifier() },
|
|
231
265
|
);
|
|
232
|
-
return
|
|
266
|
+
return true;
|
|
233
267
|
};
|
|
234
268
|
|
|
235
|
-
this.
|
|
236
|
-
|
|
269
|
+
this.allNodesCheckpointReceivedCallback = (
|
|
270
|
+
_checkpoint: CheckpointProposalCore,
|
|
271
|
+
): Promise<CheckpointAttestation[] | undefined> => {
|
|
272
|
+
throw new CheckpointProposalReceivedCallbackNotRegisteredError();
|
|
273
|
+
};
|
|
274
|
+
|
|
275
|
+
this.validatorCheckpointReceivedCallback = (
|
|
276
|
+
_checkpoint: CheckpointProposalCore,
|
|
237
277
|
): Promise<CheckpointAttestation[] | undefined> => {
|
|
238
|
-
this.logger.debug(
|
|
239
|
-
`Handler not yet registered: Checkpoint received callback not set. Received checkpoint for slot ${checkpoint.slotNumber} from peer.`,
|
|
240
|
-
);
|
|
241
278
|
return Promise.resolve(undefined);
|
|
242
279
|
};
|
|
243
280
|
}
|
|
244
281
|
|
|
245
|
-
public updateConfig(config: Partial<P2PReqRespConfig
|
|
282
|
+
public updateConfig(config: Partial<P2PReqRespConfig & Pick<P2PConfig, 'skipIncomingProposals'>>) {
|
|
246
283
|
this.reqresp.updateConfig(config);
|
|
284
|
+
this.config = merge(this.config, config);
|
|
247
285
|
}
|
|
248
286
|
|
|
249
287
|
/**
|
|
@@ -252,8 +290,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
252
290
|
* @param txPool - The transaction pool to be accessed by the service.
|
|
253
291
|
* @returns The new service.
|
|
254
292
|
*/
|
|
255
|
-
public static async new
|
|
256
|
-
clientType: T,
|
|
293
|
+
public static async new(
|
|
257
294
|
config: P2PConfig,
|
|
258
295
|
peerId: PeerId,
|
|
259
296
|
deps: {
|
|
@@ -263,9 +300,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
263
300
|
proofVerifier: ClientProtocolCircuitVerifier;
|
|
264
301
|
worldStateSynchronizer: WorldStateSynchronizer;
|
|
265
302
|
peerStore: AztecAsyncKVStore;
|
|
303
|
+
blockMinFeesProvider: BlockMinFeesProvider;
|
|
266
304
|
telemetry: TelemetryClient;
|
|
267
305
|
logger: Logger;
|
|
268
306
|
packageVersion: string;
|
|
307
|
+
txValidationCache?: TxValidationCache;
|
|
269
308
|
},
|
|
270
309
|
) {
|
|
271
310
|
const {
|
|
@@ -275,23 +314,25 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
275
314
|
mempools,
|
|
276
315
|
proofVerifier,
|
|
277
316
|
peerStore,
|
|
317
|
+
blockMinFeesProvider,
|
|
278
318
|
telemetry,
|
|
279
319
|
logger,
|
|
280
320
|
packageVersion,
|
|
321
|
+
txValidationCache,
|
|
281
322
|
} = deps;
|
|
282
323
|
const { p2pPort, maxPeerCount, listenAddress } = config;
|
|
283
324
|
const bindAddrTcp = convertToMultiaddr(listenAddress, p2pPort, 'tcp');
|
|
284
325
|
|
|
285
326
|
const datastore = new AztecDatastore(peerStore);
|
|
286
327
|
|
|
287
|
-
const otelMetricsAdapter = new OtelMetricsAdapter(telemetry);
|
|
328
|
+
const otelMetricsAdapter = new OtelMetricsAdapter(telemetry, logger.getBindings());
|
|
288
329
|
|
|
289
330
|
const peerDiscoveryService = new DiscV5Service(
|
|
290
331
|
peerId,
|
|
291
332
|
config,
|
|
292
333
|
packageVersion,
|
|
293
334
|
telemetry,
|
|
294
|
-
createLogger(`${logger.module}:discv5_service
|
|
335
|
+
createLogger(`${logger.module}:discv5_service`, logger.getBindings()),
|
|
295
336
|
);
|
|
296
337
|
|
|
297
338
|
// Seed libp2p's bootstrap discovery with private and trusted peers
|
|
@@ -305,11 +346,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
305
346
|
const versions = getVersions(config);
|
|
306
347
|
const protocolVersion = compressComponentVersions(versions);
|
|
307
348
|
|
|
308
|
-
const txTopic = createTopicString(TopicType.tx, protocolVersion);
|
|
309
|
-
const blockProposalTopic = createTopicString(TopicType.block_proposal, protocolVersion);
|
|
310
|
-
const checkpointProposalTopic = createTopicString(TopicType.checkpoint_proposal, protocolVersion);
|
|
311
|
-
const checkpointAttestationTopic = createTopicString(TopicType.checkpoint_attestation, protocolVersion);
|
|
312
|
-
|
|
313
349
|
const preferredPeersEnrs: ENR[] = config.preferredPeers.map(enr => ENR.decodeTxt(enr));
|
|
314
350
|
const directPeers = (
|
|
315
351
|
await Promise.all(
|
|
@@ -329,6 +365,19 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
329
365
|
|
|
330
366
|
const announceTcpMultiaddr = config.p2pIp ? [convertToMultiaddr(config.p2pIp, p2pPort, 'tcp')] : [];
|
|
331
367
|
|
|
368
|
+
// Create dynamic topic score params based on network configuration
|
|
369
|
+
const l1Constants = epochCache.getL1Constants();
|
|
370
|
+
const topicScoreParams = createAllTopicScoreParams(protocolVersion, {
|
|
371
|
+
slotDurationMs: l1Constants.slotDuration * 1000,
|
|
372
|
+
ethereumSlotDuration: l1Constants.ethereumSlotDuration,
|
|
373
|
+
heartbeatIntervalMs: config.gossipsubInterval,
|
|
374
|
+
targetCommitteeSize: l1Constants.targetCommitteeSize,
|
|
375
|
+
blockDurationMs: config.blockDurationMs,
|
|
376
|
+
l1PublishingTime: config.l1PublishingTime,
|
|
377
|
+
p2pPropagationTime: config.attestationPropagationTime,
|
|
378
|
+
expectedBlockProposalsPerSlot: config.expectedBlockProposalsPerSlot,
|
|
379
|
+
});
|
|
380
|
+
|
|
332
381
|
const node = await createLibp2p({
|
|
333
382
|
start: false,
|
|
334
383
|
peerId,
|
|
@@ -424,35 +473,15 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
424
473
|
scoreParams: createPeerScoreParams({
|
|
425
474
|
// IPColocation factor can be disabled for local testing - default to -5
|
|
426
475
|
IPColocationFactorWeight: config.debugDisableColocationPenalty ? 0 : -5.0,
|
|
427
|
-
topics:
|
|
428
|
-
[txTopic]: createTopicScoreParams({
|
|
429
|
-
topicWeight: 1,
|
|
430
|
-
invalidMessageDeliveriesWeight: -20,
|
|
431
|
-
invalidMessageDeliveriesDecay: 0.5,
|
|
432
|
-
}),
|
|
433
|
-
[blockProposalTopic]: createTopicScoreParams({
|
|
434
|
-
topicWeight: 1,
|
|
435
|
-
invalidMessageDeliveriesWeight: -20,
|
|
436
|
-
invalidMessageDeliveriesDecay: 0.5,
|
|
437
|
-
}),
|
|
438
|
-
[checkpointProposalTopic]: createTopicScoreParams({
|
|
439
|
-
topicWeight: 1,
|
|
440
|
-
invalidMessageDeliveriesWeight: -20,
|
|
441
|
-
invalidMessageDeliveriesDecay: 0.5,
|
|
442
|
-
}),
|
|
443
|
-
[checkpointAttestationTopic]: createTopicScoreParams({
|
|
444
|
-
topicWeight: 1,
|
|
445
|
-
invalidMessageDeliveriesWeight: -20,
|
|
446
|
-
invalidMessageDeliveriesDecay: 0.5,
|
|
447
|
-
}),
|
|
448
|
-
},
|
|
476
|
+
topics: topicScoreParams,
|
|
449
477
|
}),
|
|
450
478
|
}) as (components: GossipSubComponents) => GossipSub,
|
|
451
|
-
components: (components: { connectionManager: ConnectionManager }) => ({
|
|
479
|
+
components: (components: { connectionManager: ConnectionManager; addressManager: AddressManager }) => ({
|
|
452
480
|
connectionManager: components.connectionManager,
|
|
481
|
+
addressManager: components.addressManager,
|
|
453
482
|
}),
|
|
454
483
|
},
|
|
455
|
-
logger: createLibp2pComponentLogger(logger.module),
|
|
484
|
+
logger: createLibp2pComponentLogger(logger.module, logger.getBindings()),
|
|
456
485
|
});
|
|
457
486
|
|
|
458
487
|
const peerScoring = new PeerScoring(config, telemetry);
|
|
@@ -471,13 +500,19 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
471
500
|
epochCache,
|
|
472
501
|
);
|
|
473
502
|
|
|
474
|
-
//
|
|
475
|
-
|
|
503
|
+
// Gate req/resp data protocols for unauthenticated peers when p2pAllowOnlyValidators is enabled
|
|
504
|
+
reqresp.setShouldRejectPeer(peerId => peerManager.shouldDisableP2PGossip(peerId));
|
|
505
|
+
|
|
506
|
+
// Configure application-specific scoring for gossipsub.
|
|
507
|
+
// The weight scales app score to align with gossipsub thresholds:
|
|
508
|
+
// - Disconnect (-50) × 10 = -500 = gossipThreshold (stops receiving gossip)
|
|
509
|
+
// - Ban (-100) × 10 = -1000 = publishThreshold (cannot publish)
|
|
510
|
+
// Note: positive topic scores can offset penalties, so alignment is best-effort.
|
|
511
|
+
node.services.pubsub.score.params.appSpecificWeight = APP_SPECIFIC_WEIGHT;
|
|
476
512
|
node.services.pubsub.score.params.appSpecificScore = (peerId: string) =>
|
|
477
513
|
peerManager.shouldDisableP2PGossip(peerId) ? -Infinity : peerManager.getPeerScore(peerId);
|
|
478
514
|
|
|
479
515
|
return new LibP2PService(
|
|
480
|
-
clientType,
|
|
481
516
|
config,
|
|
482
517
|
node,
|
|
483
518
|
peerDiscoveryService,
|
|
@@ -488,8 +523,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
488
523
|
epochCache,
|
|
489
524
|
proofVerifier,
|
|
490
525
|
worldStateSynchronizer,
|
|
526
|
+
blockMinFeesProvider,
|
|
491
527
|
telemetry,
|
|
492
528
|
logger,
|
|
529
|
+
txValidationCache,
|
|
493
530
|
);
|
|
494
531
|
}
|
|
495
532
|
|
|
@@ -503,28 +540,29 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
503
540
|
throw new Error('P2P service already started');
|
|
504
541
|
}
|
|
505
542
|
|
|
506
|
-
// Get listen & announce addresses for logging
|
|
507
543
|
const { p2pIp, p2pPort } = this.config;
|
|
508
|
-
if (!p2pIp) {
|
|
509
|
-
throw new Error('Announce address not provided.');
|
|
544
|
+
if (!p2pIp && !this.config.queryForIp) {
|
|
545
|
+
throw new Error('Announce address not provided and queryForIp is not enabled.');
|
|
510
546
|
}
|
|
511
|
-
const announceTcpMultiaddr = convertToMultiaddr(p2pIp, p2pPort, 'tcp');
|
|
547
|
+
const announceTcpMultiaddr = p2pIp ? convertToMultiaddr(p2pIp, p2pPort, 'tcp') : undefined;
|
|
512
548
|
|
|
513
549
|
// Create request response protocol handlers
|
|
514
550
|
const txHandler = reqRespTxHandler(this.mempools);
|
|
515
551
|
const goodbyeHandler = reqGoodbyeHandler(this.peerManager);
|
|
516
|
-
const blockHandler = reqRespBlockHandler(this.archiver);
|
|
517
552
|
const statusHandler = reqRespStatusHandler(this.protocolVersion, this.worldStateSynchronizer, this.logger);
|
|
518
553
|
|
|
519
554
|
const requestResponseHandlers: Partial<ReqRespSubProtocolHandlers> = {
|
|
520
555
|
[ReqRespSubProtocol.PING]: pingHandler,
|
|
521
556
|
[ReqRespSubProtocol.STATUS]: statusHandler.bind(this),
|
|
522
557
|
[ReqRespSubProtocol.GOODBYE]: goodbyeHandler.bind(this),
|
|
523
|
-
[ReqRespSubProtocol.BLOCK]: blockHandler.bind(this),
|
|
524
558
|
};
|
|
525
559
|
|
|
526
560
|
if (!this.config.disableTransactions) {
|
|
527
|
-
const blockTxsHandler = reqRespBlockTxsHandler(
|
|
561
|
+
const blockTxsHandler = reqRespBlockTxsHandler(
|
|
562
|
+
this.mempools.attestationPool,
|
|
563
|
+
this.archiver,
|
|
564
|
+
this.mempools.txPool,
|
|
565
|
+
);
|
|
528
566
|
requestResponseHandlers[ReqRespSubProtocol.BLOCK_TXS] = blockTxsHandler.bind(this);
|
|
529
567
|
}
|
|
530
568
|
|
|
@@ -532,22 +570,14 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
532
570
|
requestResponseHandlers[ReqRespSubProtocol.TX] = txHandler.bind(this);
|
|
533
571
|
}
|
|
534
572
|
|
|
535
|
-
// Define the sub protocol validators - This is done within this start() method to gain a callback to the existing validateTx function
|
|
536
|
-
const reqrespSubProtocolValidators = {
|
|
537
|
-
...DEFAULT_SUB_PROTOCOL_VALIDATORS,
|
|
538
|
-
[ReqRespSubProtocol.TX]: this.validateRequestedTxs.bind(this),
|
|
539
|
-
[ReqRespSubProtocol.BLOCK_TXS]: this.validateRequestedBlockTxs.bind(this),
|
|
540
|
-
[ReqRespSubProtocol.BLOCK]: this.validateRequestedBlock.bind(this),
|
|
541
|
-
};
|
|
542
|
-
|
|
543
573
|
await this.peerManager.initializePeers();
|
|
544
574
|
|
|
545
|
-
await this.reqresp.start(requestResponseHandlers
|
|
575
|
+
await this.reqresp.start(requestResponseHandlers);
|
|
546
576
|
|
|
547
577
|
await this.node.start();
|
|
548
578
|
|
|
549
579
|
// Subscribe to standard GossipSub topics by default
|
|
550
|
-
for (const topic of
|
|
580
|
+
for (const topic of getTopicsForConfig(this.config.disableTransactions)) {
|
|
551
581
|
this.subscribeToTopic(this.topicStrings[topic]);
|
|
552
582
|
}
|
|
553
583
|
|
|
@@ -558,6 +588,38 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
558
588
|
if (!this.config.p2pDiscoveryDisabled) {
|
|
559
589
|
await this.peerDiscoveryService.start();
|
|
560
590
|
}
|
|
591
|
+
|
|
592
|
+
// Bridge discv5 IP changes to libp2p's AddressManager so peers see the updated address
|
|
593
|
+
if (this.config.queryForIp) {
|
|
594
|
+
this.discoveredP2pIp = this.config.p2pIp;
|
|
595
|
+
this.logger.info('IP change tracking enabled, bridging discv5 IP updates to libp2p AddressManager');
|
|
596
|
+
this.ipChangedHandler = (ip: string) => {
|
|
597
|
+
const addressManager = this.node.services.components.addressManager;
|
|
598
|
+
const newAddr = multiaddr(convertToMultiaddr(ip, this.config.p2pPort, 'tcp'));
|
|
599
|
+
const previousIp = this.discoveredP2pIp;
|
|
600
|
+
|
|
601
|
+
if (previousIp) {
|
|
602
|
+
const oldAddr = multiaddr(convertToMultiaddr(previousIp, this.config.p2pPort, 'tcp'));
|
|
603
|
+
addressManager.removeObservedAddr(oldAddr);
|
|
604
|
+
this.logger.info('Libp2p announce address updated due to IP change', {
|
|
605
|
+
previousIp,
|
|
606
|
+
newIp: ip,
|
|
607
|
+
newMultiaddr: newAddr.toString(),
|
|
608
|
+
});
|
|
609
|
+
} else {
|
|
610
|
+
this.logger.info('Libp2p announce address set from initial discv5 IP discovery', {
|
|
611
|
+
ip,
|
|
612
|
+
multiaddr: newAddr.toString(),
|
|
613
|
+
});
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
addressManager.addObservedAddr(newAddr);
|
|
617
|
+
addressManager.confirmObservedAddr(newAddr);
|
|
618
|
+
this.discoveredP2pIp = ip;
|
|
619
|
+
};
|
|
620
|
+
this.peerDiscoveryService.on('ip:changed', this.ipChangedHandler);
|
|
621
|
+
}
|
|
622
|
+
|
|
561
623
|
this.discoveryRunningPromise = new RunningPromise(
|
|
562
624
|
async () => {
|
|
563
625
|
await this.peerManager.heartbeat();
|
|
@@ -583,6 +645,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
583
645
|
// Remove gossip sub listener
|
|
584
646
|
this.node.services.pubsub.removeEventListener(GossipSubEvent.MESSAGE, this.gossipSubEventHandler);
|
|
585
647
|
|
|
648
|
+
if (this.ipChangedHandler) {
|
|
649
|
+
this.peerDiscoveryService.removeListener('ip:changed', this.ipChangedHandler);
|
|
650
|
+
this.ipChangedHandler = undefined;
|
|
651
|
+
}
|
|
652
|
+
|
|
586
653
|
// Stop peer manager
|
|
587
654
|
this.logger.debug('Stopping peer manager...');
|
|
588
655
|
await this.peerManager.stop();
|
|
@@ -597,12 +664,8 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
597
664
|
this.logger.info('LibP2P service stopped');
|
|
598
665
|
}
|
|
599
666
|
|
|
600
|
-
addReqRespSubProtocol(
|
|
601
|
-
subProtocol
|
|
602
|
-
handler: ReqRespSubProtocolHandler,
|
|
603
|
-
validator?: ReqRespSubProtocolValidators[ReqRespSubProtocol],
|
|
604
|
-
): Promise<void> {
|
|
605
|
-
return this.reqresp.addSubProtocol(subProtocol, handler, validator);
|
|
667
|
+
addReqRespSubProtocol(subProtocol: ReqRespSubProtocol, handler: ReqRespSubProtocolHandler): Promise<void> {
|
|
668
|
+
return this.reqresp.addSubProtocol(subProtocol, handler);
|
|
606
669
|
}
|
|
607
670
|
|
|
608
671
|
public registerThisValidatorAddresses(address: EthAddress[]): void {
|
|
@@ -613,6 +676,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
613
676
|
return this.peerManager.getPeers(includePending);
|
|
614
677
|
}
|
|
615
678
|
|
|
679
|
+
public getGossipMeshPeerCount(topicType: TopicType): number {
|
|
680
|
+
return this.node.services.pubsub.getMeshPeers(this.topicStrings[topicType]).length;
|
|
681
|
+
}
|
|
682
|
+
|
|
616
683
|
private handleGossipSubEvent(e: CustomEvent<GossipsubMessage>) {
|
|
617
684
|
this.logger.trace(`Received PUBSUB message.`);
|
|
618
685
|
|
|
@@ -626,18 +693,13 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
626
693
|
setImmediate(() => void safeJob());
|
|
627
694
|
}
|
|
628
695
|
|
|
629
|
-
|
|
630
|
-
|
|
631
|
-
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
|
|
635
|
-
|
|
636
|
-
protocol: SubProtocol,
|
|
637
|
-
requests: InstanceType<SubProtocolMap[SubProtocol]['request']>[],
|
|
638
|
-
pinnedPeerId: PeerId | undefined,
|
|
639
|
-
): Promise<InstanceType<SubProtocolMap[SubProtocol]['response']>[]> {
|
|
640
|
-
return this.reqresp.sendBatchRequest(protocol, requests, pinnedPeerId);
|
|
696
|
+
public sendRequestToPeer(
|
|
697
|
+
peerId: PeerId,
|
|
698
|
+
subProtocol: ReqRespSubProtocol,
|
|
699
|
+
payload: Buffer,
|
|
700
|
+
dialTimeout?: number,
|
|
701
|
+
): Promise<ReqRespResponse> {
|
|
702
|
+
return this.reqresp.sendRequestToPeer(peerId, subProtocol, payload, dialTimeout);
|
|
641
703
|
}
|
|
642
704
|
|
|
643
705
|
/**
|
|
@@ -652,8 +714,35 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
652
714
|
this.blockReceivedCallback = callback;
|
|
653
715
|
}
|
|
654
716
|
|
|
655
|
-
public
|
|
656
|
-
this.
|
|
717
|
+
public registerValidatorCheckpointReceivedCallback(callback: P2PCheckpointReceivedCallback) {
|
|
718
|
+
this.validatorCheckpointReceivedCallback = callback;
|
|
719
|
+
}
|
|
720
|
+
|
|
721
|
+
public registerAllNodesCheckpointReceivedCallback(callback: P2PCheckpointReceivedCallback) {
|
|
722
|
+
this.allNodesCheckpointReceivedCallback = callback;
|
|
723
|
+
}
|
|
724
|
+
|
|
725
|
+
/**
|
|
726
|
+
* Registers a callback to be invoked when a duplicate proposal is detected.
|
|
727
|
+
* This callback is triggered on the first duplicate (when count goes from 1 to 2).
|
|
728
|
+
*/
|
|
729
|
+
public registerDuplicateProposalCallback(
|
|
730
|
+
callback: (info: { slot: SlotNumber; proposer: EthAddress; type: 'checkpoint' | 'block' }) => void,
|
|
731
|
+
): void {
|
|
732
|
+
this.duplicateProposalCallback = callback;
|
|
733
|
+
}
|
|
734
|
+
|
|
735
|
+
/**
|
|
736
|
+
* Registers a callback to be invoked when a duplicate attestation is detected.
|
|
737
|
+
* A validator signing attestations for different proposals at the same slot.
|
|
738
|
+
* This callback is triggered on the first duplicate (when count goes from 1 to 2).
|
|
739
|
+
*/
|
|
740
|
+
public registerDuplicateAttestationCallback(callback: P2PDuplicateAttestationCallback): void {
|
|
741
|
+
this.duplicateAttestationCallback = callback;
|
|
742
|
+
}
|
|
743
|
+
|
|
744
|
+
public registerCheckpointAttestationCallback(callback: P2PCheckpointAttestationCallback): void {
|
|
745
|
+
this.checkpointAttestationCallback = callback;
|
|
657
746
|
}
|
|
658
747
|
|
|
659
748
|
/**
|
|
@@ -720,6 +809,9 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
720
809
|
if (!validator || !validator.addMessage(msgId)) {
|
|
721
810
|
this.instrumentation.incMessagePrevalidationStatus(false, topicType);
|
|
722
811
|
this.node.services.pubsub.reportMessageValidationResult(msgId, source.toString(), TopicValidatorResult.Ignore);
|
|
812
|
+
if (topicType === TopicType.tx) {
|
|
813
|
+
this.logger.verbose(`Ignoring already-seen tx gossip message`, { msgId, source: source.toString() });
|
|
814
|
+
}
|
|
723
815
|
return { result: false, topicType };
|
|
724
816
|
}
|
|
725
817
|
|
|
@@ -781,12 +873,19 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
781
873
|
|
|
782
874
|
// Process the message, optionally within a linked span for trace propagation
|
|
783
875
|
const processMessage = async () => {
|
|
876
|
+
if (
|
|
877
|
+
this.config.skipIncomingProposals &&
|
|
878
|
+
(msg.topic === this.topicStrings[TopicType.block_proposal] ||
|
|
879
|
+
msg.topic === this.topicStrings[TopicType.checkpoint_proposal])
|
|
880
|
+
) {
|
|
881
|
+
this.logger.warn(`Ignoring incoming proposal (skipIncomingProposals is set)`, { topic: msg.topic });
|
|
882
|
+
this.node.services.pubsub.reportMessageValidationResult(msgId, source.toString(), TopicValidatorResult.Ignore);
|
|
883
|
+
return;
|
|
884
|
+
}
|
|
784
885
|
if (msg.topic === this.topicStrings[TopicType.tx]) {
|
|
785
886
|
await this.handleGossipedTx(p2pMessage.payload, msgId, source);
|
|
786
887
|
} else if (msg.topic === this.topicStrings[TopicType.checkpoint_attestation]) {
|
|
787
|
-
|
|
788
|
-
await this.processCheckpointAttestationFromPeer(p2pMessage.payload, msgId, source);
|
|
789
|
-
}
|
|
888
|
+
await this.processCheckpointAttestationFromPeer(p2pMessage.payload, msgId, source);
|
|
790
889
|
} else if (msg.topic === this.topicStrings[TopicType.block_proposal]) {
|
|
791
890
|
await this.processBlockFromPeer(p2pMessage.payload, msgId, source);
|
|
792
891
|
} else if (msg.topic === this.topicStrings[TopicType.checkpoint_proposal]) {
|
|
@@ -842,51 +941,145 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
842
941
|
return;
|
|
843
942
|
}
|
|
844
943
|
|
|
845
|
-
protected async validateReceivedMessage<T>(
|
|
846
|
-
validationFunc: () => Promise<ReceivedMessageValidationResult<T>>,
|
|
944
|
+
protected async validateReceivedMessage<T, M = undefined>(
|
|
945
|
+
validationFunc: () => Promise<ReceivedMessageValidationResult<T, M>>,
|
|
847
946
|
msgId: string,
|
|
848
947
|
source: PeerId,
|
|
849
948
|
topicType: TopicType,
|
|
850
|
-
): Promise<ReceivedMessageValidationResult<T>> {
|
|
851
|
-
|
|
949
|
+
): Promise<ReceivedMessageValidationResult<T, M>> {
|
|
950
|
+
// Default to reject result with a penalty if validation function throws an error
|
|
951
|
+
let resultAndObj: ReceivedMessageValidationResult<T, M> = {
|
|
952
|
+
result: TopicValidatorResult.Reject,
|
|
953
|
+
severity: PeerErrorSeverity.MidToleranceError,
|
|
954
|
+
};
|
|
852
955
|
const timer = new Timer();
|
|
853
956
|
try {
|
|
854
957
|
resultAndObj = await validationFunc();
|
|
855
958
|
} catch (err) {
|
|
856
|
-
this.
|
|
857
|
-
|
|
858
|
-
|
|
859
|
-
|
|
860
|
-
|
|
861
|
-
|
|
959
|
+
this.logger.error(`Error validating gossipsub message`, err, { msgId, source: source.toString(), topicType });
|
|
960
|
+
}
|
|
961
|
+
|
|
962
|
+
const validationTimeMs = timer.ms();
|
|
963
|
+
const mcacheWindowMs = this.config.gossipsubMcacheLength * this.config.gossipsubInterval;
|
|
964
|
+
if (validationTimeMs > mcacheWindowMs * 0.75) {
|
|
965
|
+
this.instrumentation.incSlowValidation(topicType);
|
|
966
|
+
this.logger.warn(
|
|
967
|
+
`Gossip validation for ${topicType} took ${validationTimeMs}ms, approaching mcache eviction window of ${mcacheWindowMs}ms. ` +
|
|
968
|
+
`Message forwarding may be skipped if validation exceeds the window.`,
|
|
969
|
+
{ msgId, source: source.toString(), topicType, validationTimeMs, mcacheWindowMs },
|
|
970
|
+
);
|
|
862
971
|
}
|
|
863
972
|
|
|
864
973
|
if (resultAndObj.result === TopicValidatorResult.Accept) {
|
|
974
|
+
this.logger.debug(`Message ${topicType} accepted by validator`, { msgId, source: source.toString(), topicType });
|
|
865
975
|
this.instrumentation.recordMessageValidation(topicType, timer);
|
|
976
|
+
} else if (resultAndObj.result === TopicValidatorResult.Reject) {
|
|
977
|
+
this.logger.warn(`Message ${topicType} rejected by validator with severity ${resultAndObj.severity}`, {
|
|
978
|
+
msgId,
|
|
979
|
+
source: source.toString(),
|
|
980
|
+
topicType,
|
|
981
|
+
severity: resultAndObj.severity,
|
|
982
|
+
});
|
|
983
|
+
this.peerManager.penalizePeer(source, resultAndObj.severity);
|
|
984
|
+
} else {
|
|
985
|
+
this.logger.trace(`Message ${topicType} ignored by validator`, { msgId, source: source.toString(), topicType });
|
|
866
986
|
}
|
|
867
987
|
|
|
868
988
|
this.node.services.pubsub.reportMessageValidationResult(msgId, source.toString(), resultAndObj.result);
|
|
869
989
|
return resultAndObj;
|
|
870
990
|
}
|
|
871
991
|
|
|
992
|
+
private tryDeserialize<T>(deserializeFunc: () => T, msgId: string, source: PeerId): T | undefined {
|
|
993
|
+
try {
|
|
994
|
+
return deserializeFunc();
|
|
995
|
+
} catch (err) {
|
|
996
|
+
this.logger.warn(`Failed to deserialize gossipsub message from buffer`, {
|
|
997
|
+
err,
|
|
998
|
+
msgId,
|
|
999
|
+
source: source.toString(),
|
|
1000
|
+
});
|
|
1001
|
+
return undefined;
|
|
1002
|
+
}
|
|
1003
|
+
}
|
|
1004
|
+
|
|
872
1005
|
protected async handleGossipedTx(payloadData: Buffer, msgId: string, source: PeerId) {
|
|
873
1006
|
const validationFunc: () => Promise<ReceivedMessageValidationResult<Tx>> = async () => {
|
|
874
|
-
const tx = Tx.fromBuffer(payloadData);
|
|
875
|
-
|
|
876
|
-
|
|
1007
|
+
const tx = this.tryDeserialize(() => Tx.fromBuffer(payloadData), msgId, source);
|
|
1008
|
+
if (!tx) {
|
|
1009
|
+
return { result: TopicValidatorResult.Reject, severity: PeerErrorSeverity.LowToleranceError };
|
|
1010
|
+
}
|
|
1011
|
+
|
|
1012
|
+
const currentBlockNumber = await this.archiver.getBlockNumber();
|
|
1013
|
+
const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
|
|
1014
|
+
|
|
1015
|
+
// Stage 1: fast validators (metadata, data, timestamps, double-spend, gas, phases, block header)
|
|
1016
|
+
const firstStageValidators = await this.createFirstStageMessageValidators(currentBlockNumber, nextSlotTimestamp);
|
|
1017
|
+
const firstStageOutcome = await this.runValidations(tx, firstStageValidators);
|
|
1018
|
+
if (!firstStageOutcome.allPassed) {
|
|
1019
|
+
const { name } = firstStageOutcome.failure;
|
|
1020
|
+
let { severity } = firstStageOutcome.failure;
|
|
1021
|
+
|
|
1022
|
+
// Double spend validator has a special case handler. We perform more detailed examination
|
|
1023
|
+
// as to how recently the nullifier was entered into the tree and if the transaction should
|
|
1024
|
+
// have 'known' the nullifier existed. This determines the severity of the penalty applied to the peer.
|
|
1025
|
+
if (name === 'doubleSpendValidator') {
|
|
1026
|
+
const txBlockNumber = BlockNumber(currentBlockNumber + 1);
|
|
1027
|
+
severity = await this.handleDoubleSpendFailure(tx, txBlockNumber);
|
|
1028
|
+
}
|
|
1029
|
+
|
|
1030
|
+
this.logger.verbose(`Rejecting gossiped tx ${tx.getTxHash().toString()}: stage 1 validation failed`, {
|
|
1031
|
+
validator: name,
|
|
1032
|
+
severity,
|
|
1033
|
+
source: source.toString(),
|
|
1034
|
+
});
|
|
1035
|
+
return { result: TopicValidatorResult.Reject, severity };
|
|
1036
|
+
}
|
|
877
1037
|
|
|
878
|
-
this
|
|
879
|
-
|
|
880
|
-
|
|
1038
|
+
// Pool pre-check: see if the pool would accept this tx before doing expensive proof verification
|
|
1039
|
+
const canAdd = await this.mempools.txPool.canAddPendingTx(tx);
|
|
1040
|
+
if (canAdd === 'ignored') {
|
|
1041
|
+
this.logger.verbose(`Ignoring gossiped tx ${tx.getTxHash().toString()}: pool pre-check returned ignored`, {
|
|
1042
|
+
source: source.toString(),
|
|
1043
|
+
});
|
|
1044
|
+
return { result: TopicValidatorResult.Ignore, obj: tx };
|
|
1045
|
+
}
|
|
1046
|
+
|
|
1047
|
+
// Stage 2: expensive proof verification
|
|
1048
|
+
const secondStageValidators = this.createSecondStageMessageValidators();
|
|
1049
|
+
const secondStageOutcome = await this.runValidations(tx, secondStageValidators);
|
|
1050
|
+
if (!secondStageOutcome.allPassed) {
|
|
1051
|
+
const { severity, name } = secondStageOutcome.failure;
|
|
1052
|
+
this.logger.verbose(`Rejecting gossiped tx ${tx.getTxHash().toString()}: stage 2 validation failed`, {
|
|
1053
|
+
validator: name,
|
|
1054
|
+
severity,
|
|
1055
|
+
source: source.toString(),
|
|
1056
|
+
});
|
|
1057
|
+
return { result: TopicValidatorResult.Reject, severity };
|
|
1058
|
+
}
|
|
1059
|
+
|
|
1060
|
+
// Pool add: persist the tx
|
|
1061
|
+
const txHash = tx.getTxHash();
|
|
1062
|
+
const addResult = await this.mempools.txPool.addPendingTxs([tx], { source: 'gossip' });
|
|
1063
|
+
|
|
1064
|
+
const wasAccepted = addResult.accepted.some(h => h.equals(txHash));
|
|
1065
|
+
const wasIgnored = addResult.ignored.some(h => h.equals(txHash));
|
|
1066
|
+
|
|
1067
|
+
this.logger.verbose(`Validate propagated tx ${txHash.toString()}`, {
|
|
1068
|
+
wasAccepted,
|
|
1069
|
+
wasIgnored,
|
|
881
1070
|
[Attributes.P2P_ID]: source.toString(),
|
|
882
1071
|
});
|
|
883
1072
|
|
|
884
|
-
if (
|
|
885
|
-
return { result: TopicValidatorResult.
|
|
886
|
-
} else if (
|
|
1073
|
+
if (wasAccepted) {
|
|
1074
|
+
return { result: TopicValidatorResult.Accept, obj: tx };
|
|
1075
|
+
} else if (wasIgnored) {
|
|
887
1076
|
return { result: TopicValidatorResult.Ignore, obj: tx };
|
|
888
1077
|
} else {
|
|
889
|
-
|
|
1078
|
+
this.logger.warn(`Gossiped tx ${txHash.toString()} unexpectedly rejected by pool`, {
|
|
1079
|
+
source: source.toString(),
|
|
1080
|
+
txHash: txHash.toString(),
|
|
1081
|
+
});
|
|
1082
|
+
return { result: TopicValidatorResult.Reject, severity: PeerErrorSeverity.HighToleranceError };
|
|
890
1083
|
}
|
|
891
1084
|
};
|
|
892
1085
|
|
|
@@ -895,6 +1088,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
895
1088
|
return;
|
|
896
1089
|
}
|
|
897
1090
|
|
|
1091
|
+
// Tx was accepted into pool and will be propagated - just log and record metrics
|
|
898
1092
|
const txHash = tx.getTxHash();
|
|
899
1093
|
const txHashString = txHash.toString();
|
|
900
1094
|
this.logger.verbose(`Received tx ${txHashString} from external peer ${source.toString()} via gossip`, {
|
|
@@ -902,13 +1096,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
902
1096
|
txHash: txHashString,
|
|
903
1097
|
});
|
|
904
1098
|
|
|
905
|
-
if (this.config.dropTransactions && randomInt(1000) < this.config.dropTransactionsProbability * 1000) {
|
|
906
|
-
this.logger.warn(`Intentionally dropping tx ${txHashString} (probability rule)`);
|
|
907
|
-
return;
|
|
908
|
-
}
|
|
909
|
-
|
|
910
1099
|
this.instrumentation.incrementTxReceived(1);
|
|
911
|
-
await this.mempools.txPool.addTxs([tx]);
|
|
912
1100
|
}
|
|
913
1101
|
|
|
914
1102
|
/**
|
|
@@ -920,46 +1108,17 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
920
1108
|
msgId: string,
|
|
921
1109
|
source: PeerId,
|
|
922
1110
|
): Promise<void> {
|
|
923
|
-
const validationFunc: () => Promise<ReceivedMessageValidationResult<CheckpointAttestation>> = async () => {
|
|
924
|
-
const attestation = CheckpointAttestation.fromBuffer(payloadData);
|
|
925
|
-
const pool = this.mempools.attestationPool;
|
|
926
|
-
const isValid = await this.validateCheckpointAttestation(source, attestation);
|
|
927
|
-
const exists = isValid && (await pool.hasCheckpointAttestation(attestation));
|
|
928
|
-
|
|
929
|
-
let canAdd = true;
|
|
930
|
-
if (isValid && !exists) {
|
|
931
|
-
const slot = attestation.payload.header.slotNumber;
|
|
932
|
-
const { committee } = await this.epochCache.getCommittee(slot);
|
|
933
|
-
const committeeSize = committee?.length ?? 0;
|
|
934
|
-
canAdd = await pool.canAddCheckpointAttestation(attestation, committeeSize);
|
|
935
|
-
}
|
|
936
|
-
|
|
937
|
-
this.logger.trace(`Validate propagated checkpoint attestation`, {
|
|
938
|
-
isValid,
|
|
939
|
-
exists,
|
|
940
|
-
canAdd,
|
|
941
|
-
[Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber.toString(),
|
|
942
|
-
[Attributes.P2P_ID]: source.toString(),
|
|
943
|
-
});
|
|
944
|
-
|
|
945
|
-
if (!isValid) {
|
|
946
|
-
return { result: TopicValidatorResult.Reject };
|
|
947
|
-
} else if (exists) {
|
|
948
|
-
return { result: TopicValidatorResult.Ignore, obj: attestation };
|
|
949
|
-
} else if (!canAdd) {
|
|
950
|
-
this.logger.warn(`Dropping checkpoint attestation due to per-(slot, proposalId) attestation cap`, {
|
|
951
|
-
slot: attestation.payload.header.slotNumber.toString(),
|
|
952
|
-
archive: attestation.archive.toString(),
|
|
953
|
-
source: source.toString(),
|
|
954
|
-
});
|
|
955
|
-
return { result: TopicValidatorResult.Ignore, obj: attestation };
|
|
956
|
-
} else {
|
|
957
|
-
return { result: TopicValidatorResult.Accept, obj: attestation };
|
|
958
|
-
}
|
|
959
|
-
};
|
|
960
|
-
|
|
961
1111
|
const { result, obj: attestation } = await this.validateReceivedMessage<CheckpointAttestation>(
|
|
962
|
-
|
|
1112
|
+
() => {
|
|
1113
|
+
const attestation = this.tryDeserialize(() => CheckpointAttestation.fromBuffer(payloadData), msgId, source);
|
|
1114
|
+
if (!attestation) {
|
|
1115
|
+
return Promise.resolve({
|
|
1116
|
+
result: TopicValidatorResult.Reject,
|
|
1117
|
+
severity: PeerErrorSeverity.LowToleranceError,
|
|
1118
|
+
});
|
|
1119
|
+
}
|
|
1120
|
+
return this.validateAndStoreCheckpointAttestation(source, attestation);
|
|
1121
|
+
},
|
|
963
1122
|
msgId,
|
|
964
1123
|
source,
|
|
965
1124
|
TopicType.checkpoint_attestation,
|
|
@@ -969,8 +1128,8 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
969
1128
|
return;
|
|
970
1129
|
}
|
|
971
1130
|
|
|
972
|
-
this.logger.
|
|
973
|
-
`Received checkpoint attestation for slot ${attestation.slotNumber} from external peer ${source.toString()}`,
|
|
1131
|
+
this.logger.verbose(
|
|
1132
|
+
`Received valid checkpoint attestation for slot ${attestation.slotNumber} from external peer ${source.toString()}`,
|
|
974
1133
|
{
|
|
975
1134
|
p2pMessageIdentifier: await attestation.p2pMessageLoggingIdentifier(),
|
|
976
1135
|
slot: attestation.slotNumber,
|
|
@@ -978,59 +1137,169 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
978
1137
|
source: source.toString(),
|
|
979
1138
|
},
|
|
980
1139
|
);
|
|
981
|
-
|
|
982
|
-
await this.mempools.attestationPool.addCheckpointAttestations([attestation]);
|
|
983
1140
|
}
|
|
984
1141
|
|
|
985
|
-
|
|
986
|
-
|
|
987
|
-
|
|
988
|
-
|
|
989
|
-
|
|
1142
|
+
/** Validates a checkpoint attestation and adds it to the pool. Penalizes the peer if validation fails. */
|
|
1143
|
+
@trackSpan('Libp2pService.validateAndStoreCheckpointAttestation', (_peerId, attestation) => ({
|
|
1144
|
+
[Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber.toString(),
|
|
1145
|
+
}))
|
|
1146
|
+
protected async validateAndStoreCheckpointAttestation(
|
|
1147
|
+
peerId: PeerId,
|
|
1148
|
+
attestation: CheckpointAttestation,
|
|
1149
|
+
): Promise<ReceivedMessageValidationResult<CheckpointAttestation>> {
|
|
1150
|
+
const validationResult = await this.checkpointAttestationValidator.validate(attestation);
|
|
990
1151
|
|
|
991
|
-
|
|
992
|
-
|
|
1152
|
+
if (validationResult.result === 'reject') {
|
|
1153
|
+
this.logger.warn(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
|
|
1154
|
+
return { result: TopicValidatorResult.Reject, severity: validationResult.severity };
|
|
1155
|
+
}
|
|
993
1156
|
|
|
994
|
-
|
|
995
|
-
|
|
996
|
-
|
|
997
|
-
|
|
998
|
-
|
|
999
|
-
|
|
1157
|
+
if (validationResult.result === 'ignore') {
|
|
1158
|
+
return { result: TopicValidatorResult.Ignore, obj: attestation };
|
|
1159
|
+
}
|
|
1160
|
+
|
|
1161
|
+
// Try to add the attestation: this handles existence check, cap check, and adding in one call
|
|
1162
|
+
// count is the number of attestations by this signer for this slot (for duplicate detection)
|
|
1163
|
+
const slot = attestation.payload.header.slotNumber;
|
|
1164
|
+
const { added, alreadyExists, count } =
|
|
1165
|
+
await this.mempools.attestationPool.tryAddCheckpointAttestation(attestation);
|
|
1166
|
+
|
|
1167
|
+
this.logger.trace(`Validate propagated checkpoint attestation`, {
|
|
1168
|
+
added,
|
|
1169
|
+
alreadyExists,
|
|
1170
|
+
count,
|
|
1171
|
+
[Attributes.SLOT_NUMBER]: slot.toString(),
|
|
1172
|
+
[Attributes.P2P_ID]: peerId.toString(),
|
|
1173
|
+
});
|
|
1174
|
+
|
|
1175
|
+
// Exact same attestation received, no need to re-broadcast
|
|
1176
|
+
if (alreadyExists) {
|
|
1177
|
+
return { result: TopicValidatorResult.Ignore, obj: attestation };
|
|
1178
|
+
}
|
|
1179
|
+
|
|
1180
|
+
// Could not add (cap reached for signer), penalize and do not re-broadcast
|
|
1181
|
+
if (!added) {
|
|
1182
|
+
this.logger.warn(`Rejecting checkpoint attestation due to cap`, {
|
|
1183
|
+
slot: slot.toString(),
|
|
1184
|
+
archive: attestation.archive.toString(),
|
|
1185
|
+
source: peerId.toString(),
|
|
1186
|
+
attester: attestation.getSender()?.toString(),
|
|
1187
|
+
count,
|
|
1000
1188
|
});
|
|
1189
|
+
return { result: TopicValidatorResult.Reject, severity: PeerErrorSeverity.HighToleranceError };
|
|
1190
|
+
}
|
|
1001
1191
|
|
|
1002
|
-
|
|
1003
|
-
|
|
1004
|
-
|
|
1005
|
-
|
|
1006
|
-
|
|
1007
|
-
this.
|
|
1008
|
-
|
|
1009
|
-
|
|
1010
|
-
|
|
1011
|
-
|
|
1192
|
+
// Check if this is a duplicate attestation (signer attested to a different proposal at the same slot)
|
|
1193
|
+
// count is the number of attestations by this signer for this slot
|
|
1194
|
+
if (count === 2) {
|
|
1195
|
+
const attester = attestation.getSender();
|
|
1196
|
+
if (attester) {
|
|
1197
|
+
this.logger.warn(`Detected duplicate attestation (equivocation) at slot ${slot}`, {
|
|
1198
|
+
slot: slot.toString(),
|
|
1199
|
+
archive: attestation.archive.toString(),
|
|
1200
|
+
source: peerId.toString(),
|
|
1201
|
+
attester: attester.toString(),
|
|
1012
1202
|
});
|
|
1013
|
-
|
|
1014
|
-
} else {
|
|
1015
|
-
return { result: TopicValidatorResult.Accept, obj: block };
|
|
1203
|
+
this.duplicateAttestationCallback?.({ slot, attester });
|
|
1016
1204
|
}
|
|
1017
|
-
}
|
|
1205
|
+
}
|
|
1018
1206
|
|
|
1019
|
-
|
|
1020
|
-
|
|
1207
|
+
// Attestation was added successfully - accept it so other nodes can also detect the equivocation
|
|
1208
|
+
this.checkpointAttestationCallback?.(attestation);
|
|
1209
|
+
return { result: TopicValidatorResult.Accept, obj: attestation };
|
|
1210
|
+
}
|
|
1211
|
+
|
|
1212
|
+
protected async processBlockFromPeer(payloadData: Buffer, msgId: string, source: PeerId): Promise<void> {
|
|
1213
|
+
const {
|
|
1214
|
+
result,
|
|
1215
|
+
obj: block,
|
|
1216
|
+
metadata: { isEquivocated } = {},
|
|
1217
|
+
} = await this.validateReceivedMessage<BlockProposal, { isEquivocated: boolean }>(
|
|
1218
|
+
() => this.validateAndStoreBlockProposal(source, BlockProposal.fromBuffer(payloadData)),
|
|
1021
1219
|
msgId,
|
|
1022
1220
|
source,
|
|
1023
1221
|
TopicType.block_proposal,
|
|
1024
1222
|
);
|
|
1025
1223
|
|
|
1026
|
-
|
|
1224
|
+
// If not accepted or equivocated, return
|
|
1225
|
+
if (result !== TopicValidatorResult.Accept || !block || isEquivocated) {
|
|
1027
1226
|
return;
|
|
1028
1227
|
}
|
|
1029
1228
|
|
|
1030
1229
|
await this.processValidBlockProposal(block, source);
|
|
1031
1230
|
}
|
|
1032
1231
|
|
|
1033
|
-
|
|
1232
|
+
/** Validates a block proposal. Triggers a penalization to the peer that sent it if invalid. Adds to the mempool if valid. */
|
|
1233
|
+
@trackSpan('Libp2pService.validateAndStoreBlockProposal', (_peerId, block) => ({
|
|
1234
|
+
[Attributes.BLOCK_NUMBER]: block.blockNumber.toString(),
|
|
1235
|
+
[Attributes.SLOT_NUMBER]: block.slotNumber.toString(),
|
|
1236
|
+
}))
|
|
1237
|
+
protected async validateAndStoreBlockProposal(
|
|
1238
|
+
peerId: PeerId,
|
|
1239
|
+
block: BlockProposal,
|
|
1240
|
+
): Promise<ReceivedMessageValidationResult<BlockProposal, { isEquivocated: boolean }>> {
|
|
1241
|
+
const validationResult = await this.blockProposalValidator.validate(block);
|
|
1242
|
+
|
|
1243
|
+
if (validationResult.result === 'reject') {
|
|
1244
|
+
this.logger.warn(`Penalizing peer ${peerId} for block proposal validation failure`);
|
|
1245
|
+
return { result: TopicValidatorResult.Reject, severity: validationResult.severity };
|
|
1246
|
+
}
|
|
1247
|
+
|
|
1248
|
+
if (validationResult.result === 'ignore') {
|
|
1249
|
+
return { result: TopicValidatorResult.Ignore, obj: block };
|
|
1250
|
+
}
|
|
1251
|
+
|
|
1252
|
+
// Try to add the proposal: this handles existence check, cap check, and adding in one call
|
|
1253
|
+
const { added, alreadyExists, count } = await this.mempools.attestationPool.tryAddBlockProposal(block);
|
|
1254
|
+
const isEquivocated = count !== undefined && count > 1;
|
|
1255
|
+
|
|
1256
|
+
// Duplicate proposal received, no need to re-broadcast
|
|
1257
|
+
if (alreadyExists) {
|
|
1258
|
+
this.logger.debug(`Ignoring duplicate block proposal received`, {
|
|
1259
|
+
...block.toBlockInfo(),
|
|
1260
|
+
indexWithinCheckpoint: block.indexWithinCheckpoint,
|
|
1261
|
+
proposer: block.getSender()?.toString(),
|
|
1262
|
+
source: peerId.toString(),
|
|
1263
|
+
});
|
|
1264
|
+
return { result: TopicValidatorResult.Ignore, obj: block, metadata: { isEquivocated } };
|
|
1265
|
+
}
|
|
1266
|
+
|
|
1267
|
+
// Too many blocks received for this slot and index, penalize peer and do not re-broadcast
|
|
1268
|
+
if (!added) {
|
|
1269
|
+
this.logger.warn(`Penalizing peer for block proposal exceeding per-position cap`, {
|
|
1270
|
+
...block.toBlockInfo(),
|
|
1271
|
+
indexWithinCheckpoint: block.indexWithinCheckpoint,
|
|
1272
|
+
count,
|
|
1273
|
+
proposer: block.getSender()?.toString(),
|
|
1274
|
+
source: peerId.toString(),
|
|
1275
|
+
});
|
|
1276
|
+
return {
|
|
1277
|
+
result: TopicValidatorResult.Reject,
|
|
1278
|
+
metadata: { isEquivocated },
|
|
1279
|
+
severity: PeerErrorSeverity.HighToleranceError,
|
|
1280
|
+
};
|
|
1281
|
+
}
|
|
1282
|
+
|
|
1283
|
+
// If this was a duplicate proposal, do not process it, but do invoke the duplicate callback,
|
|
1284
|
+
// and do re-broadcast it so other nodes in the network know to slash the proposer
|
|
1285
|
+
if (isEquivocated) {
|
|
1286
|
+
const proposer = block.getSender();
|
|
1287
|
+
this.logger.warn(`Detected duplicate block proposal (equivocation) at slot ${block.slotNumber}`, {
|
|
1288
|
+
...block.toBlockInfo(),
|
|
1289
|
+
source: peerId.toString(),
|
|
1290
|
+
proposer: proposer?.toString(),
|
|
1291
|
+
});
|
|
1292
|
+
// Invoke the duplicate callback on the first duplicate spotted only
|
|
1293
|
+
if (proposer && count === 2) {
|
|
1294
|
+
this.duplicateProposalCallback?.({ slot: block.slotNumber, proposer, type: 'block' });
|
|
1295
|
+
}
|
|
1296
|
+
return { result: TopicValidatorResult.Accept, obj: block, metadata: { isEquivocated } };
|
|
1297
|
+
}
|
|
1298
|
+
|
|
1299
|
+
// Otherwise, we're good to go!
|
|
1300
|
+
return { result: TopicValidatorResult.Accept, obj: block };
|
|
1301
|
+
}
|
|
1302
|
+
|
|
1034
1303
|
// REFACTOR(palla): This method should be moved to the p2p_client or to a separate component,
|
|
1035
1304
|
// should not be here as it does not deal with p2p networking.
|
|
1036
1305
|
@trackSpan('Libp2pService.processValidBlockProposal', async block => ({
|
|
@@ -1038,7 +1307,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1038
1307
|
[Attributes.BLOCK_ARCHIVE]: block.archive.toString(),
|
|
1039
1308
|
[Attributes.P2P_ID]: await block.p2pMessageLoggingIdentifier().then(i => i.toString()),
|
|
1040
1309
|
}))
|
|
1041
|
-
|
|
1310
|
+
protected async processValidBlockProposal(block: BlockProposal, sender: PeerId) {
|
|
1042
1311
|
const slot = block.slotNumber;
|
|
1043
1312
|
this.logger.verbose(`Received block proposal for slot ${slot} from external peer ${sender.toString()}.`, {
|
|
1044
1313
|
p2pMessageIdentifier: await block.p2pMessageLoggingIdentifier(),
|
|
@@ -1046,30 +1315,14 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1046
1315
|
...block.toBlockInfo(),
|
|
1047
1316
|
});
|
|
1048
1317
|
|
|
1049
|
-
//
|
|
1050
|
-
|
|
1051
|
-
await this.mempools.attestationPool.addBlockProposal(block);
|
|
1052
|
-
} catch (err: unknown) {
|
|
1053
|
-
// Drop proposals if we hit per-slot cap in the attestation pool; rethrow unknown errors
|
|
1054
|
-
if (err instanceof ProposalSlotCapExceededError) {
|
|
1055
|
-
this.logger.warn(`Dropping block proposal due to per-slot proposal cap`, {
|
|
1056
|
-
slot: String(slot),
|
|
1057
|
-
archive: block.archive.toString(),
|
|
1058
|
-
error: (err as Error).message,
|
|
1059
|
-
});
|
|
1060
|
-
return;
|
|
1061
|
-
}
|
|
1062
|
-
throw err;
|
|
1063
|
-
}
|
|
1064
|
-
|
|
1065
|
-
// Mark the txs in this proposal as non-evictable
|
|
1066
|
-
await this.mempools.txPool.markTxsAsNonEvictable(block.txHashes);
|
|
1318
|
+
// Mark the txs in this proposal as protected
|
|
1319
|
+
await this.mempools.txPool.protectTxs(block.txHashes, block.blockHeader);
|
|
1067
1320
|
|
|
1068
1321
|
// Call the block received callback to validate the proposal.
|
|
1069
1322
|
// Note: Validators do NOT attest to individual blocks, only to checkpoint proposals.
|
|
1070
1323
|
const isValid = await this.blockReceivedCallback(block, sender);
|
|
1071
1324
|
if (!isValid) {
|
|
1072
|
-
this.logger.
|
|
1325
|
+
this.logger.info(`Block proposal validation failed for block ${block.blockNumber}`, block.toBlockInfo());
|
|
1073
1326
|
}
|
|
1074
1327
|
}
|
|
1075
1328
|
|
|
@@ -1077,66 +1330,165 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1077
1330
|
* Handle a gossiped checkpoint proposal.
|
|
1078
1331
|
* Validates and processes the checkpoint proposal, then triggers the callback for attestation.
|
|
1079
1332
|
*/
|
|
1080
|
-
|
|
1081
|
-
|
|
1082
|
-
|
|
1083
|
-
|
|
1084
|
-
|
|
1085
|
-
|
|
1086
|
-
|
|
1087
|
-
const exists = isValid && (await pool.hasCheckpointProposal(checkpoint));
|
|
1088
|
-
const canAdd = isValid && (await pool.canAddCheckpointProposal(checkpoint));
|
|
1089
|
-
|
|
1090
|
-
this.logger.trace(`Validate propagated checkpoint proposal`, {
|
|
1091
|
-
isValid,
|
|
1092
|
-
exists,
|
|
1093
|
-
canAdd,
|
|
1094
|
-
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
|
|
1095
|
-
[Attributes.P2P_ID]: source.toString(),
|
|
1096
|
-
});
|
|
1097
|
-
|
|
1098
|
-
if (!isValid) {
|
|
1099
|
-
return { result: TopicValidatorResult.Reject };
|
|
1100
|
-
} else if (exists) {
|
|
1101
|
-
return { result: TopicValidatorResult.Ignore, obj: checkpoint };
|
|
1102
|
-
} else if (!canAdd) {
|
|
1103
|
-
this.peerManager.penalizePeer(source, PeerErrorSeverity.MidToleranceError);
|
|
1104
|
-
this.logger.warn(`Penalizing peer for checkpoint proposal exceeding per-slot cap`, {
|
|
1105
|
-
slot: checkpoint.slotNumber.toString(),
|
|
1106
|
-
archive: checkpoint.archive.toString(),
|
|
1107
|
-
source: source.toString(),
|
|
1108
|
-
});
|
|
1109
|
-
return { result: TopicValidatorResult.Reject };
|
|
1110
|
-
} else {
|
|
1111
|
-
return { result: TopicValidatorResult.Accept, obj: checkpoint };
|
|
1112
|
-
}
|
|
1113
|
-
};
|
|
1114
|
-
|
|
1115
|
-
const { result, obj: checkpoint } = await this.validateReceivedMessage<CheckpointProposal>(
|
|
1116
|
-
validationFunc,
|
|
1333
|
+
protected async handleGossipedCheckpointProposal(payloadData: Buffer, msgId: string, source: PeerId): Promise<void> {
|
|
1334
|
+
const {
|
|
1335
|
+
result,
|
|
1336
|
+
obj: checkpoint,
|
|
1337
|
+
metadata: { isEquivocated, processBlock } = {},
|
|
1338
|
+
} = await this.validateReceivedMessage<CheckpointProposal, { isEquivocated: boolean; processBlock: boolean }>(
|
|
1339
|
+
() => this.validateAndStoreCheckpointProposal(source, CheckpointProposal.fromBuffer(payloadData)),
|
|
1117
1340
|
msgId,
|
|
1118
1341
|
source,
|
|
1119
1342
|
TopicType.checkpoint_proposal,
|
|
1120
1343
|
);
|
|
1121
1344
|
|
|
1122
|
-
|
|
1345
|
+
// Process checkpoint proposal if valid and not equivocated.
|
|
1346
|
+
const processCheckpointFn = () =>
|
|
1347
|
+
result === TopicValidatorResult.Accept && checkpoint && !isEquivocated
|
|
1348
|
+
? this.processValidCheckpointProposal(checkpoint.toCore(), source)
|
|
1349
|
+
: Promise.resolve();
|
|
1350
|
+
|
|
1351
|
+
// If the checkpoint contained a valid last block, we process it even if the checkpoint itself is to be rejected
|
|
1352
|
+
// TODO(palla/mbps): Is this ok? Should we be considering a block from a checkpoint that was equivocated?
|
|
1353
|
+
const processBlockFn = () =>
|
|
1354
|
+
processBlock && checkpoint && checkpoint.getBlockProposal()
|
|
1355
|
+
? this.processValidBlockProposal(checkpoint.getBlockProposal()!, source)
|
|
1356
|
+
: Promise.resolve();
|
|
1357
|
+
|
|
1358
|
+
// A node that skips checkpoint validation attests without re-executing the embedded last block, so run
|
|
1359
|
+
// the checkpoint callback first: this creates and broadcasts the attestation before the block is
|
|
1360
|
+
// processed. Otherwise the block's re-execution — which can stall until the re-execution deadline
|
|
1361
|
+
// waiting for a parent that may never arrive — would delay the attestation past the slot's attestation
|
|
1362
|
+
// window, after which peers reject it as stale.
|
|
1363
|
+
if (this.config.skipCheckpointProposalValidation) {
|
|
1364
|
+
await processCheckpointFn();
|
|
1365
|
+
await processBlockFn();
|
|
1123
1366
|
return;
|
|
1124
1367
|
}
|
|
1125
1368
|
|
|
1126
|
-
|
|
1369
|
+
// Process the block first, since it's required for the checkpoint proposal validation.
|
|
1370
|
+
await processBlockFn();
|
|
1371
|
+
await processCheckpointFn();
|
|
1372
|
+
}
|
|
1373
|
+
|
|
1374
|
+
/**
|
|
1375
|
+
* Validates a checkpoint proposal. Penalizes peer if validation fails. Adds the checkpoint and
|
|
1376
|
+
* its last block (if present) to the mempool if valid. Triggers equivocation detection on both.
|
|
1377
|
+
*/
|
|
1378
|
+
@trackSpan('Libp2pService.validateAndStoreCheckpointProposal', (_peerId, checkpoint) => ({
|
|
1379
|
+
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
|
|
1380
|
+
}))
|
|
1381
|
+
protected async validateAndStoreCheckpointProposal(
|
|
1382
|
+
peerId: PeerId,
|
|
1383
|
+
checkpoint: CheckpointProposal,
|
|
1384
|
+
): Promise<ReceivedMessageValidationResult<CheckpointProposal, { isEquivocated: boolean; processBlock: boolean }>> {
|
|
1385
|
+
const validationResult = await this.checkpointProposalValidator.validate(checkpoint);
|
|
1386
|
+
|
|
1387
|
+
if (validationResult.result === 'reject') {
|
|
1388
|
+
this.logger.warn(`Penalizing peer ${peerId} for checkpoint proposal validation failure`);
|
|
1389
|
+
return { result: TopicValidatorResult.Reject, severity: validationResult.severity };
|
|
1390
|
+
}
|
|
1391
|
+
|
|
1392
|
+
if (validationResult.result === 'ignore') {
|
|
1393
|
+
return { result: TopicValidatorResult.Ignore, obj: checkpoint };
|
|
1394
|
+
}
|
|
1395
|
+
|
|
1396
|
+
// Extract and try to add the block proposal first if present
|
|
1397
|
+
const blockProposal = checkpoint.getBlockProposal();
|
|
1398
|
+
let processBlock = false;
|
|
1399
|
+
if (blockProposal) {
|
|
1400
|
+
this.logger.debug(`Validating block proposal from propagated checkpoint`, {
|
|
1401
|
+
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
|
|
1402
|
+
[Attributes.P2P_ID]: peerId.toString(),
|
|
1403
|
+
});
|
|
1404
|
+
const blockProposalResult = await this.validateAndStoreBlockProposal(peerId, blockProposal);
|
|
1405
|
+
const { obj, metadata: { isEquivocated } = {} } = blockProposalResult;
|
|
1406
|
+
if (blockProposalResult.result === TopicValidatorResult.Reject || !obj || isEquivocated) {
|
|
1407
|
+
this.logger.debug(`Rejecting checkpoint due to invalid last block proposal`, {
|
|
1408
|
+
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
|
|
1409
|
+
[Attributes.P2P_ID]: peerId.toString(),
|
|
1410
|
+
isEquivocated,
|
|
1411
|
+
result: blockProposalResult.result,
|
|
1412
|
+
});
|
|
1413
|
+
return {
|
|
1414
|
+
result: TopicValidatorResult.Reject,
|
|
1415
|
+
severity:
|
|
1416
|
+
'severity' in blockProposalResult ? blockProposalResult.severity : PeerErrorSeverity.MidToleranceError,
|
|
1417
|
+
};
|
|
1418
|
+
} else if (blockProposalResult.result === TopicValidatorResult.Accept && obj && !isEquivocated) {
|
|
1419
|
+
processBlock = true;
|
|
1420
|
+
}
|
|
1421
|
+
}
|
|
1422
|
+
|
|
1423
|
+
// Try to add the checkpoint proposal core: this handles existence check, cap check, and adding in one call
|
|
1424
|
+
const checkpointCore = checkpoint.toCore();
|
|
1425
|
+
const tryAddResult = await this.mempools.attestationPool.tryAddCheckpointProposal(checkpointCore);
|
|
1426
|
+
const { added, alreadyExists, count } = tryAddResult;
|
|
1427
|
+
const isEquivocated = count !== undefined && count > 1;
|
|
1428
|
+
|
|
1429
|
+
// Duplicate proposal received, do not re-broadcast
|
|
1430
|
+
if (alreadyExists) {
|
|
1431
|
+
this.logger.debug(`Ignoring duplicate checkpoint proposal received`, {
|
|
1432
|
+
...checkpoint.toCheckpointInfo(),
|
|
1433
|
+
source: peerId.toString(),
|
|
1434
|
+
});
|
|
1435
|
+
return {
|
|
1436
|
+
result: TopicValidatorResult.Ignore,
|
|
1437
|
+
obj: checkpoint,
|
|
1438
|
+
metadata: { isEquivocated, processBlock },
|
|
1439
|
+
};
|
|
1440
|
+
}
|
|
1441
|
+
|
|
1442
|
+
// Too many checkpoint proposals received for this slot, penalize peer and do not re-broadcast
|
|
1443
|
+
// Note: We still return the checkpoint obj so the lastBlock can be processed if valid
|
|
1444
|
+
if (!added) {
|
|
1445
|
+
this.logger.warn(`Penalizing peer for checkpoint proposal exceeding per-slot cap`, {
|
|
1446
|
+
...checkpoint.toCheckpointInfo(),
|
|
1447
|
+
count,
|
|
1448
|
+
source: peerId.toString(),
|
|
1449
|
+
});
|
|
1450
|
+
return {
|
|
1451
|
+
result: TopicValidatorResult.Reject,
|
|
1452
|
+
obj: checkpoint,
|
|
1453
|
+
metadata: { isEquivocated, processBlock },
|
|
1454
|
+
severity: PeerErrorSeverity.HighToleranceError,
|
|
1455
|
+
};
|
|
1456
|
+
}
|
|
1457
|
+
|
|
1458
|
+
// If this was a duplicate proposal, do not process it, but do invoke the duplicate callback,
|
|
1459
|
+
// and do re-broadcast it so other nodes in the network know to slash the proposer
|
|
1460
|
+
if (isEquivocated) {
|
|
1461
|
+
const proposer = checkpoint.getSender();
|
|
1462
|
+
this.logger.warn(`Detected duplicate checkpoint proposal (equivocation) at slot ${checkpoint.slotNumber}`, {
|
|
1463
|
+
...checkpoint.toCheckpointInfo(),
|
|
1464
|
+
source: peerId.toString(),
|
|
1465
|
+
proposer: proposer?.toString(),
|
|
1466
|
+
});
|
|
1467
|
+
// Invoke the duplicate callback on the first duplicate spotted only
|
|
1468
|
+
if (proposer && count === 2) {
|
|
1469
|
+
this.duplicateProposalCallback?.({ slot: checkpoint.slotNumber, proposer, type: 'checkpoint' });
|
|
1470
|
+
}
|
|
1471
|
+
return {
|
|
1472
|
+
result: TopicValidatorResult.Accept,
|
|
1473
|
+
obj: checkpoint,
|
|
1474
|
+
metadata: { isEquivocated, processBlock },
|
|
1475
|
+
};
|
|
1476
|
+
}
|
|
1477
|
+
|
|
1478
|
+
// Otherwise, we're good to go!
|
|
1479
|
+
return { result: TopicValidatorResult.Accept, obj: checkpoint, metadata: { processBlock, isEquivocated } };
|
|
1127
1480
|
}
|
|
1128
1481
|
|
|
1129
1482
|
/**
|
|
1130
1483
|
* Process a validated checkpoint proposal.
|
|
1131
|
-
*
|
|
1132
|
-
* The block callback is invoked before the checkpoint callback.
|
|
1484
|
+
* Note: The proposal was already added to the pool by tryAddCheckpointProposal in handleGossipedCheckpointProposal.
|
|
1133
1485
|
*/
|
|
1134
1486
|
@trackSpan('Libp2pService.processValidCheckpointProposal', async checkpoint => ({
|
|
1135
1487
|
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber,
|
|
1136
1488
|
[Attributes.BLOCK_ARCHIVE]: checkpoint.archive.toString(),
|
|
1137
1489
|
[Attributes.P2P_ID]: await checkpoint.p2pMessageLoggingIdentifier().then(i => i.toString()),
|
|
1138
1490
|
}))
|
|
1139
|
-
|
|
1491
|
+
protected async processValidCheckpointProposal(checkpoint: CheckpointProposalCore, sender: PeerId) {
|
|
1140
1492
|
const slot = checkpoint.slotNumber;
|
|
1141
1493
|
this.logger.verbose(`Received checkpoint proposal for slot ${slot} from external peer ${sender.toString()}.`, {
|
|
1142
1494
|
p2pMessageIdentifier: await checkpoint.p2pMessageLoggingIdentifier(),
|
|
@@ -1145,37 +1497,14 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1145
1497
|
source: sender.toString(),
|
|
1146
1498
|
});
|
|
1147
1499
|
|
|
1148
|
-
|
|
1149
|
-
const blockProposal = checkpoint.getBlockProposal();
|
|
1150
|
-
|
|
1151
|
-
// Add proposal to the pool (this extracts and stores block proposal separately)
|
|
1152
|
-
await this.mempools.attestationPool.addCheckpointProposal(checkpoint);
|
|
1153
|
-
|
|
1154
|
-
// Mark txs as non-evictable if present (from the last block)
|
|
1155
|
-
if (checkpoint.txHashes.length > 0) {
|
|
1156
|
-
await this.mempools.txPool.markTxsAsNonEvictable(checkpoint.txHashes);
|
|
1157
|
-
}
|
|
1158
|
-
|
|
1159
|
-
// If there was a last block proposal, invoke the block callback first for validation.
|
|
1160
|
-
// Note: The block proposal is already stored in the pool by addCheckpointProposal.
|
|
1161
|
-
if (blockProposal) {
|
|
1162
|
-
const isValid = await this.blockReceivedCallback(blockProposal, sender);
|
|
1163
|
-
if (!isValid) {
|
|
1164
|
-
this.logger.warn(`Block proposal from checkpoint failed validation`, {
|
|
1165
|
-
slot: slot.toString(),
|
|
1166
|
-
archive: checkpoint.archive.toString(),
|
|
1167
|
-
blockNumber: blockProposal.blockNumber.toString(),
|
|
1168
|
-
});
|
|
1169
|
-
return;
|
|
1170
|
-
}
|
|
1171
|
-
}
|
|
1500
|
+
await this.allNodesCheckpointReceivedCallback(checkpoint, sender);
|
|
1172
1501
|
|
|
1173
1502
|
// Call the checkpoint received callback with the core version (without lastBlock)
|
|
1174
1503
|
// to validate and potentially generate attestations
|
|
1175
|
-
const attestations = await this.
|
|
1504
|
+
const attestations = await this.validatorCheckpointReceivedCallback(checkpoint, sender);
|
|
1176
1505
|
if (attestations && attestations.length > 0) {
|
|
1177
1506
|
// If the callback returned attestations, add them to the pool and propagate them
|
|
1178
|
-
await this.mempools.attestationPool.
|
|
1507
|
+
await this.mempools.attestationPool.addOwnCheckpointAttestations(attestations);
|
|
1179
1508
|
for (const attestation of attestations) {
|
|
1180
1509
|
await this.propagate(attestation);
|
|
1181
1510
|
}
|
|
@@ -1195,27 +1524,36 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1195
1524
|
}
|
|
1196
1525
|
|
|
1197
1526
|
/**
|
|
1198
|
-
* Validate the requested block transactions
|
|
1527
|
+
* Validate the requested block transactions request-response consistency.
|
|
1528
|
+
* It does NOT validate the transactions themselves.
|
|
1199
1529
|
* @param request - The block transactions request.
|
|
1200
1530
|
* @param response - The block transactions response.
|
|
1201
1531
|
* @param peerId - The ID of the peer that made the request.
|
|
1202
|
-
* @returns True if the
|
|
1532
|
+
* @returns True if the request-response is consistent, false otherwise.
|
|
1203
1533
|
*/
|
|
1204
|
-
@trackSpan('Libp2pService.
|
|
1205
|
-
[Attributes.
|
|
1534
|
+
@trackSpan('Libp2pService.validateRequestedBlockTxsConsistency', request => ({
|
|
1535
|
+
[Attributes.BLOCK_ARCHIVE]: request.archiveRoot.toString(),
|
|
1206
1536
|
}))
|
|
1207
|
-
|
|
1537
|
+
protected async validateRequestedBlockTxsConsistency(
|
|
1208
1538
|
request: BlockTxsRequest,
|
|
1209
1539
|
response: BlockTxsResponse,
|
|
1210
1540
|
peerId: PeerId,
|
|
1211
1541
|
): Promise<boolean> {
|
|
1212
|
-
const requestedTxValidator = this.createRequestedTxValidator();
|
|
1213
|
-
|
|
1214
1542
|
try {
|
|
1215
|
-
|
|
1543
|
+
// A response with archiveRoot=Fr.zero is the documented "I don't have the block" signal from
|
|
1544
|
+
// reqRespBlockTxsHandler (block_txs_handler.ts:54-58): the peer lacked the block in its
|
|
1545
|
+
// attestation pool and archiver, but matched the requested hashes against its tx pool and
|
|
1546
|
+
// shipped what it found. This is legitimate behaviour, not misbehaviour — we just can't verify
|
|
1547
|
+
// membership/order without the block, so we drop the response without penalising the peer.
|
|
1548
|
+
if (response.archiveRoot.isZero()) {
|
|
1549
|
+
this.logger.debug(`Peer ${peerId.toString()} signalled missing block with Fr.zero archive root`);
|
|
1550
|
+
return false;
|
|
1551
|
+
}
|
|
1552
|
+
|
|
1553
|
+
if (!response.archiveRoot.equals(request.archiveRoot)) {
|
|
1216
1554
|
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
|
|
1217
1555
|
throw new ValidationError(
|
|
1218
|
-
`Received block txs for unexpected
|
|
1556
|
+
`Received block txs for unexpected archive root: expected ${request.archiveRoot.toString()}, got ${response.archiveRoot.toString()}`,
|
|
1219
1557
|
);
|
|
1220
1558
|
}
|
|
1221
1559
|
|
|
@@ -1244,18 +1582,26 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1244
1582
|
);
|
|
1245
1583
|
}
|
|
1246
1584
|
|
|
1247
|
-
//
|
|
1248
|
-
|
|
1249
|
-
|
|
1585
|
+
// To verify membership/order of the returned txs we need the canonical tx hash list for the
|
|
1586
|
+
// block. Prefer the block proposal (held while a block is in flight), but fall back to the
|
|
1587
|
+
// archiver for blocks we only know as mined — e.g. a prover collecting txs to prove a block it
|
|
1588
|
+
// never received a proposal for. This mirrors the responder side (reqRespBlockTxsHandler),
|
|
1589
|
+
// which serves from proposal-or-archiver.
|
|
1590
|
+
const proposal = await this.mempools.attestationPool.getBlockProposalByArchive(request.archiveRoot.toString());
|
|
1591
|
+
const blockTxHashes =
|
|
1592
|
+
proposal?.txHashes ??
|
|
1593
|
+
(await this.archiver.getBlock({ archive: request.archiveRoot }))?.body.txEffects.map(e => e.txHash);
|
|
1594
|
+
|
|
1595
|
+
if (blockTxHashes) {
|
|
1250
1596
|
// Build intersected indices
|
|
1251
1597
|
const intersectIdx = request.txIndices.getTrueIndices().filter(i => response.txIndices.isSet(i));
|
|
1252
1598
|
|
|
1253
1599
|
// Enforce subset membership and preserve increasing order by index.
|
|
1254
|
-
const
|
|
1255
|
-
|
|
1600
|
+
const hashToIndexInBlock = new Map<string, number>(
|
|
1601
|
+
blockTxHashes.map((h, i) => [h.toString(), i] as [string, number]),
|
|
1256
1602
|
);
|
|
1257
1603
|
const allowedIndexSet = new Set(intersectIdx);
|
|
1258
|
-
const indices = returnedHashes.map(h =>
|
|
1604
|
+
const indices = returnedHashes.map(h => hashToIndexInBlock.get(h));
|
|
1259
1605
|
const allAllowed = indices.every(idx => idx !== undefined && allowedIndexSet.has(idx));
|
|
1260
1606
|
const strictlyIncreasing = indices.every((idx, i) => (i === 0 ? idx !== undefined : idx! > indices[i - 1]!));
|
|
1261
1607
|
if (!allAllowed || !strictlyIncreasing) {
|
|
@@ -1263,14 +1609,14 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1263
1609
|
throw new ValidationError('Returned txs do not match expected subset/order for requested indices');
|
|
1264
1610
|
}
|
|
1265
1611
|
} else {
|
|
1266
|
-
//
|
|
1612
|
+
// Neither a local proposal nor an archived block: we cannot verify membership/order of the
|
|
1613
|
+
// returned txs. This is a local-state gap, not a peer fault, so we do not penalize.
|
|
1267
1614
|
this.logger.warn(
|
|
1268
|
-
`Block
|
|
1615
|
+
`Block ${request.archiveRoot.toString()} not found in attestation pool or archiver; cannot validate membership/order of returned txs`,
|
|
1269
1616
|
);
|
|
1270
1617
|
return false;
|
|
1271
1618
|
}
|
|
1272
1619
|
|
|
1273
|
-
await Promise.all(response.txs.map(tx => this.validateRequestedTx(tx, peerId, requestedTxValidator)));
|
|
1274
1620
|
return true;
|
|
1275
1621
|
} catch (e: any) {
|
|
1276
1622
|
if (e instanceof ValidationError) {
|
|
@@ -1283,218 +1629,85 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1283
1629
|
}
|
|
1284
1630
|
}
|
|
1285
1631
|
|
|
1286
|
-
|
|
1287
|
-
|
|
1288
|
-
*
|
|
1289
|
-
* The core component of this validator is that each tx hash MUST match the requested tx hash,
|
|
1290
|
-
* In order to perform this check, the tx proof must be verified.
|
|
1291
|
-
*
|
|
1292
|
-
* Note: This function is called from within `ReqResp.sendRequest` as part of the
|
|
1293
|
-
* ReqRespSubProtocol.TX subprotocol validation.
|
|
1294
|
-
*
|
|
1295
|
-
* @param requestedTxHash - The collection of the txs that was requested.
|
|
1296
|
-
* @param responseTx - The collection of txs that was received as a response to the request.
|
|
1297
|
-
* @param peerId - The peer ID of the peer that sent the tx.
|
|
1298
|
-
* @returns True if the whole collection of txs is valid, false otherwise.
|
|
1299
|
-
*/
|
|
1300
|
-
@trackSpan('Libp2pService.validateRequestedTx', (requestedTxHash, _responseTx) => ({
|
|
1301
|
-
[Attributes.TX_HASH]: requestedTxHash.toString(),
|
|
1302
|
-
}))
|
|
1303
|
-
private async validateRequestedTxs(requestedTxHash: TxHash[], responseTx: Tx[], peerId: PeerId): Promise<boolean> {
|
|
1304
|
-
const requested = new Set(requestedTxHash.map(h => h.toString()));
|
|
1305
|
-
const requestedTxValidator = this.createRequestedTxValidator();
|
|
1306
|
-
|
|
1307
|
-
//TODO: (mralj) - this is somewhat naive implementation, if single tx is invlid we consider the whole response invalid.
|
|
1308
|
-
// I think we should still extract the valid txs and return them, so that we can still use the response.
|
|
1309
|
-
try {
|
|
1310
|
-
await Promise.all(responseTx.map(tx => this.validateRequestedTx(tx, peerId, requestedTxValidator, requested)));
|
|
1311
|
-
return true;
|
|
1312
|
-
} catch (e: any) {
|
|
1313
|
-
if (e instanceof ValidationError) {
|
|
1314
|
-
this.logger.warn(`Failed to validate requested txs from peer ${peerId.toString()}, reason ${e.message}`);
|
|
1315
|
-
} else {
|
|
1316
|
-
this.logger.error(`Error during validation of requested txs`, e);
|
|
1317
|
-
}
|
|
1318
|
-
|
|
1319
|
-
return false;
|
|
1320
|
-
}
|
|
1632
|
+
private getGasFees(): Promise<GasFees> {
|
|
1633
|
+
return this.blockMinFeesProvider.getCurrentMinFees();
|
|
1321
1634
|
}
|
|
1322
1635
|
|
|
1323
1636
|
/**
|
|
1324
|
-
*
|
|
1325
|
-
*
|
|
1326
|
-
* If a local copy exists, enforces hash equality. If missing, rejects (no penalty) since the hash cannot be verified.
|
|
1327
|
-
* Penalizes on block number mismatch or hash mismatch.
|
|
1328
|
-
*
|
|
1329
|
-
* @param requestedBlockNumber - The requested block number.
|
|
1330
|
-
* @param responseBlock - The block returned by the peer.
|
|
1331
|
-
* @param peerId - The peer that returned the block.
|
|
1332
|
-
* @returns True if the response is valid, false otherwise.
|
|
1637
|
+
* Get the BatchTxRequesterLibP2PService dependencies for creating BatchTxRequester instances
|
|
1333
1638
|
*/
|
|
1334
|
-
|
|
1335
|
-
|
|
1336
|
-
|
|
1337
|
-
|
|
1338
|
-
|
|
1339
|
-
|
|
1340
|
-
|
|
1341
|
-
|
|
1342
|
-
|
|
1343
|
-
|
|
1344
|
-
|
|
1345
|
-
|
|
1346
|
-
|
|
1347
|
-
}
|
|
1348
|
-
|
|
1349
|
-
const local = await this.archiver.getBlock(BlockNumber(reqNum));
|
|
1350
|
-
if (!local) {
|
|
1351
|
-
// We are missing the local block; we cannot verify the hash yet. Reject without penalizing.
|
|
1352
|
-
// TODO: Consider extending this validator to accept an expected hash or
|
|
1353
|
-
// performing quorum-based checks when using P2P syncing prior to L1 sync.
|
|
1354
|
-
this.logger.warn(`Local block ${reqNum} not found; rejecting BLOCK response without hash verification`);
|
|
1355
|
-
return false;
|
|
1356
|
-
}
|
|
1357
|
-
const [localHash, respHash] = await Promise.all([local.hash(), responseBlock.hash()]);
|
|
1358
|
-
if (!localHash.equals(respHash)) {
|
|
1359
|
-
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
|
|
1360
|
-
return false;
|
|
1361
|
-
}
|
|
1362
|
-
|
|
1363
|
-
return true;
|
|
1364
|
-
} catch (e) {
|
|
1365
|
-
this.logger.warn(`Error validating requested block`, e);
|
|
1366
|
-
return false;
|
|
1367
|
-
}
|
|
1639
|
+
public getBatchTxRequesterService(): BatchTxRequesterLibP2PService {
|
|
1640
|
+
return {
|
|
1641
|
+
reqResp: this.reqresp,
|
|
1642
|
+
connectionSampler: this.reqresp.getConnectionSampler(),
|
|
1643
|
+
txValidatorConfig: {
|
|
1644
|
+
l1ChainId: this.config.l1ChainId,
|
|
1645
|
+
rollupVersion: this.config.rollupVersion,
|
|
1646
|
+
proofVerifier: this.proofVerifier,
|
|
1647
|
+
txValidationCache: this.txValidationCache,
|
|
1648
|
+
},
|
|
1649
|
+
peerScoring: this.peerManager,
|
|
1650
|
+
validateRequestedBlockTxsConsistency: this.validateRequestedBlockTxsConsistency.bind(this),
|
|
1651
|
+
};
|
|
1368
1652
|
}
|
|
1369
1653
|
|
|
1370
|
-
|
|
1371
|
-
|
|
1372
|
-
|
|
1373
|
-
|
|
1374
|
-
|
|
1375
|
-
|
|
1376
|
-
protocolContractsHash,
|
|
1377
|
-
vkTreeRoot: getVKTreeRoot(),
|
|
1378
|
-
}),
|
|
1379
|
-
new TxProofValidator(this.proofVerifier),
|
|
1654
|
+
public async validateTxsReceivedInBlockProposal(txs: Tx[]): Promise<void> {
|
|
1655
|
+
const validator = createTxValidatorForBlockProposalReceivedTxs(
|
|
1656
|
+
this.proofVerifier,
|
|
1657
|
+
{ l1ChainId: this.config.l1ChainId, rollupVersion: this.config.rollupVersion },
|
|
1658
|
+
this.logger.getBindings(),
|
|
1659
|
+
this.txValidationCache,
|
|
1380
1660
|
);
|
|
1381
|
-
}
|
|
1382
|
-
|
|
1383
|
-
private async validateRequestedTx(tx: Tx, peerId: PeerId, txValidator: TxValidator, requested?: Set<`0x${string}`>) {
|
|
1384
|
-
const penalize = (severity: PeerErrorSeverity) => this.peerManager.penalizePeer(peerId, severity);
|
|
1385
|
-
|
|
1386
|
-
if (!(await tx.validateTxHash())) {
|
|
1387
|
-
penalize(PeerErrorSeverity.MidToleranceError);
|
|
1388
|
-
throw new ValidationError(`Received tx with invalid hash ${tx.getTxHash().toString()}.`);
|
|
1389
|
-
}
|
|
1390
|
-
|
|
1391
|
-
if (requested && !requested.has(tx.getTxHash().toString())) {
|
|
1392
|
-
penalize(PeerErrorSeverity.MidToleranceError);
|
|
1393
|
-
throw new ValidationError(`Received tx with hash ${tx.getTxHash().toString()} that was not requested.`);
|
|
1394
|
-
}
|
|
1395
|
-
|
|
1396
|
-
const { result } = await txValidator.validateTx(tx);
|
|
1397
|
-
if (result === 'invalid') {
|
|
1398
|
-
penalize(PeerErrorSeverity.LowToleranceError);
|
|
1399
|
-
throw new ValidationError(`Received tx with hash ${tx.getTxHash().toString()} that is invalid.`);
|
|
1400
|
-
}
|
|
1401
|
-
}
|
|
1402
1661
|
|
|
1403
|
-
|
|
1404
|
-
[Attributes.TX_HASH]: tx.getTxHash().toString(),
|
|
1405
|
-
}))
|
|
1406
|
-
private async validatePropagatedTx(tx: Tx, peerId: PeerId): Promise<boolean> {
|
|
1407
|
-
const currentBlockNumber = await this.archiver.getBlockNumber();
|
|
1408
|
-
|
|
1409
|
-
// We accept transactions if they are not expired by the next slot (checked based on the IncludeByTimestamp field)
|
|
1410
|
-
const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
|
|
1411
|
-
const messageValidators = await this.createMessageValidators(currentBlockNumber, nextSlotTimestamp);
|
|
1412
|
-
|
|
1413
|
-
for (const validator of messageValidators) {
|
|
1414
|
-
const outcome = await this.runValidations(tx, validator);
|
|
1415
|
-
|
|
1416
|
-
if (outcome.allPassed) {
|
|
1417
|
-
continue;
|
|
1418
|
-
}
|
|
1419
|
-
const { name } = outcome.failure;
|
|
1420
|
-
let { severity } = outcome.failure;
|
|
1421
|
-
|
|
1422
|
-
// Double spend validator has a special case handler
|
|
1423
|
-
if (name === 'doubleSpendValidator') {
|
|
1424
|
-
const txBlockNumber = BlockNumber(currentBlockNumber + 1); // tx is expected to be in the next block
|
|
1425
|
-
severity = await this.handleDoubleSpendFailure(tx, txBlockNumber);
|
|
1426
|
-
}
|
|
1427
|
-
|
|
1428
|
-
this.peerManager.penalizePeer(peerId, severity);
|
|
1429
|
-
return false;
|
|
1430
|
-
}
|
|
1431
|
-
return true;
|
|
1432
|
-
}
|
|
1433
|
-
|
|
1434
|
-
private async getGasFees(blockNumber: BlockNumber): Promise<GasFees> {
|
|
1435
|
-
if (blockNumber === this.feesCache?.blockNumber) {
|
|
1436
|
-
return this.feesCache.gasFees;
|
|
1437
|
-
}
|
|
1438
|
-
|
|
1439
|
-
const header = await this.archiver.getBlockHeader(blockNumber);
|
|
1440
|
-
const gasFees = header?.globalVariables.gasFees ?? GasFees.empty();
|
|
1441
|
-
this.feesCache = { blockNumber, gasFees };
|
|
1442
|
-
return gasFees;
|
|
1443
|
-
}
|
|
1444
|
-
|
|
1445
|
-
public async validate(txs: Tx[]): Promise<void> {
|
|
1446
|
-
const currentBlockNumber = await this.archiver.getBlockNumber();
|
|
1447
|
-
|
|
1448
|
-
// We accept transactions if they are not expired by the next slot (checked based on the IncludeByTimestamp field)
|
|
1449
|
-
const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
|
|
1450
|
-
const messageValidators = await this.createMessageValidators(currentBlockNumber, nextSlotTimestamp);
|
|
1451
|
-
|
|
1452
|
-
await Promise.all(
|
|
1662
|
+
const results = await Promise.all(
|
|
1453
1663
|
txs.map(async tx => {
|
|
1454
|
-
|
|
1455
|
-
|
|
1456
|
-
if (!outcome.allPassed) {
|
|
1457
|
-
throw new Error('Invalid tx detected', { cause: { outcome } });
|
|
1458
|
-
}
|
|
1459
|
-
}
|
|
1664
|
+
const result = await validator.validateTx(tx);
|
|
1665
|
+
return result.result !== 'invalid';
|
|
1460
1666
|
}),
|
|
1461
1667
|
);
|
|
1668
|
+
if (results.some(value => value === false)) {
|
|
1669
|
+
throw new Error('Invalid tx detected');
|
|
1670
|
+
}
|
|
1462
1671
|
}
|
|
1463
1672
|
|
|
1464
|
-
/**
|
|
1465
|
-
|
|
1466
|
-
*
|
|
1467
|
-
* Each validator is a pair of a validator and a severity.
|
|
1468
|
-
* If a validator fails, the peer is penalized with the severity of the validator.
|
|
1469
|
-
*
|
|
1470
|
-
* @param currentBlockNumber - The current synced block number.
|
|
1471
|
-
* @param nextSlotTimestamp - The timestamp of the next slot (used to validate txs are not expired).
|
|
1472
|
-
* @returns The message validators.
|
|
1473
|
-
*/
|
|
1474
|
-
private async createMessageValidators(
|
|
1673
|
+
/** Creates the first stage (fast) validators for gossiped transactions. */
|
|
1674
|
+
protected async createFirstStageMessageValidators(
|
|
1475
1675
|
currentBlockNumber: BlockNumber,
|
|
1476
1676
|
nextSlotTimestamp: UInt64,
|
|
1477
|
-
): Promise<Record<string,
|
|
1478
|
-
const gasFees = await this.getGasFees(
|
|
1479
|
-
const allowedInSetup =
|
|
1480
|
-
|
|
1481
|
-
|
|
1482
|
-
|
|
1483
|
-
|
|
1677
|
+
): Promise<Record<string, TransactionValidator>> {
|
|
1678
|
+
const gasFees = await this.getGasFees();
|
|
1679
|
+
const allowedInSetup = [
|
|
1680
|
+
...(await getDefaultAllowedSetupFunctions()),
|
|
1681
|
+
...(this.config.txPublicSetupAllowListExtend ?? []),
|
|
1682
|
+
];
|
|
1683
|
+
const blockNumber = BlockNumber(currentBlockNumber + 1);
|
|
1684
|
+
const l1Constants = await this.archiver.getL1Constants();
|
|
1685
|
+
|
|
1686
|
+
return createFirstStageTxValidationsForGossipedTransactions(
|
|
1484
1687
|
nextSlotTimestamp,
|
|
1485
|
-
|
|
1688
|
+
blockNumber,
|
|
1486
1689
|
this.worldStateSynchronizer,
|
|
1487
1690
|
gasFees,
|
|
1488
1691
|
this.config.l1ChainId,
|
|
1489
1692
|
this.config.rollupVersion,
|
|
1490
1693
|
protocolContractsHash,
|
|
1491
1694
|
this.archiver,
|
|
1492
|
-
this.proofVerifier,
|
|
1493
1695
|
!this.config.disableTransactions,
|
|
1494
1696
|
allowedInSetup,
|
|
1697
|
+
this.logger.getBindings(),
|
|
1698
|
+
{
|
|
1699
|
+
rollupManaLimit: l1Constants.rollupManaLimit,
|
|
1700
|
+
maxBlockL2Gas: this.config.validateMaxL2BlockGas,
|
|
1701
|
+
maxBlockDAGas: this.config.validateMaxDABlockGas,
|
|
1702
|
+
},
|
|
1495
1703
|
);
|
|
1496
1704
|
}
|
|
1497
1705
|
|
|
1706
|
+
/** Creates the second stage (expensive proof verification) validators for gossiped transactions. */
|
|
1707
|
+
protected createSecondStageMessageValidators(): Record<string, TransactionValidator> {
|
|
1708
|
+
return createSecondStageTxValidationsForGossipedTransactions(this.proofVerifier, this.logger.getBindings());
|
|
1709
|
+
}
|
|
1710
|
+
|
|
1498
1711
|
/**
|
|
1499
1712
|
* Run validations on a tx.
|
|
1500
1713
|
* @param tx - The tx to validate.
|
|
@@ -1503,7 +1716,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1503
1716
|
*/
|
|
1504
1717
|
private async runValidations(
|
|
1505
1718
|
tx: Tx,
|
|
1506
|
-
messageValidators: Record<string,
|
|
1719
|
+
messageValidators: Record<string, TransactionValidator>,
|
|
1507
1720
|
): Promise<ValidationOutcome> {
|
|
1508
1721
|
const validationPromises = Object.entries(messageValidators).map(async ([name, { validator, severity }]) => {
|
|
1509
1722
|
const { result } = await validator.validateTx(tx);
|
|
@@ -1512,8 +1725,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1512
1725
|
|
|
1513
1726
|
// A promise that resolves when all validations have been run
|
|
1514
1727
|
const allValidations = await Promise.all(validationPromises);
|
|
1515
|
-
const
|
|
1516
|
-
if (
|
|
1728
|
+
const failures = allValidations.filter(x => !x.isValid);
|
|
1729
|
+
if (failures.length > 0) {
|
|
1730
|
+
// Pick the most severe failure (lowest tolerance = harshest penalty)
|
|
1731
|
+
const failed = maxBy(failures, f => PeerErrorSeverityByHarshness.indexOf(f.severity))!;
|
|
1517
1732
|
return {
|
|
1518
1733
|
allPassed: false,
|
|
1519
1734
|
failure: {
|
|
@@ -1545,15 +1760,18 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1545
1760
|
return PeerErrorSeverity.HighToleranceError;
|
|
1546
1761
|
}
|
|
1547
1762
|
|
|
1548
|
-
const snapshotValidator = new DoubleSpendTxValidator(
|
|
1549
|
-
|
|
1550
|
-
|
|
1551
|
-
|
|
1552
|
-
|
|
1553
|
-
|
|
1554
|
-
|
|
1763
|
+
const snapshotValidator = new DoubleSpendTxValidator(
|
|
1764
|
+
{
|
|
1765
|
+
nullifiersExist: async (nullifiers: Buffer[]) => {
|
|
1766
|
+
const merkleTree = this.worldStateSynchronizer.getSnapshot(
|
|
1767
|
+
BlockNumber(blockNumber - this.config.doubleSpendSeverePeerPenaltyWindow),
|
|
1768
|
+
);
|
|
1769
|
+
const indices = await merkleTree.findLeafIndices(MerkleTreeId.NULLIFIER_TREE, nullifiers);
|
|
1770
|
+
return indices.map(index => index !== undefined);
|
|
1771
|
+
},
|
|
1555
1772
|
},
|
|
1556
|
-
|
|
1773
|
+
this.logger.getBindings(),
|
|
1774
|
+
);
|
|
1557
1775
|
|
|
1558
1776
|
const validSnapshot = await snapshotValidator.validateTx(tx);
|
|
1559
1777
|
if (validSnapshot.result !== 'valid') {
|
|
@@ -1563,68 +1781,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1563
1781
|
return PeerErrorSeverity.HighToleranceError;
|
|
1564
1782
|
}
|
|
1565
1783
|
|
|
1566
|
-
/**
|
|
1567
|
-
* Validate a checkpoint attestation.
|
|
1568
|
-
*
|
|
1569
|
-
* @param attestation - The checkpoint attestation to validate.
|
|
1570
|
-
* @returns True if the checkpoint attestation is valid, false otherwise.
|
|
1571
|
-
*/
|
|
1572
|
-
@trackSpan('Libp2pService.validateCheckpointAttestation', async (_, attestation) => ({
|
|
1573
|
-
[Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber,
|
|
1574
|
-
[Attributes.BLOCK_ARCHIVE]: attestation.archive.toString(),
|
|
1575
|
-
[Attributes.P2P_ID]: await attestation.p2pMessageLoggingIdentifier().then(i => i.toString()),
|
|
1576
|
-
}))
|
|
1577
|
-
public async validateCheckpointAttestation(peerId: PeerId, attestation: CheckpointAttestation): Promise<boolean> {
|
|
1578
|
-
const severity = await this.checkpointAttestationValidator.validate(attestation);
|
|
1579
|
-
if (severity) {
|
|
1580
|
-
this.logger.debug(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
|
|
1581
|
-
this.peerManager.penalizePeer(peerId, severity);
|
|
1582
|
-
return false;
|
|
1583
|
-
}
|
|
1584
|
-
|
|
1585
|
-
return true;
|
|
1586
|
-
}
|
|
1587
|
-
|
|
1588
|
-
/**
|
|
1589
|
-
* Validate a block proposal.
|
|
1590
|
-
*
|
|
1591
|
-
* @param block - The block proposal to validate.
|
|
1592
|
-
* @returns True if the block proposal is valid, false otherwise.
|
|
1593
|
-
*/
|
|
1594
|
-
@trackSpan('Libp2pService.validateBlockProposal', (_peerId, block) => ({
|
|
1595
|
-
[Attributes.SLOT_NUMBER]: block.slotNumber.toString(),
|
|
1596
|
-
}))
|
|
1597
|
-
public async validateBlockProposal(peerId: PeerId, block: BlockProposal): Promise<boolean> {
|
|
1598
|
-
const severity = await this.blockProposalValidator.validate(block);
|
|
1599
|
-
if (severity) {
|
|
1600
|
-
this.logger.debug(`Penalizing peer ${peerId} for block proposal validation failure`);
|
|
1601
|
-
this.peerManager.penalizePeer(peerId, severity);
|
|
1602
|
-
return false;
|
|
1603
|
-
}
|
|
1604
|
-
|
|
1605
|
-
return true;
|
|
1606
|
-
}
|
|
1607
|
-
|
|
1608
|
-
/**
|
|
1609
|
-
* Validate a checkpoint proposal.
|
|
1610
|
-
*
|
|
1611
|
-
* @param checkpoint - The checkpoint proposal to validate.
|
|
1612
|
-
* @returns True if the checkpoint proposal is valid, false otherwise.
|
|
1613
|
-
*/
|
|
1614
|
-
@trackSpan('Libp2pService.validateCheckpointProposal', (_peerId, checkpoint) => ({
|
|
1615
|
-
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
|
|
1616
|
-
}))
|
|
1617
|
-
public async validateCheckpointProposal(peerId: PeerId, checkpoint: CheckpointProposal): Promise<boolean> {
|
|
1618
|
-
const severity = await this.checkpointProposalValidator.validate(checkpoint);
|
|
1619
|
-
if (severity) {
|
|
1620
|
-
this.logger.debug(`Penalizing peer ${peerId} for checkpoint proposal validation failure`);
|
|
1621
|
-
this.peerManager.penalizePeer(peerId, severity);
|
|
1622
|
-
return false;
|
|
1623
|
-
}
|
|
1624
|
-
|
|
1625
|
-
return true;
|
|
1626
|
-
}
|
|
1627
|
-
|
|
1628
1784
|
public getPeerScore(peerId: PeerId): number {
|
|
1629
1785
|
return this.node.services.pubsub.score.score(peerId.toString());
|
|
1630
1786
|
}
|