@aztec/p2p 0.0.1-commit.96dac018d → 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 +1 -1
- package/dest/bootstrap/bootstrap.d.ts.map +1 -1
- package/dest/bootstrap/bootstrap.js +9 -1
- package/dest/client/factory.d.ts +7 -7
- package/dest/client/factory.d.ts.map +1 -1
- package/dest/client/factory.js +34 -16
- package/dest/client/interface.d.ts +17 -8
- package/dest/client/interface.d.ts.map +1 -1
- package/dest/client/p2p_client.d.ts +15 -11
- package/dest/client/p2p_client.d.ts.map +1 -1
- package/dest/client/p2p_client.js +93 -49
- package/dest/config.d.ts +153 -102
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +134 -35
- 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/index.d.ts +1 -2
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +0 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +99 -59
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.js +267 -197
- package/dest/mem_pools/attestation_pool/attestation_pool_test_suite.d.ts +1 -1
- 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 +181 -65
- package/dest/mem_pools/attestation_pool/mocks.d.ts +1 -1
- package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/mocks.js +6 -4
- package/dest/mem_pools/index.d.ts +1 -2
- package/dest/mem_pools/index.d.ts.map +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 +33 -15
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.js +2 -1
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.js +2 -0
- package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts +2 -1
- package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/index.js +1 -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 +7 -1
- package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.js +2 -2
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.js +8 -6
- package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts +2 -2
- package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.js +2 -2
- package/dest/mem_pools/tx_pool_v2/index.d.ts +2 -2
- package/dest/mem_pools/tx_pool_v2/index.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/index.js +1 -1
- package/dest/mem_pools/tx_pool_v2/interfaces.d.ts +12 -5
- package/dest/mem_pools/tx_pool_v2/interfaces.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/interfaces.js +2 -1
- package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts +44 -12
- package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/tx_metadata.js +81 -22
- package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts +1 -1
- package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/tx_pool_indices.js +26 -44
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts +4 -2
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.js +6 -0
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts +2 -1
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.js +133 -110
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts +9 -3
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts.map +1 -1
- package/dest/msg_validators/attestation_validator/attestation_validator.js +37 -12
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts +7 -3
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts.map +1 -1
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.js +4 -5
- package/dest/msg_validators/clock_tolerance.d.ts +12 -1
- package/dest/msg_validators/clock_tolerance.d.ts.map +1 -1
- package/dest/msg_validators/clock_tolerance.js +61 -3
- 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 +90 -44
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.js +8 -15
- 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.js +1 -1
- 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 +2 -1
- package/dest/msg_validators/tx_validator/data_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/data_validator.js +36 -2
- package/dest/msg_validators/tx_validator/factory.d.ts +27 -7
- package/dest/msg_validators/tx_validator/factory.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/factory.js +47 -17
- package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts +1 -1
- package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/fee_payer_balance.js +6 -2
- package/dest/msg_validators/tx_validator/gas_validator.d.ts +48 -7
- package/dest/msg_validators/tx_validator/gas_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/gas_validator.js +88 -41
- package/dest/msg_validators/tx_validator/index.d.ts +4 -1
- package/dest/msg_validators/tx_validator/index.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/index.js +3 -0
- package/dest/msg_validators/tx_validator/metadata_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/metadata_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/metadata_validator.js +4 -4
- package/dest/msg_validators/tx_validator/phases_validator.d.ts +22 -2
- package/dest/msg_validators/tx_validator/phases_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/phases_validator.js +72 -24
- package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts +2 -1
- package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/tx_proof_validator.js +2 -0
- 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 +5 -5
- 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 +35 -8
- package/dest/services/dummy_service.d.ts +11 -15
- package/dest/services/dummy_service.d.ts.map +1 -1
- package/dest/services/dummy_service.js +12 -16
- package/dest/services/encoding.d.ts +6 -2
- package/dest/services/encoding.d.ts.map +1 -1
- package/dest/services/encoding.js +14 -8
- package/dest/services/gossipsub/topic_score_params.d.ts +13 -2
- package/dest/services/gossipsub/topic_score_params.d.ts.map +1 -1
- package/dest/services/gossipsub/topic_score_params.js +21 -4
- 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 +14 -0
- package/dest/services/libp2p/libp2p_service.d.ts +39 -50
- package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
- package/dest/services/libp2p/libp2p_service.js +299 -250
- package/dest/services/peer-manager/metrics.d.ts +3 -1
- package/dest/services/peer-manager/metrics.d.ts.map +1 -1
- package/dest/services/peer-manager/metrics.js +6 -0
- 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 +32 -10
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts +11 -8
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts.map +1 -1
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.js +84 -71
- package/dest/services/reqresp/batch-tx-requester/interface.d.ts +10 -6
- package/dest/services/reqresp/batch-tx-requester/interface.d.ts.map +1 -1
- package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts +5 -4
- package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts.map +1 -1
- package/dest/services/reqresp/batch-tx-requester/missing_txs.js +13 -7
- package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts +3 -1
- package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts.map +1 -1
- package/dest/services/reqresp/batch-tx-requester/peer_collection.js +3 -0
- package/dest/services/reqresp/batch-tx-requester/tx_validator.d.ts +5 -14
- package/dest/services/reqresp/batch-tx-requester/tx_validator.d.ts.map +1 -1
- package/dest/services/reqresp/batch-tx-requester/tx_validator.js +6 -20
- package/dest/services/reqresp/config.d.ts +3 -3
- package/dest/services/reqresp/config.d.ts.map +1 -1
- package/dest/services/reqresp/interface.d.ts +16 -18
- package/dest/services/reqresp/interface.d.ts.map +1 -1
- package/dest/services/reqresp/interface.js +10 -20
- package/dest/services/reqresp/metrics.d.ts +1 -1
- package/dest/services/reqresp/metrics.d.ts.map +1 -1
- package/dest/services/reqresp/metrics.js +0 -1
- package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts +1 -1
- 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 +4 -2
- 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/tx.d.ts +1 -1
- package/dest/services/reqresp/protocols/tx.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/tx.js +1 -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 +7 -29
- package/dest/services/reqresp/reqresp.d.ts.map +1 -1
- package/dest/services/reqresp/reqresp.js +43 -215
- package/dest/services/service.d.ts +9 -12
- package/dest/services/service.d.ts.map +1 -1
- package/dest/services/tx_collection/config.d.ts +2 -23
- package/dest/services/tx_collection/config.d.ts.map +1 -1
- package/dest/services/tx_collection/config.js +2 -55
- package/dest/services/tx_collection/file_store_tx_collection.d.ts +12 -28
- package/dest/services/tx_collection/file_store_tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/file_store_tx_collection.js +43 -83
- package/dest/services/tx_collection/file_store_tx_source.d.ts +5 -4
- package/dest/services/tx_collection/file_store_tx_source.d.ts.map +1 -1
- package/dest/services/tx_collection/file_store_tx_source.js +39 -29
- package/dest/services/tx_collection/index.d.ts +2 -3
- package/dest/services/tx_collection/index.d.ts.map +1 -1
- package/dest/services/tx_collection/index.js +0 -1
- 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 +0 -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 +36 -55
- package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_collection.js +275 -119
- package/dest/services/tx_collection/tx_collection_sink.d.ts +1 -1
- package/dest/services/tx_collection/tx_collection_sink.js +2 -2
- package/dest/services/tx_collection/tx_source.d.ts +6 -5
- package/dest/services/tx_collection/tx_source.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_source.js +9 -7
- package/dest/services/tx_file_store/tx_file_store.d.ts +1 -3
- package/dest/services/tx_file_store/tx_file_store.d.ts.map +1 -1
- package/dest/services/tx_file_store/tx_file_store.js +4 -14
- package/dest/services/tx_provider.d.ts +3 -1
- package/dest/services/tx_provider.d.ts.map +1 -1
- package/dest/services/tx_provider.js +3 -0
- package/dest/test-helpers/make-test-p2p-clients.d.ts +5 -6
- 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 +24 -11
- package/dest/test-helpers/mock-pubsub.d.ts.map +1 -1
- package/dest/test-helpers/mock-pubsub.js +45 -45
- 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 +17 -19
- package/dest/test-helpers/test_tx_provider.d.ts +3 -1
- package/dest/test-helpers/test_tx_provider.d.ts.map +1 -1
- package/dest/test-helpers/test_tx_provider.js +3 -0
- package/dest/test-helpers/testbench-utils.d.ts +12 -14
- package/dest/test-helpers/testbench-utils.d.ts.map +1 -1
- package/dest/test-helpers/testbench-utils.js +42 -15
- package/dest/testbench/p2p_client_testbench_worker.d.ts +3 -5
- package/dest/testbench/p2p_client_testbench_worker.d.ts.map +1 -1
- package/dest/testbench/p2p_client_testbench_worker.js +88 -42
- package/dest/testbench/worker_client_manager.d.ts +12 -6
- package/dest/testbench/worker_client_manager.d.ts.map +1 -1
- package/dest/testbench/worker_client_manager.js +57 -11
- package/dest/util.d.ts +12 -7
- 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 +15 -14
- package/src/bootstrap/bootstrap.ts +9 -1
- package/src/client/factory.ts +65 -21
- package/src/client/interface.ts +18 -20
- package/src/client/p2p_client.ts +108 -82
- package/src/client/test/{tx_proposal_collector/README.md → p2p_client.batch_tx_requester.bench.README.md} +23 -53
- package/src/config.ts +226 -36
- package/src/errors/p2p-service.error.ts +11 -0
- package/src/errors/reqresp.error.ts +0 -25
- package/src/index.ts +0 -1
- package/src/mem_pools/attestation_pool/attestation_pool.ts +318 -242
- package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +204 -68
- package/src/mem_pools/attestation_pool/mocks.ts +13 -8
- package/src/mem_pools/index.ts +0 -3
- package/src/mem_pools/instrumentation.ts +22 -14
- package/src/mem_pools/tx_pool_v2/README.md +9 -1
- package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.ts +2 -1
- package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.ts +3 -0
- package/src/mem_pools/tx_pool_v2/eviction/index.ts +1 -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 +11 -1
- package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.ts +3 -3
- package/src/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.ts +15 -6
- package/src/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.ts +2 -1
- package/src/mem_pools/tx_pool_v2/index.ts +1 -1
- package/src/mem_pools/tx_pool_v2/interfaces.ts +12 -4
- package/src/mem_pools/tx_pool_v2/tx_metadata.ts +121 -27
- package/src/mem_pools/tx_pool_v2/tx_pool_indices.ts +29 -43
- package/src/mem_pools/tx_pool_v2/tx_pool_v2.ts +16 -1
- package/src/mem_pools/tx_pool_v2/tx_pool_v2_impl.ts +141 -114
- package/src/msg_validators/attestation_validator/README.md +49 -0
- package/src/msg_validators/attestation_validator/attestation_validator.ts +41 -9
- package/src/msg_validators/attestation_validator/fisherman_attestation_validator.ts +14 -7
- package/src/msg_validators/clock_tolerance.ts +79 -3
- 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 +114 -47
- package/src/msg_validators/tx_validator/README.md +15 -3
- package/src/msg_validators/tx_validator/aggregate_tx_validator.ts +3 -12
- 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 +1 -1
- 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 +44 -1
- package/src/msg_validators/tx_validator/factory.ts +61 -10
- package/src/msg_validators/tx_validator/fee_payer_balance.ts +6 -2
- package/src/msg_validators/tx_validator/gas_validator.ts +121 -39
- package/src/msg_validators/tx_validator/index.ts +3 -0
- package/src/msg_validators/tx_validator/metadata_validator.ts +12 -4
- package/src/msg_validators/tx_validator/phases_validator.ts +82 -27
- package/src/msg_validators/tx_validator/tx_proof_validator.ts +2 -0
- package/src/msg_validators/tx_validator/tx_validation_cache.ts +102 -0
- package/src/services/data_store.ts +5 -13
- package/src/services/discv5/discV5_service.ts +38 -5
- package/src/services/dummy_service.ts +14 -39
- package/src/services/encoding.ts +14 -7
- package/src/services/gossipsub/topic_score_params.ts +36 -4
- package/src/services/libp2p/instrumentation.ts +14 -0
- package/src/services/libp2p/libp2p_service.ts +326 -287
- package/src/services/peer-manager/metrics.ts +7 -0
- package/src/services/peer-manager/peer_manager.ts +46 -11
- package/src/services/peer-manager/peer_scoring.ts +27 -5
- package/src/services/reqresp/README.md +215 -0
- package/src/services/reqresp/batch-tx-requester/README.md +46 -7
- package/src/services/reqresp/batch-tx-requester/batch_tx_requester.ts +83 -77
- package/src/services/reqresp/batch-tx-requester/interface.ts +13 -5
- package/src/services/reqresp/batch-tx-requester/missing_txs.ts +13 -6
- package/src/services/reqresp/batch-tx-requester/peer_collection.ts +5 -0
- package/src/services/reqresp/batch-tx-requester/tx_validator.ts +12 -25
- package/src/services/reqresp/config.ts +2 -2
- package/src/services/reqresp/interface.ts +21 -47
- package/src/services/reqresp/metrics.ts +0 -1
- package/src/services/reqresp/protocols/block_txs/block_txs_handler.ts +4 -2
- package/src/services/reqresp/protocols/index.ts +0 -1
- package/src/services/reqresp/protocols/tx.ts +1 -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 +48 -261
- package/src/services/service.ts +12 -28
- package/src/services/tx_collection/config.ts +3 -80
- package/src/services/tx_collection/file_store_tx_collection.ts +54 -103
- package/src/services/tx_collection/file_store_tx_source.ts +43 -31
- package/src/services/tx_collection/index.ts +1 -6
- package/src/services/tx_collection/instrumentation.ts +1 -7
- package/src/services/tx_collection/request_tracker.ts +127 -0
- package/src/services/tx_collection/tx_collection.ts +331 -176
- package/src/services/tx_collection/tx_collection_sink.ts +2 -2
- package/src/services/tx_collection/tx_source.ts +8 -7
- package/src/services/tx_file_store/tx_file_store.ts +5 -17
- package/src/services/tx_provider.ts +5 -0
- package/src/test-helpers/make-test-p2p-clients.ts +4 -3
- package/src/test-helpers/mock-pubsub.ts +49 -66
- package/src/test-helpers/reqresp-nodes.ts +15 -28
- package/src/test-helpers/test_tx_provider.ts +5 -0
- package/src/test-helpers/testbench-utils.ts +53 -28
- package/src/testbench/p2p_client_testbench_worker.ts +91 -61
- package/src/testbench/worker_client_manager.ts +72 -25
- package/src/util.ts +33 -18
- package/src/versioning.ts +3 -33
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.d.ts +0 -2
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.d.ts.map +0 -1
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.js +0 -305
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.d.ts +0 -73
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.d.ts.map +0 -1
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.js +0 -8
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts +0 -125
- 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 -596
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts +0 -32
- 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 -112
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts +0 -157
- 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 -52
- package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts +0 -16
- package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.js +0 -122
- 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_mining_rule.js +0 -84
- 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 -78
- 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/eviction/nullifier_conflict_pre_add_rule.d.ts +0 -25
- package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.js +0 -57
- 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 -212
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts +0 -64
- 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 -151
- 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 -54
- package/dest/services/tx_collection/fast_tx_collection.d.ts.map +0 -1
- package/dest/services/tx_collection/fast_tx_collection.js +0 -327
- package/dest/services/tx_collection/missing_txs_tracker.d.ts +0 -32
- package/dest/services/tx_collection/missing_txs_tracker.d.ts.map +0 -1
- package/dest/services/tx_collection/missing_txs_tracker.js +0 -27
- package/dest/services/tx_collection/proposal_tx_collector.d.ts +0 -49
- package/dest/services/tx_collection/proposal_tx_collector.d.ts.map +0 -1
- package/dest/services/tx_collection/proposal_tx_collector.js +0 -50
- package/dest/services/tx_collection/slow_tx_collection.d.ts +0 -57
- package/dest/services/tx_collection/slow_tx_collection.d.ts.map +0 -1
- package/dest/services/tx_collection/slow_tx_collection.js +0 -211
- package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker.ts +0 -346
- package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker_protocol.ts +0 -43
- package/src/mem_pools/tx_pool/README.md +0 -270
- package/src/mem_pools/tx_pool/aztec_kv_tx_pool.ts +0 -746
- package/src/mem_pools/tx_pool/eviction/eviction_manager.ts +0 -132
- package/src/mem_pools/tx_pool/eviction/eviction_strategy.ts +0 -208
- package/src/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.ts +0 -162
- 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 -93
- package/src/mem_pools/tx_pool/eviction/low_priority_eviction_rule.ts +0 -106
- package/src/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.ts +0 -75
- 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 -230
- package/src/services/reqresp/connection-sampler/batch_connection_sampler.ts +0 -161
- package/src/services/reqresp/protocols/block.ts +0 -37
- package/src/services/tx_collection/fast_tx_collection.ts +0 -387
- package/src/services/tx_collection/missing_txs_tracker.ts +0 -52
- package/src/services/tx_collection/proposal_tx_collector.ts +0 -113
- package/src/services/tx_collection/slow_tx_collection.ts +0 -266
|
@@ -1,14 +1,14 @@
|
|
|
1
1
|
import type { EpochCacheInterface } from '@aztec/epoch-cache';
|
|
2
2
|
import { BlockNumber, type SlotNumber } from '@aztec/foundation/branded-types';
|
|
3
|
-
import {
|
|
3
|
+
import { maxBy, merge } from '@aztec/foundation/collection';
|
|
4
4
|
import { type Logger, createLibp2pComponentLogger, createLogger } from '@aztec/foundation/log';
|
|
5
5
|
import { RunningPromise } from '@aztec/foundation/running-promise';
|
|
6
6
|
import { Timer } from '@aztec/foundation/timer';
|
|
7
7
|
import type { AztecAsyncKVStore } from '@aztec/kv-store';
|
|
8
8
|
import { protocolContractsHash } from '@aztec/protocol-contracts';
|
|
9
|
-
import type { EthAddress,
|
|
9
|
+
import type { EthAddress, L2BlockSource } from '@aztec/stdlib/block';
|
|
10
10
|
import type { ContractDataSource } from '@aztec/stdlib/contract';
|
|
11
|
-
import { GasFees } from '@aztec/stdlib/gas';
|
|
11
|
+
import { type BlockMinFeesProvider, GasFees } from '@aztec/stdlib/gas';
|
|
12
12
|
import type { ClientProtocolCircuitVerifier, PeerInfo, WorldStateSynchronizer } from '@aztec/stdlib/interfaces/server';
|
|
13
13
|
import {
|
|
14
14
|
BlockProposal,
|
|
@@ -16,17 +16,16 @@ import {
|
|
|
16
16
|
CheckpointProposal,
|
|
17
17
|
type CheckpointProposalCore,
|
|
18
18
|
type Gossipable,
|
|
19
|
-
P2PClientType,
|
|
20
19
|
P2PMessage,
|
|
21
|
-
type ValidationResult as P2PValidationResult,
|
|
22
20
|
PeerErrorSeverity,
|
|
21
|
+
PeerErrorSeverityByHarshness,
|
|
23
22
|
TopicType,
|
|
24
23
|
createTopicString,
|
|
25
|
-
|
|
24
|
+
getTopicsForConfig,
|
|
26
25
|
metricsTopicStrToLabels,
|
|
27
26
|
} from '@aztec/stdlib/p2p';
|
|
28
27
|
import { MerkleTreeId } from '@aztec/stdlib/trees';
|
|
29
|
-
import { Tx, type
|
|
28
|
+
import { Tx, type TxValidationResult } from '@aztec/stdlib/tx';
|
|
30
29
|
import type { UInt64 } from '@aztec/stdlib/types';
|
|
31
30
|
import { compressComponentVersions } from '@aztec/stdlib/versioning';
|
|
32
31
|
import {
|
|
@@ -51,13 +50,15 @@ import { yamux } from '@chainsafe/libp2p-yamux';
|
|
|
51
50
|
import { bootstrap } from '@libp2p/bootstrap';
|
|
52
51
|
import { identify } from '@libp2p/identify';
|
|
53
52
|
import { type Message, type MultiaddrConnection, type PeerId, TopicValidatorResult } from '@libp2p/interface';
|
|
54
|
-
import type { ConnectionManager } from '@libp2p/interface-internal';
|
|
53
|
+
import type { AddressManager, ConnectionManager } from '@libp2p/interface-internal';
|
|
55
54
|
import { mplex } from '@libp2p/mplex';
|
|
56
55
|
import { tcp } from '@libp2p/tcp';
|
|
56
|
+
import { multiaddr } from '@multiformats/multiaddr';
|
|
57
57
|
import { ENR } from '@nethermindeth/enr';
|
|
58
58
|
import { createLibp2p } from 'libp2p';
|
|
59
59
|
|
|
60
60
|
import type { P2PConfig } from '../../config.js';
|
|
61
|
+
import { CheckpointProposalReceivedCallbackNotRegisteredError } from '../../errors/p2p-service.error.js';
|
|
61
62
|
import type { MemPools } from '../../mem_pools/interface.js';
|
|
62
63
|
import {
|
|
63
64
|
BlockProposalValidator,
|
|
@@ -73,8 +74,8 @@ import {
|
|
|
73
74
|
createFirstStageTxValidationsForGossipedTransactions,
|
|
74
75
|
createSecondStageTxValidationsForGossipedTransactions,
|
|
75
76
|
createTxValidatorForBlockProposalReceivedTxs,
|
|
76
|
-
createTxValidatorForReqResponseReceivedTxs,
|
|
77
77
|
} from '../../msg_validators/tx_validator/factory.js';
|
|
78
|
+
import { TxValidationCache } from '../../msg_validators/tx_validator/tx_validation_cache.js';
|
|
78
79
|
import { GossipSubEvent } from '../../types/index.js';
|
|
79
80
|
import { type PubSubLibp2p, convertToMultiaddr } from '../../util.js';
|
|
80
81
|
import { getVersions } from '../../versioning.js';
|
|
@@ -92,19 +93,15 @@ import {
|
|
|
92
93
|
AuthRequest,
|
|
93
94
|
BlockTxsRequest,
|
|
94
95
|
BlockTxsResponse,
|
|
95
|
-
DEFAULT_SUB_PROTOCOL_VALIDATORS,
|
|
96
96
|
type ReqRespInterface,
|
|
97
97
|
type ReqRespResponse,
|
|
98
98
|
ReqRespSubProtocol,
|
|
99
99
|
type ReqRespSubProtocolHandler,
|
|
100
100
|
type ReqRespSubProtocolHandlers,
|
|
101
|
-
type ReqRespSubProtocolValidators,
|
|
102
101
|
StatusMessage,
|
|
103
|
-
type SubProtocolMap,
|
|
104
102
|
ValidationError,
|
|
105
103
|
pingHandler,
|
|
106
104
|
reqGoodbyeHandler,
|
|
107
|
-
reqRespBlockHandler,
|
|
108
105
|
reqRespBlockTxsHandler,
|
|
109
106
|
reqRespStatusHandler,
|
|
110
107
|
reqRespTxHandler,
|
|
@@ -112,6 +109,7 @@ import {
|
|
|
112
109
|
import { ReqResp } from '../reqresp/reqresp.js';
|
|
113
110
|
import type {
|
|
114
111
|
P2PBlockReceivedCallback,
|
|
112
|
+
P2PCheckpointAttestationCallback,
|
|
115
113
|
P2PCheckpointReceivedCallback,
|
|
116
114
|
P2PDuplicateAttestationCallback,
|
|
117
115
|
P2PService,
|
|
@@ -130,12 +128,12 @@ type ValidationOutcome = { allPassed: true } | { allPassed: false; failure: Vali
|
|
|
130
128
|
// REFACTOR: Unify with the type above
|
|
131
129
|
type ReceivedMessageValidationResult<T, M = undefined> =
|
|
132
130
|
| { obj: T; result: Exclude<TopicValidatorResult, TopicValidatorResult.Reject>; metadata?: M }
|
|
133
|
-
| { obj?: T; result: 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,8 +145,6 @@ 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
|
-
private feesCache: { blockNumber: BlockNumber; gasFees: GasFees } | undefined;
|
|
151
|
-
|
|
152
148
|
/** Callback invoked when a duplicate proposal is detected (triggers slashing). */
|
|
153
149
|
private duplicateProposalCallback?: (info: {
|
|
154
150
|
slot: SlotNumber;
|
|
@@ -159,6 +155,9 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
159
155
|
/** Callback invoked when a duplicate attestation is detected (triggers slashing). */
|
|
160
156
|
private duplicateAttestationCallback?: P2PDuplicateAttestationCallback;
|
|
161
157
|
|
|
158
|
+
/** Callback invoked when a valid checkpoint attestation is accepted into the pool. */
|
|
159
|
+
private checkpointAttestationCallback?: P2PCheckpointAttestationCallback;
|
|
160
|
+
|
|
162
161
|
/**
|
|
163
162
|
* Callback for when a block is received from a peer.
|
|
164
163
|
* @param block - The block received from the peer.
|
|
@@ -171,10 +170,19 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
171
170
|
* @param checkpoint - The checkpoint proposal received from the peer.
|
|
172
171
|
* @returns The attestations for the checkpoint, if any.
|
|
173
172
|
*/
|
|
174
|
-
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;
|
|
175
180
|
|
|
176
181
|
private gossipSubEventHandler: (e: CustomEvent<GossipsubMessage>) => void;
|
|
177
182
|
|
|
183
|
+
private ipChangedHandler?: (ip: string) => void;
|
|
184
|
+
private discoveredP2pIp?: string;
|
|
185
|
+
|
|
178
186
|
private instrumentation: P2PInstrumentation;
|
|
179
187
|
|
|
180
188
|
private telemetry: TelemetryClient;
|
|
@@ -182,7 +190,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
182
190
|
protected logger: Logger;
|
|
183
191
|
|
|
184
192
|
constructor(
|
|
185
|
-
private clientType: T,
|
|
186
193
|
private config: P2PConfig,
|
|
187
194
|
protected node: PubSubLibp2p,
|
|
188
195
|
private peerDiscoveryService: PeerDiscoveryService,
|
|
@@ -193,8 +200,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
193
200
|
private epochCache: EpochCacheInterface,
|
|
194
201
|
private proofVerifier: ClientProtocolCircuitVerifier,
|
|
195
202
|
private worldStateSynchronizer: WorldStateSynchronizer,
|
|
203
|
+
private blockMinFeesProvider: BlockMinFeesProvider,
|
|
196
204
|
telemetry: TelemetryClient,
|
|
197
205
|
logger: Logger = createLogger('p2p:libp2p_service'),
|
|
206
|
+
private txValidationCache?: TxValidationCache,
|
|
198
207
|
) {
|
|
199
208
|
super(telemetry, 'LibP2PService');
|
|
200
209
|
this.telemetry = telemetry;
|
|
@@ -224,36 +233,55 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
224
233
|
this.protocolVersion,
|
|
225
234
|
);
|
|
226
235
|
|
|
227
|
-
|
|
228
|
-
|
|
236
|
+
const p2pPropagationTime = config.attestationPropagationTime;
|
|
237
|
+
const proposalValidatorOpts = {
|
|
229
238
|
txsPermitted: !config.disableTransactions,
|
|
230
|
-
|
|
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
|
+
};
|
|
231
255
|
this.checkpointAttestationValidator = config.fishermanMode
|
|
232
|
-
? new FishermanAttestationValidator(epochCache, mempools.attestationPool, telemetry)
|
|
233
|
-
: new CheckpointAttestationValidator(epochCache);
|
|
256
|
+
? new FishermanAttestationValidator(epochCache, mempools.attestationPool, telemetry, attestationValidatorOpts)
|
|
257
|
+
: new CheckpointAttestationValidator(epochCache, attestationValidatorOpts);
|
|
234
258
|
|
|
235
259
|
this.gossipSubEventHandler = this.handleGossipSubEvent.bind(this);
|
|
236
260
|
|
|
237
261
|
this.blockReceivedCallback = async (block: BlockProposal): Promise<boolean> => {
|
|
238
|
-
this.logger.
|
|
239
|
-
`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.`,
|
|
240
264
|
{ p2pMessageIdentifier: await block.p2pMessageLoggingIdentifier() },
|
|
241
265
|
);
|
|
242
|
-
return
|
|
266
|
+
return true;
|
|
243
267
|
};
|
|
244
268
|
|
|
245
|
-
this.
|
|
246
|
-
|
|
269
|
+
this.allNodesCheckpointReceivedCallback = (
|
|
270
|
+
_checkpoint: CheckpointProposalCore,
|
|
271
|
+
): Promise<CheckpointAttestation[] | undefined> => {
|
|
272
|
+
throw new CheckpointProposalReceivedCallbackNotRegisteredError();
|
|
273
|
+
};
|
|
274
|
+
|
|
275
|
+
this.validatorCheckpointReceivedCallback = (
|
|
276
|
+
_checkpoint: CheckpointProposalCore,
|
|
247
277
|
): Promise<CheckpointAttestation[] | undefined> => {
|
|
248
|
-
this.logger.debug(
|
|
249
|
-
`Handler not yet registered: Checkpoint received callback not set. Received checkpoint for slot ${checkpoint.slotNumber} from peer.`,
|
|
250
|
-
);
|
|
251
278
|
return Promise.resolve(undefined);
|
|
252
279
|
};
|
|
253
280
|
}
|
|
254
281
|
|
|
255
|
-
public updateConfig(config: Partial<P2PReqRespConfig
|
|
282
|
+
public updateConfig(config: Partial<P2PReqRespConfig & Pick<P2PConfig, 'skipIncomingProposals'>>) {
|
|
256
283
|
this.reqresp.updateConfig(config);
|
|
284
|
+
this.config = merge(this.config, config);
|
|
257
285
|
}
|
|
258
286
|
|
|
259
287
|
/**
|
|
@@ -262,8 +290,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
262
290
|
* @param txPool - The transaction pool to be accessed by the service.
|
|
263
291
|
* @returns The new service.
|
|
264
292
|
*/
|
|
265
|
-
public static async new
|
|
266
|
-
clientType: T,
|
|
293
|
+
public static async new(
|
|
267
294
|
config: P2PConfig,
|
|
268
295
|
peerId: PeerId,
|
|
269
296
|
deps: {
|
|
@@ -273,9 +300,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
273
300
|
proofVerifier: ClientProtocolCircuitVerifier;
|
|
274
301
|
worldStateSynchronizer: WorldStateSynchronizer;
|
|
275
302
|
peerStore: AztecAsyncKVStore;
|
|
303
|
+
blockMinFeesProvider: BlockMinFeesProvider;
|
|
276
304
|
telemetry: TelemetryClient;
|
|
277
305
|
logger: Logger;
|
|
278
306
|
packageVersion: string;
|
|
307
|
+
txValidationCache?: TxValidationCache;
|
|
279
308
|
},
|
|
280
309
|
) {
|
|
281
310
|
const {
|
|
@@ -285,9 +314,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
285
314
|
mempools,
|
|
286
315
|
proofVerifier,
|
|
287
316
|
peerStore,
|
|
317
|
+
blockMinFeesProvider,
|
|
288
318
|
telemetry,
|
|
289
319
|
logger,
|
|
290
320
|
packageVersion,
|
|
321
|
+
txValidationCache,
|
|
291
322
|
} = deps;
|
|
292
323
|
const { p2pPort, maxPeerCount, listenAddress } = config;
|
|
293
324
|
const bindAddrTcp = convertToMultiaddr(listenAddress, p2pPort, 'tcp');
|
|
@@ -338,9 +369,12 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
338
369
|
const l1Constants = epochCache.getL1Constants();
|
|
339
370
|
const topicScoreParams = createAllTopicScoreParams(protocolVersion, {
|
|
340
371
|
slotDurationMs: l1Constants.slotDuration * 1000,
|
|
372
|
+
ethereumSlotDuration: l1Constants.ethereumSlotDuration,
|
|
341
373
|
heartbeatIntervalMs: config.gossipsubInterval,
|
|
342
374
|
targetCommitteeSize: l1Constants.targetCommitteeSize,
|
|
343
375
|
blockDurationMs: config.blockDurationMs,
|
|
376
|
+
l1PublishingTime: config.l1PublishingTime,
|
|
377
|
+
p2pPropagationTime: config.attestationPropagationTime,
|
|
344
378
|
expectedBlockProposalsPerSlot: config.expectedBlockProposalsPerSlot,
|
|
345
379
|
});
|
|
346
380
|
|
|
@@ -442,8 +476,9 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
442
476
|
topics: topicScoreParams,
|
|
443
477
|
}),
|
|
444
478
|
}) as (components: GossipSubComponents) => GossipSub,
|
|
445
|
-
components: (components: { connectionManager: ConnectionManager }) => ({
|
|
479
|
+
components: (components: { connectionManager: ConnectionManager; addressManager: AddressManager }) => ({
|
|
446
480
|
connectionManager: components.connectionManager,
|
|
481
|
+
addressManager: components.addressManager,
|
|
447
482
|
}),
|
|
448
483
|
},
|
|
449
484
|
logger: createLibp2pComponentLogger(logger.module, logger.getBindings()),
|
|
@@ -465,6 +500,9 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
465
500
|
epochCache,
|
|
466
501
|
);
|
|
467
502
|
|
|
503
|
+
// Gate req/resp data protocols for unauthenticated peers when p2pAllowOnlyValidators is enabled
|
|
504
|
+
reqresp.setShouldRejectPeer(peerId => peerManager.shouldDisableP2PGossip(peerId));
|
|
505
|
+
|
|
468
506
|
// Configure application-specific scoring for gossipsub.
|
|
469
507
|
// The weight scales app score to align with gossipsub thresholds:
|
|
470
508
|
// - Disconnect (-50) × 10 = -500 = gossipThreshold (stops receiving gossip)
|
|
@@ -475,7 +513,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
475
513
|
peerManager.shouldDisableP2PGossip(peerId) ? -Infinity : peerManager.getPeerScore(peerId);
|
|
476
514
|
|
|
477
515
|
return new LibP2PService(
|
|
478
|
-
clientType,
|
|
479
516
|
config,
|
|
480
517
|
node,
|
|
481
518
|
peerDiscoveryService,
|
|
@@ -486,8 +523,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
486
523
|
epochCache,
|
|
487
524
|
proofVerifier,
|
|
488
525
|
worldStateSynchronizer,
|
|
526
|
+
blockMinFeesProvider,
|
|
489
527
|
telemetry,
|
|
490
528
|
logger,
|
|
529
|
+
txValidationCache,
|
|
491
530
|
);
|
|
492
531
|
}
|
|
493
532
|
|
|
@@ -501,24 +540,21 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
501
540
|
throw new Error('P2P service already started');
|
|
502
541
|
}
|
|
503
542
|
|
|
504
|
-
// Get listen & announce addresses for logging
|
|
505
543
|
const { p2pIp, p2pPort } = this.config;
|
|
506
|
-
if (!p2pIp) {
|
|
507
|
-
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.');
|
|
508
546
|
}
|
|
509
|
-
const announceTcpMultiaddr = convertToMultiaddr(p2pIp, p2pPort, 'tcp');
|
|
547
|
+
const announceTcpMultiaddr = p2pIp ? convertToMultiaddr(p2pIp, p2pPort, 'tcp') : undefined;
|
|
510
548
|
|
|
511
549
|
// Create request response protocol handlers
|
|
512
550
|
const txHandler = reqRespTxHandler(this.mempools);
|
|
513
551
|
const goodbyeHandler = reqGoodbyeHandler(this.peerManager);
|
|
514
|
-
const blockHandler = reqRespBlockHandler(this.archiver);
|
|
515
552
|
const statusHandler = reqRespStatusHandler(this.protocolVersion, this.worldStateSynchronizer, this.logger);
|
|
516
553
|
|
|
517
554
|
const requestResponseHandlers: Partial<ReqRespSubProtocolHandlers> = {
|
|
518
555
|
[ReqRespSubProtocol.PING]: pingHandler,
|
|
519
556
|
[ReqRespSubProtocol.STATUS]: statusHandler.bind(this),
|
|
520
557
|
[ReqRespSubProtocol.GOODBYE]: goodbyeHandler.bind(this),
|
|
521
|
-
[ReqRespSubProtocol.BLOCK]: blockHandler.bind(this),
|
|
522
558
|
};
|
|
523
559
|
|
|
524
560
|
if (!this.config.disableTransactions) {
|
|
@@ -534,22 +570,14 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
534
570
|
requestResponseHandlers[ReqRespSubProtocol.TX] = txHandler.bind(this);
|
|
535
571
|
}
|
|
536
572
|
|
|
537
|
-
// Define the sub protocol validators - This is done within this start() method to gain a callback to the existing validateTx function
|
|
538
|
-
const reqrespSubProtocolValidators = {
|
|
539
|
-
...DEFAULT_SUB_PROTOCOL_VALIDATORS,
|
|
540
|
-
[ReqRespSubProtocol.TX]: this.validateRequestedTxs.bind(this),
|
|
541
|
-
[ReqRespSubProtocol.BLOCK_TXS]: this.validateRequestedBlockTxs.bind(this),
|
|
542
|
-
[ReqRespSubProtocol.BLOCK]: this.validateRequestedBlock.bind(this),
|
|
543
|
-
};
|
|
544
|
-
|
|
545
573
|
await this.peerManager.initializePeers();
|
|
546
574
|
|
|
547
|
-
await this.reqresp.start(requestResponseHandlers
|
|
575
|
+
await this.reqresp.start(requestResponseHandlers);
|
|
548
576
|
|
|
549
577
|
await this.node.start();
|
|
550
578
|
|
|
551
579
|
// Subscribe to standard GossipSub topics by default
|
|
552
|
-
for (const topic of
|
|
580
|
+
for (const topic of getTopicsForConfig(this.config.disableTransactions)) {
|
|
553
581
|
this.subscribeToTopic(this.topicStrings[topic]);
|
|
554
582
|
}
|
|
555
583
|
|
|
@@ -560,6 +588,38 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
560
588
|
if (!this.config.p2pDiscoveryDisabled) {
|
|
561
589
|
await this.peerDiscoveryService.start();
|
|
562
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
|
+
|
|
563
623
|
this.discoveryRunningPromise = new RunningPromise(
|
|
564
624
|
async () => {
|
|
565
625
|
await this.peerManager.heartbeat();
|
|
@@ -585,6 +645,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
585
645
|
// Remove gossip sub listener
|
|
586
646
|
this.node.services.pubsub.removeEventListener(GossipSubEvent.MESSAGE, this.gossipSubEventHandler);
|
|
587
647
|
|
|
648
|
+
if (this.ipChangedHandler) {
|
|
649
|
+
this.peerDiscoveryService.removeListener('ip:changed', this.ipChangedHandler);
|
|
650
|
+
this.ipChangedHandler = undefined;
|
|
651
|
+
}
|
|
652
|
+
|
|
588
653
|
// Stop peer manager
|
|
589
654
|
this.logger.debug('Stopping peer manager...');
|
|
590
655
|
await this.peerManager.stop();
|
|
@@ -599,12 +664,8 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
599
664
|
this.logger.info('LibP2P service stopped');
|
|
600
665
|
}
|
|
601
666
|
|
|
602
|
-
addReqRespSubProtocol(
|
|
603
|
-
subProtocol
|
|
604
|
-
handler: ReqRespSubProtocolHandler,
|
|
605
|
-
validator?: ReqRespSubProtocolValidators[ReqRespSubProtocol],
|
|
606
|
-
): Promise<void> {
|
|
607
|
-
return this.reqresp.addSubProtocol(subProtocol, handler, validator);
|
|
667
|
+
addReqRespSubProtocol(subProtocol: ReqRespSubProtocol, handler: ReqRespSubProtocolHandler): Promise<void> {
|
|
668
|
+
return this.reqresp.addSubProtocol(subProtocol, handler);
|
|
608
669
|
}
|
|
609
670
|
|
|
610
671
|
public registerThisValidatorAddresses(address: EthAddress[]): void {
|
|
@@ -632,20 +693,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
632
693
|
setImmediate(() => void safeJob());
|
|
633
694
|
}
|
|
634
695
|
|
|
635
|
-
/**
|
|
636
|
-
* Send a batch of requests to peers, and return the responses
|
|
637
|
-
* @param protocol - The request response protocol to use
|
|
638
|
-
* @param requests - The requests to send to the peers
|
|
639
|
-
* @returns The responses to the requests
|
|
640
|
-
*/
|
|
641
|
-
sendBatchRequest<SubProtocol extends ReqRespSubProtocol>(
|
|
642
|
-
protocol: SubProtocol,
|
|
643
|
-
requests: InstanceType<SubProtocolMap[SubProtocol]['request']>[],
|
|
644
|
-
pinnedPeerId: PeerId | undefined,
|
|
645
|
-
): Promise<InstanceType<SubProtocolMap[SubProtocol]['response']>[]> {
|
|
646
|
-
return this.reqresp.sendBatchRequest(protocol, requests, pinnedPeerId);
|
|
647
|
-
}
|
|
648
|
-
|
|
649
696
|
public sendRequestToPeer(
|
|
650
697
|
peerId: PeerId,
|
|
651
698
|
subProtocol: ReqRespSubProtocol,
|
|
@@ -667,8 +714,12 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
667
714
|
this.blockReceivedCallback = callback;
|
|
668
715
|
}
|
|
669
716
|
|
|
670
|
-
public
|
|
671
|
-
this.
|
|
717
|
+
public registerValidatorCheckpointReceivedCallback(callback: P2PCheckpointReceivedCallback) {
|
|
718
|
+
this.validatorCheckpointReceivedCallback = callback;
|
|
719
|
+
}
|
|
720
|
+
|
|
721
|
+
public registerAllNodesCheckpointReceivedCallback(callback: P2PCheckpointReceivedCallback) {
|
|
722
|
+
this.allNodesCheckpointReceivedCallback = callback;
|
|
672
723
|
}
|
|
673
724
|
|
|
674
725
|
/**
|
|
@@ -690,6 +741,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
690
741
|
this.duplicateAttestationCallback = callback;
|
|
691
742
|
}
|
|
692
743
|
|
|
744
|
+
public registerCheckpointAttestationCallback(callback: P2PCheckpointAttestationCallback): void {
|
|
745
|
+
this.checkpointAttestationCallback = callback;
|
|
746
|
+
}
|
|
747
|
+
|
|
693
748
|
/**
|
|
694
749
|
* Subscribes to a topic.
|
|
695
750
|
* @param topic - The topic to subscribe to.
|
|
@@ -754,6 +809,9 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
754
809
|
if (!validator || !validator.addMessage(msgId)) {
|
|
755
810
|
this.instrumentation.incMessagePrevalidationStatus(false, topicType);
|
|
756
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
|
+
}
|
|
757
815
|
return { result: false, topicType };
|
|
758
816
|
}
|
|
759
817
|
|
|
@@ -815,12 +873,19 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
815
873
|
|
|
816
874
|
// Process the message, optionally within a linked span for trace propagation
|
|
817
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
|
+
}
|
|
818
885
|
if (msg.topic === this.topicStrings[TopicType.tx]) {
|
|
819
886
|
await this.handleGossipedTx(p2pMessage.payload, msgId, source);
|
|
820
887
|
} else if (msg.topic === this.topicStrings[TopicType.checkpoint_attestation]) {
|
|
821
|
-
|
|
822
|
-
await this.processCheckpointAttestationFromPeer(p2pMessage.payload, msgId, source);
|
|
823
|
-
}
|
|
888
|
+
await this.processCheckpointAttestationFromPeer(p2pMessage.payload, msgId, source);
|
|
824
889
|
} else if (msg.topic === this.topicStrings[TopicType.block_proposal]) {
|
|
825
890
|
await this.processBlockFromPeer(p2pMessage.payload, msgId, source);
|
|
826
891
|
} else if (msg.topic === this.topicStrings[TopicType.checkpoint_proposal]) {
|
|
@@ -882,30 +947,67 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
882
947
|
source: PeerId,
|
|
883
948
|
topicType: TopicType,
|
|
884
949
|
): Promise<ReceivedMessageValidationResult<T, M>> {
|
|
885
|
-
|
|
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
|
+
};
|
|
886
955
|
const timer = new Timer();
|
|
887
956
|
try {
|
|
888
957
|
resultAndObj = await validationFunc();
|
|
889
958
|
} catch (err) {
|
|
890
|
-
this.
|
|
891
|
-
|
|
892
|
-
|
|
893
|
-
|
|
894
|
-
|
|
895
|
-
|
|
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
|
+
);
|
|
896
971
|
}
|
|
897
972
|
|
|
898
973
|
if (resultAndObj.result === TopicValidatorResult.Accept) {
|
|
974
|
+
this.logger.debug(`Message ${topicType} accepted by validator`, { msgId, source: source.toString(), topicType });
|
|
899
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 });
|
|
900
986
|
}
|
|
901
987
|
|
|
902
988
|
this.node.services.pubsub.reportMessageValidationResult(msgId, source.toString(), resultAndObj.result);
|
|
903
989
|
return resultAndObj;
|
|
904
990
|
}
|
|
905
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
|
+
|
|
906
1005
|
protected async handleGossipedTx(payloadData: Buffer, msgId: string, source: PeerId) {
|
|
907
1006
|
const validationFunc: () => Promise<ReceivedMessageValidationResult<Tx>> = async () => {
|
|
908
|
-
const tx = Tx.fromBuffer(payloadData);
|
|
1007
|
+
const tx = this.tryDeserialize(() => Tx.fromBuffer(payloadData), msgId, source);
|
|
1008
|
+
if (!tx) {
|
|
1009
|
+
return { result: TopicValidatorResult.Reject, severity: PeerErrorSeverity.LowToleranceError };
|
|
1010
|
+
}
|
|
909
1011
|
|
|
910
1012
|
const currentBlockNumber = await this.archiver.getBlockNumber();
|
|
911
1013
|
const { ts: nextSlotTimestamp } = this.epochCache.getEpochAndSlotInNextL1Slot();
|
|
@@ -925,13 +1027,20 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
925
1027
|
severity = await this.handleDoubleSpendFailure(tx, txBlockNumber);
|
|
926
1028
|
}
|
|
927
1029
|
|
|
928
|
-
this.
|
|
929
|
-
|
|
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 };
|
|
930
1036
|
}
|
|
931
1037
|
|
|
932
1038
|
// Pool pre-check: see if the pool would accept this tx before doing expensive proof verification
|
|
933
1039
|
const canAdd = await this.mempools.txPool.canAddPendingTx(tx);
|
|
934
1040
|
if (canAdd === 'ignored') {
|
|
1041
|
+
this.logger.verbose(`Ignoring gossiped tx ${tx.getTxHash().toString()}: pool pre-check returned ignored`, {
|
|
1042
|
+
source: source.toString(),
|
|
1043
|
+
});
|
|
935
1044
|
return { result: TopicValidatorResult.Ignore, obj: tx };
|
|
936
1045
|
}
|
|
937
1046
|
|
|
@@ -939,9 +1048,13 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
939
1048
|
const secondStageValidators = this.createSecondStageMessageValidators();
|
|
940
1049
|
const secondStageOutcome = await this.runValidations(tx, secondStageValidators);
|
|
941
1050
|
if (!secondStageOutcome.allPassed) {
|
|
942
|
-
const { severity } = secondStageOutcome.failure;
|
|
943
|
-
this.
|
|
944
|
-
|
|
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 };
|
|
945
1058
|
}
|
|
946
1059
|
|
|
947
1060
|
// Pool add: persist the tx
|
|
@@ -951,7 +1064,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
951
1064
|
const wasAccepted = addResult.accepted.some(h => h.equals(txHash));
|
|
952
1065
|
const wasIgnored = addResult.ignored.some(h => h.equals(txHash));
|
|
953
1066
|
|
|
954
|
-
this.logger.
|
|
1067
|
+
this.logger.verbose(`Validate propagated tx ${txHash.toString()}`, {
|
|
955
1068
|
wasAccepted,
|
|
956
1069
|
wasIgnored,
|
|
957
1070
|
[Attributes.P2P_ID]: source.toString(),
|
|
@@ -962,7 +1075,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
962
1075
|
} else if (wasIgnored) {
|
|
963
1076
|
return { result: TopicValidatorResult.Ignore, obj: tx };
|
|
964
1077
|
} else {
|
|
965
|
-
|
|
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 };
|
|
966
1083
|
}
|
|
967
1084
|
};
|
|
968
1085
|
|
|
@@ -992,7 +1109,16 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
992
1109
|
source: PeerId,
|
|
993
1110
|
): Promise<void> {
|
|
994
1111
|
const { result, obj: attestation } = await this.validateReceivedMessage<CheckpointAttestation>(
|
|
995
|
-
() =>
|
|
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
|
+
},
|
|
996
1122
|
msgId,
|
|
997
1123
|
source,
|
|
998
1124
|
TopicType.checkpoint_attestation,
|
|
@@ -1025,8 +1151,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1025
1151
|
|
|
1026
1152
|
if (validationResult.result === 'reject') {
|
|
1027
1153
|
this.logger.warn(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
|
|
1028
|
-
|
|
1029
|
-
return { result: TopicValidatorResult.Reject };
|
|
1154
|
+
return { result: TopicValidatorResult.Reject, severity: validationResult.severity };
|
|
1030
1155
|
}
|
|
1031
1156
|
|
|
1032
1157
|
if (validationResult.result === 'ignore') {
|
|
@@ -1052,16 +1177,16 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1052
1177
|
return { result: TopicValidatorResult.Ignore, obj: attestation };
|
|
1053
1178
|
}
|
|
1054
1179
|
|
|
1055
|
-
// Could not add (cap reached for signer),
|
|
1180
|
+
// Could not add (cap reached for signer), penalize and do not re-broadcast
|
|
1056
1181
|
if (!added) {
|
|
1057
|
-
this.logger.warn(`
|
|
1182
|
+
this.logger.warn(`Rejecting checkpoint attestation due to cap`, {
|
|
1058
1183
|
slot: slot.toString(),
|
|
1059
1184
|
archive: attestation.archive.toString(),
|
|
1060
1185
|
source: peerId.toString(),
|
|
1061
1186
|
attester: attestation.getSender()?.toString(),
|
|
1062
1187
|
count,
|
|
1063
1188
|
});
|
|
1064
|
-
return { result: TopicValidatorResult.
|
|
1189
|
+
return { result: TopicValidatorResult.Reject, severity: PeerErrorSeverity.HighToleranceError };
|
|
1065
1190
|
}
|
|
1066
1191
|
|
|
1067
1192
|
// Check if this is a duplicate attestation (signer attested to a different proposal at the same slot)
|
|
@@ -1080,6 +1205,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1080
1205
|
}
|
|
1081
1206
|
|
|
1082
1207
|
// Attestation was added successfully - accept it so other nodes can also detect the equivocation
|
|
1208
|
+
this.checkpointAttestationCallback?.(attestation);
|
|
1083
1209
|
return { result: TopicValidatorResult.Accept, obj: attestation };
|
|
1084
1210
|
}
|
|
1085
1211
|
|
|
@@ -1116,8 +1242,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1116
1242
|
|
|
1117
1243
|
if (validationResult.result === 'reject') {
|
|
1118
1244
|
this.logger.warn(`Penalizing peer ${peerId} for block proposal validation failure`);
|
|
1119
|
-
|
|
1120
|
-
return { result: TopicValidatorResult.Reject };
|
|
1245
|
+
return { result: TopicValidatorResult.Reject, severity: validationResult.severity };
|
|
1121
1246
|
}
|
|
1122
1247
|
|
|
1123
1248
|
if (validationResult.result === 'ignore') {
|
|
@@ -1141,7 +1266,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1141
1266
|
|
|
1142
1267
|
// Too many blocks received for this slot and index, penalize peer and do not re-broadcast
|
|
1143
1268
|
if (!added) {
|
|
1144
|
-
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.HighToleranceError);
|
|
1145
1269
|
this.logger.warn(`Penalizing peer for block proposal exceeding per-position cap`, {
|
|
1146
1270
|
...block.toBlockInfo(),
|
|
1147
1271
|
indexWithinCheckpoint: block.indexWithinCheckpoint,
|
|
@@ -1149,7 +1273,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1149
1273
|
proposer: block.getSender()?.toString(),
|
|
1150
1274
|
source: peerId.toString(),
|
|
1151
1275
|
});
|
|
1152
|
-
return {
|
|
1276
|
+
return {
|
|
1277
|
+
result: TopicValidatorResult.Reject,
|
|
1278
|
+
metadata: { isEquivocated },
|
|
1279
|
+
severity: PeerErrorSeverity.HighToleranceError,
|
|
1280
|
+
};
|
|
1153
1281
|
}
|
|
1154
1282
|
|
|
1155
1283
|
// If this was a duplicate proposal, do not process it, but do invoke the duplicate callback,
|
|
@@ -1194,7 +1322,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1194
1322
|
// Note: Validators do NOT attest to individual blocks, only to checkpoint proposals.
|
|
1195
1323
|
const isValid = await this.blockReceivedCallback(block, sender);
|
|
1196
1324
|
if (!isValid) {
|
|
1197
|
-
this.logger.
|
|
1325
|
+
this.logger.info(`Block proposal validation failed for block ${block.blockNumber}`, block.toBlockInfo());
|
|
1198
1326
|
}
|
|
1199
1327
|
}
|
|
1200
1328
|
|
|
@@ -1214,17 +1342,33 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1214
1342
|
TopicType.checkpoint_proposal,
|
|
1215
1343
|
);
|
|
1216
1344
|
|
|
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
|
+
|
|
1217
1351
|
// If the checkpoint contained a valid last block, we process it even if the checkpoint itself is to be rejected
|
|
1218
1352
|
// TODO(palla/mbps): Is this ok? Should we be considering a block from a checkpoint that was equivocated?
|
|
1219
|
-
|
|
1220
|
-
|
|
1221
|
-
|
|
1222
|
-
|
|
1223
|
-
|
|
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();
|
|
1224
1366
|
return;
|
|
1225
1367
|
}
|
|
1226
1368
|
|
|
1227
|
-
|
|
1369
|
+
// Process the block first, since it's required for the checkpoint proposal validation.
|
|
1370
|
+
await processBlockFn();
|
|
1371
|
+
await processCheckpointFn();
|
|
1228
1372
|
}
|
|
1229
1373
|
|
|
1230
1374
|
/**
|
|
@@ -1242,8 +1386,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1242
1386
|
|
|
1243
1387
|
if (validationResult.result === 'reject') {
|
|
1244
1388
|
this.logger.warn(`Penalizing peer ${peerId} for checkpoint proposal validation failure`);
|
|
1245
|
-
|
|
1246
|
-
return { result: TopicValidatorResult.Reject };
|
|
1389
|
+
return { result: TopicValidatorResult.Reject, severity: validationResult.severity };
|
|
1247
1390
|
}
|
|
1248
1391
|
|
|
1249
1392
|
if (validationResult.result === 'ignore') {
|
|
@@ -1258,20 +1401,21 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1258
1401
|
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
|
|
1259
1402
|
[Attributes.P2P_ID]: peerId.toString(),
|
|
1260
1403
|
});
|
|
1261
|
-
const
|
|
1262
|
-
|
|
1263
|
-
|
|
1264
|
-
metadata: { isEquivocated } = {},
|
|
1265
|
-
} = await this.validateAndStoreBlockProposal(peerId, blockProposal);
|
|
1266
|
-
if (result === TopicValidatorResult.Reject || !obj || isEquivocated) {
|
|
1404
|
+
const blockProposalResult = await this.validateAndStoreBlockProposal(peerId, blockProposal);
|
|
1405
|
+
const { obj, metadata: { isEquivocated } = {} } = blockProposalResult;
|
|
1406
|
+
if (blockProposalResult.result === TopicValidatorResult.Reject || !obj || isEquivocated) {
|
|
1267
1407
|
this.logger.debug(`Rejecting checkpoint due to invalid last block proposal`, {
|
|
1268
1408
|
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
|
|
1269
1409
|
[Attributes.P2P_ID]: peerId.toString(),
|
|
1270
1410
|
isEquivocated,
|
|
1271
|
-
result,
|
|
1411
|
+
result: blockProposalResult.result,
|
|
1272
1412
|
});
|
|
1273
|
-
return {
|
|
1274
|
-
|
|
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) {
|
|
1275
1419
|
processBlock = true;
|
|
1276
1420
|
}
|
|
1277
1421
|
}
|
|
@@ -1298,13 +1442,17 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1298
1442
|
// Too many checkpoint proposals received for this slot, penalize peer and do not re-broadcast
|
|
1299
1443
|
// Note: We still return the checkpoint obj so the lastBlock can be processed if valid
|
|
1300
1444
|
if (!added) {
|
|
1301
|
-
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.HighToleranceError);
|
|
1302
1445
|
this.logger.warn(`Penalizing peer for checkpoint proposal exceeding per-slot cap`, {
|
|
1303
1446
|
...checkpoint.toCheckpointInfo(),
|
|
1304
1447
|
count,
|
|
1305
1448
|
source: peerId.toString(),
|
|
1306
1449
|
});
|
|
1307
|
-
return {
|
|
1450
|
+
return {
|
|
1451
|
+
result: TopicValidatorResult.Reject,
|
|
1452
|
+
obj: checkpoint,
|
|
1453
|
+
metadata: { isEquivocated, processBlock },
|
|
1454
|
+
severity: PeerErrorSeverity.HighToleranceError,
|
|
1455
|
+
};
|
|
1308
1456
|
}
|
|
1309
1457
|
|
|
1310
1458
|
// If this was a duplicate proposal, do not process it, but do invoke the duplicate callback,
|
|
@@ -1349,9 +1497,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1349
1497
|
source: sender.toString(),
|
|
1350
1498
|
});
|
|
1351
1499
|
|
|
1500
|
+
await this.allNodesCheckpointReceivedCallback(checkpoint, sender);
|
|
1501
|
+
|
|
1352
1502
|
// Call the checkpoint received callback with the core version (without lastBlock)
|
|
1353
1503
|
// to validate and potentially generate attestations
|
|
1354
|
-
const attestations = await this.
|
|
1504
|
+
const attestations = await this.validatorCheckpointReceivedCallback(checkpoint, sender);
|
|
1355
1505
|
if (attestations && attestations.length > 0) {
|
|
1356
1506
|
// If the callback returned attestations, add them to the pool and propagate them
|
|
1357
1507
|
await this.mempools.attestationPool.addOwnCheckpointAttestations(attestations);
|
|
@@ -1374,23 +1524,32 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1374
1524
|
}
|
|
1375
1525
|
|
|
1376
1526
|
/**
|
|
1377
|
-
* Validate the requested block transactions
|
|
1527
|
+
* Validate the requested block transactions request-response consistency.
|
|
1528
|
+
* It does NOT validate the transactions themselves.
|
|
1378
1529
|
* @param request - The block transactions request.
|
|
1379
1530
|
* @param response - The block transactions response.
|
|
1380
1531
|
* @param peerId - The ID of the peer that made the request.
|
|
1381
|
-
* @returns True if the
|
|
1532
|
+
* @returns True if the request-response is consistent, false otherwise.
|
|
1382
1533
|
*/
|
|
1383
|
-
@trackSpan('Libp2pService.
|
|
1534
|
+
@trackSpan('Libp2pService.validateRequestedBlockTxsConsistency', request => ({
|
|
1384
1535
|
[Attributes.BLOCK_ARCHIVE]: request.archiveRoot.toString(),
|
|
1385
1536
|
}))
|
|
1386
|
-
protected async
|
|
1537
|
+
protected async validateRequestedBlockTxsConsistency(
|
|
1387
1538
|
request: BlockTxsRequest,
|
|
1388
1539
|
response: BlockTxsResponse,
|
|
1389
1540
|
peerId: PeerId,
|
|
1390
1541
|
): Promise<boolean> {
|
|
1391
|
-
const requestedTxValidator = this.createRequestedTxValidator();
|
|
1392
|
-
|
|
1393
1542
|
try {
|
|
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
|
+
|
|
1394
1553
|
if (!response.archiveRoot.equals(request.archiveRoot)) {
|
|
1395
1554
|
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
|
|
1396
1555
|
throw new ValidationError(
|
|
@@ -1423,18 +1582,26 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1423
1582
|
);
|
|
1424
1583
|
}
|
|
1425
1584
|
|
|
1426
|
-
//
|
|
1427
|
-
|
|
1428
|
-
|
|
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) {
|
|
1429
1596
|
// Build intersected indices
|
|
1430
1597
|
const intersectIdx = request.txIndices.getTrueIndices().filter(i => response.txIndices.isSet(i));
|
|
1431
1598
|
|
|
1432
1599
|
// Enforce subset membership and preserve increasing order by index.
|
|
1433
|
-
const
|
|
1434
|
-
|
|
1600
|
+
const hashToIndexInBlock = new Map<string, number>(
|
|
1601
|
+
blockTxHashes.map((h, i) => [h.toString(), i] as [string, number]),
|
|
1435
1602
|
);
|
|
1436
1603
|
const allowedIndexSet = new Set(intersectIdx);
|
|
1437
|
-
const indices = returnedHashes.map(h =>
|
|
1604
|
+
const indices = returnedHashes.map(h => hashToIndexInBlock.get(h));
|
|
1438
1605
|
const allAllowed = indices.every(idx => idx !== undefined && allowedIndexSet.has(idx));
|
|
1439
1606
|
const strictlyIncreasing = indices.every((idx, i) => (i === 0 ? idx !== undefined : idx! > indices[i - 1]!));
|
|
1440
1607
|
if (!allAllowed || !strictlyIncreasing) {
|
|
@@ -1442,14 +1609,14 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1442
1609
|
throw new ValidationError('Returned txs do not match expected subset/order for requested indices');
|
|
1443
1610
|
}
|
|
1444
1611
|
} else {
|
|
1445
|
-
//
|
|
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.
|
|
1446
1614
|
this.logger.warn(
|
|
1447
|
-
`Block
|
|
1615
|
+
`Block ${request.archiveRoot.toString()} not found in attestation pool or archiver; cannot validate membership/order of returned txs`,
|
|
1448
1616
|
);
|
|
1449
1617
|
return false;
|
|
1450
1618
|
}
|
|
1451
1619
|
|
|
1452
|
-
await Promise.all(response.txs.map(tx => this.validateRequestedTx(tx, peerId, requestedTxValidator)));
|
|
1453
1620
|
return true;
|
|
1454
1621
|
} catch (e: any) {
|
|
1455
1622
|
if (e instanceof ValidationError) {
|
|
@@ -1462,125 +1629,8 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1462
1629
|
}
|
|
1463
1630
|
}
|
|
1464
1631
|
|
|
1465
|
-
|
|
1466
|
-
|
|
1467
|
-
*
|
|
1468
|
-
* The core component of this validator is that each tx hash MUST match the requested tx hash,
|
|
1469
|
-
* In order to perform this check, the tx proof must be verified.
|
|
1470
|
-
*
|
|
1471
|
-
* Note: This function is called from within `ReqResp.sendRequest` as part of the
|
|
1472
|
-
* ReqRespSubProtocol.TX subprotocol validation.
|
|
1473
|
-
*
|
|
1474
|
-
* @param requestedTxHash - The collection of the txs that was requested.
|
|
1475
|
-
* @param responseTx - The collection of txs that was received as a response to the request.
|
|
1476
|
-
* @param peerId - The peer ID of the peer that sent the tx.
|
|
1477
|
-
* @returns True if the whole collection of txs is valid, false otherwise.
|
|
1478
|
-
*/
|
|
1479
|
-
@trackSpan('Libp2pService.validateRequestedTx', (requestedTxHash, _responseTx) => ({
|
|
1480
|
-
[Attributes.TX_HASH]: requestedTxHash.toString(),
|
|
1481
|
-
}))
|
|
1482
|
-
private async validateRequestedTxs(requestedTxHash: TxHash[], responseTx: Tx[], peerId: PeerId): Promise<boolean> {
|
|
1483
|
-
const requested = new Set(requestedTxHash.map(h => h.toString()));
|
|
1484
|
-
const requestedTxValidator = this.createRequestedTxValidator();
|
|
1485
|
-
|
|
1486
|
-
//TODO: (mralj) - this is somewhat naive implementation, if single tx is invalid we consider the whole response invalid.
|
|
1487
|
-
// I think we should still extract the valid txs and return them, so that we can still use the response.
|
|
1488
|
-
try {
|
|
1489
|
-
await Promise.all(responseTx.map(tx => this.validateRequestedTx(tx, peerId, requestedTxValidator, requested)));
|
|
1490
|
-
return true;
|
|
1491
|
-
} catch (e: any) {
|
|
1492
|
-
if (e instanceof ValidationError) {
|
|
1493
|
-
this.logger.warn(`Failed to validate requested txs from peer ${peerId.toString()}, reason ${e.message}`);
|
|
1494
|
-
} else {
|
|
1495
|
-
this.logger.error(`Error during validation of requested txs`, e);
|
|
1496
|
-
}
|
|
1497
|
-
|
|
1498
|
-
return false;
|
|
1499
|
-
}
|
|
1500
|
-
}
|
|
1501
|
-
|
|
1502
|
-
/**
|
|
1503
|
-
* Validates a BLOCK response.
|
|
1504
|
-
*
|
|
1505
|
-
* If a local copy exists, enforces hash equality. If missing, rejects (no penalty) since the hash cannot be verified.
|
|
1506
|
-
* Penalizes on block number mismatch or hash mismatch.
|
|
1507
|
-
*
|
|
1508
|
-
* @param requestedBlockNumber - The requested block number.
|
|
1509
|
-
* @param responseBlock - The block returned by the peer.
|
|
1510
|
-
* @param peerId - The peer that returned the block.
|
|
1511
|
-
* @returns True if the response is valid, false otherwise.
|
|
1512
|
-
*/
|
|
1513
|
-
@trackSpan('Libp2pService.validateRequestedBlock', (requestedBlockNumber, _responseBlock) => ({
|
|
1514
|
-
[Attributes.BLOCK_NUMBER]: requestedBlockNumber.toString(),
|
|
1515
|
-
}))
|
|
1516
|
-
protected async validateRequestedBlock(
|
|
1517
|
-
requestedBlockNumber: Fr,
|
|
1518
|
-
responseBlock: L2Block,
|
|
1519
|
-
peerId: PeerId,
|
|
1520
|
-
): Promise<boolean> {
|
|
1521
|
-
try {
|
|
1522
|
-
const reqNum = Number(requestedBlockNumber.toString());
|
|
1523
|
-
if (responseBlock.number !== reqNum) {
|
|
1524
|
-
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.LowToleranceError);
|
|
1525
|
-
return false;
|
|
1526
|
-
}
|
|
1527
|
-
|
|
1528
|
-
const local = await this.archiver.getBlock(BlockNumber(reqNum));
|
|
1529
|
-
if (!local) {
|
|
1530
|
-
// We are missing the local block; we cannot verify the hash yet. Reject without penalizing.
|
|
1531
|
-
// TODO: Consider extending this validator to accept an expected hash or
|
|
1532
|
-
// performing quorum-based checks when using P2P syncing prior to L1 sync.
|
|
1533
|
-
this.logger.warn(`Local block ${reqNum} not found; rejecting BLOCK response without hash verification`);
|
|
1534
|
-
return false;
|
|
1535
|
-
}
|
|
1536
|
-
const [localHash, respHash] = await Promise.all([local.hash(), responseBlock.hash()]);
|
|
1537
|
-
if (!localHash.equals(respHash)) {
|
|
1538
|
-
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
|
|
1539
|
-
return false;
|
|
1540
|
-
}
|
|
1541
|
-
|
|
1542
|
-
return true;
|
|
1543
|
-
} catch (e) {
|
|
1544
|
-
this.logger.warn(`Error validating requested block`, e);
|
|
1545
|
-
return false;
|
|
1546
|
-
}
|
|
1547
|
-
}
|
|
1548
|
-
|
|
1549
|
-
protected async validateRequestedTx(
|
|
1550
|
-
tx: Tx,
|
|
1551
|
-
peerId: PeerId,
|
|
1552
|
-
txValidator: TxValidator,
|
|
1553
|
-
requested?: Set<`0x${string}`>,
|
|
1554
|
-
) {
|
|
1555
|
-
const penalize = (severity: PeerErrorSeverity) => this.peerManager.penalizePeer(peerId, severity);
|
|
1556
|
-
if (requested && !requested.has(tx.getTxHash().toString())) {
|
|
1557
|
-
penalize(PeerErrorSeverity.MidToleranceError);
|
|
1558
|
-
throw new ValidationError(`Received tx with hash ${tx.getTxHash().toString()} that was not requested.`);
|
|
1559
|
-
}
|
|
1560
|
-
|
|
1561
|
-
const { result } = await txValidator.validateTx(tx);
|
|
1562
|
-
if (result === 'invalid') {
|
|
1563
|
-
penalize(PeerErrorSeverity.LowToleranceError);
|
|
1564
|
-
throw new ValidationError(`Received tx with hash ${tx.getTxHash().toString()} that is invalid.`);
|
|
1565
|
-
}
|
|
1566
|
-
}
|
|
1567
|
-
|
|
1568
|
-
protected createRequestedTxValidator(): TxValidator {
|
|
1569
|
-
return createTxValidatorForReqResponseReceivedTxs(this.proofVerifier, {
|
|
1570
|
-
l1ChainId: this.config.l1ChainId,
|
|
1571
|
-
rollupVersion: this.config.rollupVersion,
|
|
1572
|
-
});
|
|
1573
|
-
}
|
|
1574
|
-
|
|
1575
|
-
private async getGasFees(blockNumber: BlockNumber): Promise<GasFees> {
|
|
1576
|
-
if (blockNumber === this.feesCache?.blockNumber) {
|
|
1577
|
-
return this.feesCache.gasFees;
|
|
1578
|
-
}
|
|
1579
|
-
|
|
1580
|
-
const header = await this.archiver.getBlockHeader(blockNumber);
|
|
1581
|
-
const gasFees = header?.globalVariables.gasFees ?? GasFees.empty();
|
|
1582
|
-
this.feesCache = { blockNumber, gasFees };
|
|
1583
|
-
return gasFees;
|
|
1632
|
+
private getGasFees(): Promise<GasFees> {
|
|
1633
|
+
return this.blockMinFeesProvider.getCurrentMinFees();
|
|
1584
1634
|
}
|
|
1585
1635
|
|
|
1586
1636
|
/**
|
|
@@ -1594,8 +1644,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1594
1644
|
l1ChainId: this.config.l1ChainId,
|
|
1595
1645
|
rollupVersion: this.config.rollupVersion,
|
|
1596
1646
|
proofVerifier: this.proofVerifier,
|
|
1647
|
+
txValidationCache: this.txValidationCache,
|
|
1597
1648
|
},
|
|
1598
1649
|
peerScoring: this.peerManager,
|
|
1650
|
+
validateRequestedBlockTxsConsistency: this.validateRequestedBlockTxsConsistency.bind(this),
|
|
1599
1651
|
};
|
|
1600
1652
|
}
|
|
1601
1653
|
|
|
@@ -1604,6 +1656,7 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1604
1656
|
this.proofVerifier,
|
|
1605
1657
|
{ l1ChainId: this.config.l1ChainId, rollupVersion: this.config.rollupVersion },
|
|
1606
1658
|
this.logger.getBindings(),
|
|
1659
|
+
this.txValidationCache,
|
|
1607
1660
|
);
|
|
1608
1661
|
|
|
1609
1662
|
const results = await Promise.all(
|
|
@@ -1622,9 +1675,13 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1622
1675
|
currentBlockNumber: BlockNumber,
|
|
1623
1676
|
nextSlotTimestamp: UInt64,
|
|
1624
1677
|
): Promise<Record<string, TransactionValidator>> {
|
|
1625
|
-
const gasFees = await this.getGasFees(
|
|
1626
|
-
const allowedInSetup =
|
|
1678
|
+
const gasFees = await this.getGasFees();
|
|
1679
|
+
const allowedInSetup = [
|
|
1680
|
+
...(await getDefaultAllowedSetupFunctions()),
|
|
1681
|
+
...(this.config.txPublicSetupAllowListExtend ?? []),
|
|
1682
|
+
];
|
|
1627
1683
|
const blockNumber = BlockNumber(currentBlockNumber + 1);
|
|
1684
|
+
const l1Constants = await this.archiver.getL1Constants();
|
|
1628
1685
|
|
|
1629
1686
|
return createFirstStageTxValidationsForGossipedTransactions(
|
|
1630
1687
|
nextSlotTimestamp,
|
|
@@ -1638,6 +1695,11 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1638
1695
|
!this.config.disableTransactions,
|
|
1639
1696
|
allowedInSetup,
|
|
1640
1697
|
this.logger.getBindings(),
|
|
1698
|
+
{
|
|
1699
|
+
rollupManaLimit: l1Constants.rollupManaLimit,
|
|
1700
|
+
maxBlockL2Gas: this.config.validateMaxL2BlockGas,
|
|
1701
|
+
maxBlockDAGas: this.config.validateMaxDABlockGas,
|
|
1702
|
+
},
|
|
1641
1703
|
);
|
|
1642
1704
|
}
|
|
1643
1705
|
|
|
@@ -1663,8 +1725,10 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1663
1725
|
|
|
1664
1726
|
// A promise that resolves when all validations have been run
|
|
1665
1727
|
const allValidations = await Promise.all(validationPromises);
|
|
1666
|
-
const
|
|
1667
|
-
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))!;
|
|
1668
1732
|
return {
|
|
1669
1733
|
allPassed: false,
|
|
1670
1734
|
failure: {
|
|
@@ -1717,31 +1781,6 @@ export class LibP2PService<T extends P2PClientType = P2PClientType.Full> extends
|
|
|
1717
1781
|
return PeerErrorSeverity.HighToleranceError;
|
|
1718
1782
|
}
|
|
1719
1783
|
|
|
1720
|
-
/**
|
|
1721
|
-
* Validate a checkpoint attestation.
|
|
1722
|
-
*
|
|
1723
|
-
* @param attestation - The checkpoint attestation to validate.
|
|
1724
|
-
* @returns True if the checkpoint attestation is valid, false otherwise.
|
|
1725
|
-
*/
|
|
1726
|
-
@trackSpan('Libp2pService.validateCheckpointAttestation', async (_, attestation) => ({
|
|
1727
|
-
[Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber,
|
|
1728
|
-
[Attributes.BLOCK_ARCHIVE]: attestation.archive.toString(),
|
|
1729
|
-
[Attributes.P2P_ID]: await attestation.p2pMessageLoggingIdentifier().then(i => i.toString()),
|
|
1730
|
-
}))
|
|
1731
|
-
public async validateCheckpointAttestation(
|
|
1732
|
-
peerId: PeerId,
|
|
1733
|
-
attestation: CheckpointAttestation,
|
|
1734
|
-
): Promise<P2PValidationResult> {
|
|
1735
|
-
const result = await this.checkpointAttestationValidator.validate(attestation);
|
|
1736
|
-
|
|
1737
|
-
if (result.result === 'reject') {
|
|
1738
|
-
this.logger.warn(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
|
|
1739
|
-
this.peerManager.penalizePeer(peerId, result.severity);
|
|
1740
|
-
}
|
|
1741
|
-
|
|
1742
|
-
return result;
|
|
1743
|
-
}
|
|
1744
|
-
|
|
1745
1784
|
public getPeerScore(peerId: PeerId): number {
|
|
1746
1785
|
return this.node.services.pubsub.score.score(peerId.toString());
|
|
1747
1786
|
}
|