@aztec/p2p 0.0.1-commit.b655e406 → 0.0.1-commit.d1f2d6c
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dest/bootstrap/bootstrap.d.ts +1 -1
- package/dest/bootstrap/bootstrap.d.ts.map +1 -1
- package/dest/client/factory.d.ts +2 -2
- package/dest/client/factory.d.ts.map +1 -1
- package/dest/client/factory.js +2 -3
- package/dest/client/index.d.ts +1 -1
- package/dest/client/interface.d.ts +18 -3
- package/dest/client/interface.d.ts.map +1 -1
- package/dest/client/p2p_client.d.ts +16 -37
- package/dest/client/p2p_client.d.ts.map +1 -1
- package/dest/client/p2p_client.js +464 -126
- package/dest/config.d.ts +62 -59
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +21 -14
- package/dest/enr/generate-enr.d.ts +1 -1
- package/dest/enr/index.d.ts +1 -1
- package/dest/errors/attestation-pool.error.d.ts +7 -0
- package/dest/errors/attestation-pool.error.d.ts.map +1 -0
- package/dest/errors/attestation-pool.error.js +12 -0
- package/dest/errors/reqresp.error.d.ts +1 -1
- package/dest/errors/reqresp.error.d.ts.map +1 -1
- package/dest/index.d.ts +1 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +77 -36
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -1
- 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 +241 -266
- package/dest/mem_pools/attestation_pool/index.d.ts +1 -1
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts +24 -14
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.js +123 -95
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts +18 -12
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.js +91 -108
- package/dest/mem_pools/attestation_pool/mocks.d.ts +234 -10
- package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/mocks.js +17 -13
- package/dest/mem_pools/index.d.ts +1 -1
- package/dest/mem_pools/instrumentation.d.ts +9 -1
- package/dest/mem_pools/instrumentation.d.ts.map +1 -1
- package/dest/mem_pools/instrumentation.js +36 -9
- package/dest/mem_pools/interface.d.ts +3 -4
- package/dest/mem_pools/interface.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts +33 -58
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.js +314 -335
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts +32 -0
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool/eviction/eviction_manager.js +112 -0
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts +157 -0
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool/eviction/eviction_strategy.js +52 -0
- package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts +16 -0
- package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.js +122 -0
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts +17 -0
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.js +84 -0
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts +19 -0
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.js +78 -0
- package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts +26 -0
- package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool/eviction/low_priority_eviction_rule.js +84 -0
- package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.d.ts +25 -0
- package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.js +57 -0
- package/dest/mem_pools/tx_pool/index.d.ts +1 -2
- package/dest/mem_pools/tx_pool/index.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/index.js +0 -1
- package/dest/mem_pools/tx_pool/priority.d.ts +5 -1
- package/dest/mem_pools/tx_pool/priority.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/priority.js +6 -1
- package/dest/mem_pools/tx_pool/tx_pool.d.ts +11 -6
- package/dest/mem_pools/tx_pool/tx_pool.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/tx_pool_test_suite.d.ts +1 -1
- package/dest/mem_pools/tx_pool/tx_pool_test_suite.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/tx_pool_test_suite.js +30 -24
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts +7 -6
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts.map +1 -1
- package/dest/msg_validators/attestation_validator/attestation_validator.js +57 -24
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts +20 -0
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.d.ts.map +1 -0
- package/dest/msg_validators/attestation_validator/fisherman_attestation_validator.js +71 -0
- package/dest/msg_validators/attestation_validator/index.d.ts +2 -1
- package/dest/msg_validators/attestation_validator/index.d.ts.map +1 -1
- package/dest/msg_validators/attestation_validator/index.js +1 -0
- package/dest/msg_validators/clock_tolerance.d.ts +21 -0
- package/dest/msg_validators/clock_tolerance.d.ts.map +1 -0
- package/dest/msg_validators/clock_tolerance.js +37 -0
- package/dest/msg_validators/index.d.ts +2 -2
- package/dest/msg_validators/index.d.ts.map +1 -1
- package/dest/msg_validators/index.js +1 -1
- package/dest/msg_validators/msg_seen_validator/msg_seen_validator.d.ts +1 -1
- package/dest/msg_validators/msg_seen_validator/msg_seen_validator.d.ts.map +1 -1
- package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts +9 -0
- package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/block_proposal_validator.js +6 -0
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts +9 -0
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.js +6 -0
- package/dest/msg_validators/proposal_validator/index.d.ts +4 -0
- package/dest/msg_validators/proposal_validator/index.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/index.js +3 -0
- package/dest/msg_validators/proposal_validator/proposal_validator.d.ts +13 -0
- package/dest/msg_validators/proposal_validator/proposal_validator.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/proposal_validator.js +104 -0
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts +23 -0
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts.map +1 -0
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.js +212 -0
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/allowed_public_setup.d.ts +1 -1
- package/dest/msg_validators/tx_validator/archive_cache.d.ts +2 -2
- package/dest/msg_validators/tx_validator/archive_cache.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/block_header_validator.d.ts +2 -2
- package/dest/msg_validators/tx_validator/block_header_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/data_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/data_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/double_spend_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/double_spend_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/factory.d.ts +4 -3
- package/dest/msg_validators/tx_validator/factory.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/factory.js +1 -1
- package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts +10 -0
- package/dest/msg_validators/tx_validator/fee_payer_balance.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/fee_payer_balance.js +20 -0
- package/dest/msg_validators/tx_validator/gas_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/gas_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/gas_validator.js +8 -14
- package/dest/msg_validators/tx_validator/index.d.ts +2 -1
- package/dest/msg_validators/tx_validator/index.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/index.js +1 -0
- package/dest/msg_validators/tx_validator/metadata_validator.d.ts +2 -2
- package/dest/msg_validators/tx_validator/metadata_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/phases_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/phases_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/phases_validator.js +3 -1
- package/dest/msg_validators/tx_validator/size_validator.d.ts +6 -0
- package/dest/msg_validators/tx_validator/size_validator.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/size_validator.js +20 -0
- package/dest/msg_validators/tx_validator/test_utils.d.ts +2 -2
- package/dest/msg_validators/tx_validator/test_utils.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/timestamp_validator.d.ts +3 -2
- package/dest/msg_validators/tx_validator/timestamp_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/tx_permitted_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/tx_permitted_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts.map +1 -1
- package/dest/services/data_store.d.ts +1 -1
- package/dest/services/data_store.d.ts.map +1 -1
- package/dest/services/discv5/discV5_service.d.ts +1 -1
- package/dest/services/discv5/discV5_service.d.ts.map +1 -1
- package/dest/services/dummy_service.d.ts +6 -2
- package/dest/services/dummy_service.d.ts.map +1 -1
- package/dest/services/dummy_service.js +3 -0
- package/dest/services/encoding.d.ts +1 -1
- package/dest/services/encoding.d.ts.map +1 -1
- package/dest/services/encoding.js +7 -6
- package/dest/services/gossipsub/scoring.d.ts +1 -1
- package/dest/services/index.d.ts +1 -1
- 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 +26 -72
- package/dest/services/libp2p/libp2p_service.d.ts +41 -80
- package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
- package/dest/services/libp2p/libp2p_service.js +899 -175
- package/dest/services/peer-manager/interface.d.ts +1 -1
- package/dest/services/peer-manager/metrics.d.ts +8 -1
- package/dest/services/peer-manager/metrics.d.ts.map +1 -1
- package/dest/services/peer-manager/metrics.js +24 -16
- package/dest/services/peer-manager/peer_manager.d.ts +2 -33
- package/dest/services/peer-manager/peer_manager.d.ts.map +1 -1
- package/dest/services/peer-manager/peer_manager.js +6 -12
- 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 +37 -2
- package/dest/services/reqresp/config.d.ts +1 -1
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts +1 -1
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts.map +1 -1
- package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts +1 -4
- package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts.map +1 -1
- 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/index.d.ts +1 -1
- package/dest/services/reqresp/interface.d.ts +2 -2
- package/dest/services/reqresp/interface.d.ts.map +1 -1
- package/dest/services/reqresp/interface.js +1 -1
- 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 +5 -21
- package/dest/services/reqresp/protocols/auth.d.ts +2 -2
- package/dest/services/reqresp/protocols/auth.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/auth.js +2 -2
- package/dest/services/reqresp/protocols/block.d.ts +1 -1
- package/dest/services/reqresp/protocols/block.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/block.js +3 -2
- package/dest/services/reqresp/protocols/block_txs/bitvector.d.ts +1 -1
- package/dest/services/reqresp/protocols/block_txs/bitvector.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/block_txs/bitvector.js +7 -0
- package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts +1 -1
- package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts +4 -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 +1 -1
- package/dest/services/reqresp/protocols/block_txs/index.d.ts +1 -1
- package/dest/services/reqresp/protocols/goodbye.d.ts +1 -1
- package/dest/services/reqresp/protocols/goodbye.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/index.d.ts +1 -1
- package/dest/services/reqresp/protocols/ping.d.ts +1 -1
- package/dest/services/reqresp/protocols/status.d.ts +6 -5
- package/dest/services/reqresp/protocols/status.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/status.js +7 -3
- package/dest/services/reqresp/protocols/tx.d.ts +2 -3
- package/dest/services/reqresp/protocols/tx.d.ts.map +1 -1
- package/dest/services/reqresp/rate-limiter/index.d.ts +1 -1
- package/dest/services/reqresp/rate-limiter/rate_limiter.d.ts +2 -2
- package/dest/services/reqresp/rate-limiter/rate_limiter.d.ts.map +1 -1
- package/dest/services/reqresp/rate-limiter/rate_limits.d.ts +1 -1
- package/dest/services/reqresp/reqresp.d.ts +1 -41
- package/dest/services/reqresp/reqresp.d.ts.map +1 -1
- package/dest/services/reqresp/reqresp.js +402 -24
- package/dest/services/reqresp/status.d.ts +2 -2
- package/dest/services/reqresp/status.d.ts.map +1 -1
- package/dest/services/service.d.ts +16 -3
- package/dest/services/service.d.ts.map +1 -1
- package/dest/services/tx_collection/config.d.ts +1 -1
- package/dest/services/tx_collection/config.js +1 -1
- package/dest/services/tx_collection/fast_tx_collection.d.ts +4 -9
- package/dest/services/tx_collection/fast_tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/index.d.ts +1 -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 +4 -14
- package/dest/services/tx_collection/slow_tx_collection.d.ts +4 -5
- package/dest/services/tx_collection/slow_tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/slow_tx_collection.js +2 -1
- package/dest/services/tx_collection/tx_collection.d.ts +7 -7
- package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_collection_sink.d.ts +3 -3
- package/dest/services/tx_collection/tx_collection_sink.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_source.d.ts +1 -1
- package/dest/services/tx_collection/tx_source.d.ts.map +1 -1
- package/dest/services/tx_provider.d.ts +4 -2
- package/dest/services/tx_provider.d.ts.map +1 -1
- package/dest/services/tx_provider.js +11 -2
- package/dest/services/tx_provider_instrumentation.d.ts +5 -2
- package/dest/services/tx_provider_instrumentation.d.ts.map +1 -1
- package/dest/services/tx_provider_instrumentation.js +13 -13
- package/dest/test-helpers/generate-peer-id-private-keys.d.ts +1 -1
- package/dest/test-helpers/get-ports.d.ts +1 -1
- package/dest/test-helpers/get-ports.d.ts.map +1 -1
- package/dest/test-helpers/index.d.ts +1 -1
- package/dest/test-helpers/make-enrs.d.ts +1 -1
- package/dest/test-helpers/make-test-p2p-clients.d.ts +2 -2
- package/dest/test-helpers/make-test-p2p-clients.d.ts.map +1 -1
- package/dest/test-helpers/mock-pubsub.d.ts +4 -4
- package/dest/test-helpers/mock-pubsub.d.ts.map +1 -1
- package/dest/test-helpers/mock-tx-helpers.d.ts +2 -2
- package/dest/test-helpers/mock-tx-helpers.d.ts.map +1 -1
- package/dest/test-helpers/mock-tx-helpers.js +1 -1
- package/dest/test-helpers/reqresp-nodes.d.ts +2 -2
- package/dest/test-helpers/reqresp-nodes.d.ts.map +1 -1
- package/dest/testbench/p2p_client_testbench_worker.d.ts +1 -1
- package/dest/testbench/p2p_client_testbench_worker.js +39 -22
- package/dest/testbench/parse_log_file.d.ts +1 -1
- package/dest/testbench/testbench.d.ts +1 -1
- package/dest/testbench/worker_client_manager.d.ts +1 -1
- package/dest/testbench/worker_client_manager.d.ts.map +1 -1
- package/dest/testbench/worker_client_manager.js +6 -1
- package/dest/types/index.d.ts +1 -1
- package/dest/util.d.ts +2 -1
- package/dest/util.d.ts.map +1 -1
- package/dest/util.js +11 -2
- package/dest/versioning.d.ts +1 -1
- package/package.json +19 -18
- package/src/client/factory.ts +5 -10
- package/src/client/interface.ts +20 -2
- package/src/client/p2p_client.ts +108 -155
- package/src/config.ts +30 -19
- package/src/errors/attestation-pool.error.ts +13 -0
- package/src/mem_pools/attestation_pool/attestation_pool.ts +86 -35
- package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +243 -278
- package/src/mem_pools/attestation_pool/kv_attestation_pool.ts +175 -111
- package/src/mem_pools/attestation_pool/memory_attestation_pool.ts +147 -136
- package/src/mem_pools/attestation_pool/mocks.ts +21 -15
- package/src/mem_pools/instrumentation.ts +47 -10
- package/src/mem_pools/interface.ts +2 -4
- package/src/mem_pools/tx_pool/README.md +270 -0
- package/src/mem_pools/tx_pool/aztec_kv_tx_pool.ts +367 -371
- package/src/mem_pools/tx_pool/eviction/eviction_manager.ts +132 -0
- package/src/mem_pools/tx_pool/eviction/eviction_strategy.ts +208 -0
- package/src/mem_pools/tx_pool/eviction/fee_payer_balance_eviction_rule.ts +162 -0
- package/src/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.ts +104 -0
- package/src/mem_pools/tx_pool/eviction/invalid_txs_after_reorg_rule.ts +93 -0
- package/src/mem_pools/tx_pool/eviction/low_priority_eviction_rule.ts +106 -0
- package/src/mem_pools/tx_pool/eviction/nullifier_conflict_pre_add_rule.ts +75 -0
- package/src/mem_pools/tx_pool/index.ts +0 -1
- package/src/mem_pools/tx_pool/priority.ts +8 -1
- package/src/mem_pools/tx_pool/tx_pool.ts +11 -5
- package/src/mem_pools/tx_pool/tx_pool_test_suite.ts +23 -17
- package/src/msg_validators/attestation_validator/attestation_validator.ts +45 -32
- package/src/msg_validators/attestation_validator/fisherman_attestation_validator.ts +88 -0
- package/src/msg_validators/attestation_validator/index.ts +1 -0
- package/src/msg_validators/clock_tolerance.ts +51 -0
- package/src/msg_validators/index.ts +1 -1
- package/src/msg_validators/proposal_validator/block_proposal_validator.ts +10 -0
- package/src/msg_validators/proposal_validator/checkpoint_proposal_validator.ts +13 -0
- package/src/msg_validators/proposal_validator/index.ts +3 -0
- package/src/msg_validators/proposal_validator/proposal_validator.ts +92 -0
- package/src/msg_validators/proposal_validator/proposal_validator_test_suite.ts +230 -0
- package/src/msg_validators/tx_validator/archive_cache.ts +1 -1
- package/src/msg_validators/tx_validator/block_header_validator.ts +1 -1
- package/src/msg_validators/tx_validator/data_validator.ts +12 -4
- package/src/msg_validators/tx_validator/factory.ts +3 -2
- package/src/msg_validators/tx_validator/fee_payer_balance.ts +40 -0
- package/src/msg_validators/tx_validator/gas_validator.ts +8 -25
- package/src/msg_validators/tx_validator/index.ts +1 -0
- package/src/msg_validators/tx_validator/metadata_validator.ts +13 -5
- package/src/msg_validators/tx_validator/phases_validator.ts +3 -1
- package/src/msg_validators/tx_validator/size_validator.ts +18 -0
- package/src/msg_validators/tx_validator/test_utils.ts +1 -1
- package/src/msg_validators/tx_validator/timestamp_validator.ts +5 -2
- package/src/services/dummy_service.ts +6 -0
- package/src/services/encoding.ts +6 -5
- package/src/services/libp2p/instrumentation.ts +26 -71
- package/src/services/libp2p/libp2p_service.ts +580 -160
- package/src/services/peer-manager/metrics.ts +27 -16
- package/src/services/peer-manager/peer_manager.ts +7 -4
- package/src/services/peer-manager/peer_scoring.ts +42 -3
- package/src/services/reqresp/connection-sampler/connection_sampler.ts +3 -1
- package/src/services/reqresp/constants.ts +14 -0
- package/src/services/reqresp/interface.ts +1 -1
- package/src/services/reqresp/metrics.ts +7 -23
- package/src/services/reqresp/protocols/auth.ts +2 -2
- package/src/services/reqresp/protocols/block.ts +3 -2
- package/src/services/reqresp/protocols/block_txs/bitvector.ts +9 -0
- package/src/services/reqresp/protocols/block_txs/block_txs_reqresp.ts +1 -1
- package/src/services/reqresp/protocols/status.ts +16 -12
- package/src/services/reqresp/protocols/tx.ts +1 -2
- package/src/services/service.ts +19 -4
- package/src/services/tx_collection/config.ts +1 -1
- package/src/services/tx_collection/fast_tx_collection.ts +3 -2
- package/src/services/tx_collection/instrumentation.ts +4 -21
- package/src/services/tx_collection/slow_tx_collection.ts +5 -4
- package/src/services/tx_collection/tx_collection.ts +6 -5
- package/src/services/tx_provider.ts +19 -3
- package/src/services/tx_provider_instrumentation.ts +18 -14
- package/src/test-helpers/mock-pubsub.ts +1 -1
- package/src/test-helpers/mock-tx-helpers.ts +1 -1
- package/src/test-helpers/reqresp-nodes.ts +1 -1
- package/src/testbench/p2p_client_testbench_worker.ts +45 -22
- package/src/testbench/worker_client_manager.ts +6 -1
- package/src/util.ts +12 -2
- package/dest/mem_pools/tx_pool/memory_tx_pool.d.ts +0 -80
- package/dest/mem_pools/tx_pool/memory_tx_pool.d.ts.map +0 -1
- package/dest/mem_pools/tx_pool/memory_tx_pool.js +0 -238
- package/dest/msg_validators/block_proposal_validator/block_proposal_validator.d.ts +0 -12
- package/dest/msg_validators/block_proposal_validator/block_proposal_validator.d.ts.map +0 -1
- package/dest/msg_validators/block_proposal_validator/block_proposal_validator.js +0 -70
- package/dest/msg_validators/block_proposal_validator/index.d.ts +0 -2
- package/dest/msg_validators/block_proposal_validator/index.d.ts.map +0 -1
- package/dest/msg_validators/block_proposal_validator/index.js +0 -1
- package/src/mem_pools/tx_pool/memory_tx_pool.ts +0 -283
- package/src/msg_validators/block_proposal_validator/block_proposal_validator.ts +0 -81
- package/src/msg_validators/block_proposal_validator/index.ts +0 -1
|
@@ -1,23 +1,390 @@
|
|
|
1
|
-
function
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
1
|
+
function applyDecs2203RFactory() {
|
|
2
|
+
function createAddInitializerMethod(initializers, decoratorFinishedRef) {
|
|
3
|
+
return function addInitializer(initializer) {
|
|
4
|
+
assertNotFinished(decoratorFinishedRef, "addInitializer");
|
|
5
|
+
assertCallable(initializer, "An initializer");
|
|
6
|
+
initializers.push(initializer);
|
|
7
|
+
};
|
|
8
|
+
}
|
|
9
|
+
function memberDec(dec, name, desc, initializers, kind, isStatic, isPrivate, metadata, value) {
|
|
10
|
+
var kindStr;
|
|
11
|
+
switch(kind){
|
|
12
|
+
case 1:
|
|
13
|
+
kindStr = "accessor";
|
|
14
|
+
break;
|
|
15
|
+
case 2:
|
|
16
|
+
kindStr = "method";
|
|
17
|
+
break;
|
|
18
|
+
case 3:
|
|
19
|
+
kindStr = "getter";
|
|
20
|
+
break;
|
|
21
|
+
case 4:
|
|
22
|
+
kindStr = "setter";
|
|
23
|
+
break;
|
|
24
|
+
default:
|
|
25
|
+
kindStr = "field";
|
|
26
|
+
}
|
|
27
|
+
var ctx = {
|
|
28
|
+
kind: kindStr,
|
|
29
|
+
name: isPrivate ? "#" + name : name,
|
|
30
|
+
static: isStatic,
|
|
31
|
+
private: isPrivate,
|
|
32
|
+
metadata: metadata
|
|
33
|
+
};
|
|
34
|
+
var decoratorFinishedRef = {
|
|
35
|
+
v: false
|
|
36
|
+
};
|
|
37
|
+
ctx.addInitializer = createAddInitializerMethod(initializers, decoratorFinishedRef);
|
|
38
|
+
var get, set;
|
|
39
|
+
if (kind === 0) {
|
|
40
|
+
if (isPrivate) {
|
|
41
|
+
get = desc.get;
|
|
42
|
+
set = desc.set;
|
|
43
|
+
} else {
|
|
44
|
+
get = function() {
|
|
45
|
+
return this[name];
|
|
46
|
+
};
|
|
47
|
+
set = function(v) {
|
|
48
|
+
this[name] = v;
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
} else if (kind === 2) {
|
|
52
|
+
get = function() {
|
|
53
|
+
return desc.value;
|
|
54
|
+
};
|
|
55
|
+
} else {
|
|
56
|
+
if (kind === 1 || kind === 3) {
|
|
57
|
+
get = function() {
|
|
58
|
+
return desc.get.call(this);
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
if (kind === 1 || kind === 4) {
|
|
62
|
+
set = function(v) {
|
|
63
|
+
desc.set.call(this, v);
|
|
64
|
+
};
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
ctx.access = get && set ? {
|
|
68
|
+
get: get,
|
|
69
|
+
set: set
|
|
70
|
+
} : get ? {
|
|
71
|
+
get: get
|
|
72
|
+
} : {
|
|
73
|
+
set: set
|
|
74
|
+
};
|
|
75
|
+
try {
|
|
76
|
+
return dec(value, ctx);
|
|
77
|
+
} finally{
|
|
78
|
+
decoratorFinishedRef.v = true;
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
function assertNotFinished(decoratorFinishedRef, fnName) {
|
|
82
|
+
if (decoratorFinishedRef.v) {
|
|
83
|
+
throw new Error("attempted to call " + fnName + " after decoration was finished");
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
function assertCallable(fn, hint) {
|
|
87
|
+
if (typeof fn !== "function") {
|
|
88
|
+
throw new TypeError(hint + " must be a function");
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
function assertValidReturnValue(kind, value) {
|
|
92
|
+
var type = typeof value;
|
|
93
|
+
if (kind === 1) {
|
|
94
|
+
if (type !== "object" || value === null) {
|
|
95
|
+
throw new TypeError("accessor decorators must return an object with get, set, or init properties or void 0");
|
|
96
|
+
}
|
|
97
|
+
if (value.get !== undefined) {
|
|
98
|
+
assertCallable(value.get, "accessor.get");
|
|
99
|
+
}
|
|
100
|
+
if (value.set !== undefined) {
|
|
101
|
+
assertCallable(value.set, "accessor.set");
|
|
102
|
+
}
|
|
103
|
+
if (value.init !== undefined) {
|
|
104
|
+
assertCallable(value.init, "accessor.init");
|
|
105
|
+
}
|
|
106
|
+
} else if (type !== "function") {
|
|
107
|
+
var hint;
|
|
108
|
+
if (kind === 0) {
|
|
109
|
+
hint = "field";
|
|
110
|
+
} else if (kind === 10) {
|
|
111
|
+
hint = "class";
|
|
112
|
+
} else {
|
|
113
|
+
hint = "method";
|
|
114
|
+
}
|
|
115
|
+
throw new TypeError(hint + " decorators must return a function or void 0");
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
function applyMemberDec(ret, base, decInfo, name, kind, isStatic, isPrivate, initializers, metadata) {
|
|
119
|
+
var decs = decInfo[0];
|
|
120
|
+
var desc, init, value;
|
|
121
|
+
if (isPrivate) {
|
|
122
|
+
if (kind === 0 || kind === 1) {
|
|
123
|
+
desc = {
|
|
124
|
+
get: decInfo[3],
|
|
125
|
+
set: decInfo[4]
|
|
126
|
+
};
|
|
127
|
+
} else if (kind === 3) {
|
|
128
|
+
desc = {
|
|
129
|
+
get: decInfo[3]
|
|
130
|
+
};
|
|
131
|
+
} else if (kind === 4) {
|
|
132
|
+
desc = {
|
|
133
|
+
set: decInfo[3]
|
|
134
|
+
};
|
|
135
|
+
} else {
|
|
136
|
+
desc = {
|
|
137
|
+
value: decInfo[3]
|
|
138
|
+
};
|
|
139
|
+
}
|
|
140
|
+
} else if (kind !== 0) {
|
|
141
|
+
desc = Object.getOwnPropertyDescriptor(base, name);
|
|
142
|
+
}
|
|
143
|
+
if (kind === 1) {
|
|
144
|
+
value = {
|
|
145
|
+
get: desc.get,
|
|
146
|
+
set: desc.set
|
|
147
|
+
};
|
|
148
|
+
} else if (kind === 2) {
|
|
149
|
+
value = desc.value;
|
|
150
|
+
} else if (kind === 3) {
|
|
151
|
+
value = desc.get;
|
|
152
|
+
} else if (kind === 4) {
|
|
153
|
+
value = desc.set;
|
|
154
|
+
}
|
|
155
|
+
var newValue, get, set;
|
|
156
|
+
if (typeof decs === "function") {
|
|
157
|
+
newValue = memberDec(decs, name, desc, initializers, kind, isStatic, isPrivate, metadata, value);
|
|
158
|
+
if (newValue !== void 0) {
|
|
159
|
+
assertValidReturnValue(kind, newValue);
|
|
160
|
+
if (kind === 0) {
|
|
161
|
+
init = newValue;
|
|
162
|
+
} else if (kind === 1) {
|
|
163
|
+
init = newValue.init;
|
|
164
|
+
get = newValue.get || value.get;
|
|
165
|
+
set = newValue.set || value.set;
|
|
166
|
+
value = {
|
|
167
|
+
get: get,
|
|
168
|
+
set: set
|
|
169
|
+
};
|
|
170
|
+
} else {
|
|
171
|
+
value = newValue;
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
} else {
|
|
175
|
+
for(var i = decs.length - 1; i >= 0; i--){
|
|
176
|
+
var dec = decs[i];
|
|
177
|
+
newValue = memberDec(dec, name, desc, initializers, kind, isStatic, isPrivate, metadata, value);
|
|
178
|
+
if (newValue !== void 0) {
|
|
179
|
+
assertValidReturnValue(kind, newValue);
|
|
180
|
+
var newInit;
|
|
181
|
+
if (kind === 0) {
|
|
182
|
+
newInit = newValue;
|
|
183
|
+
} else if (kind === 1) {
|
|
184
|
+
newInit = newValue.init;
|
|
185
|
+
get = newValue.get || value.get;
|
|
186
|
+
set = newValue.set || value.set;
|
|
187
|
+
value = {
|
|
188
|
+
get: get,
|
|
189
|
+
set: set
|
|
190
|
+
};
|
|
191
|
+
} else {
|
|
192
|
+
value = newValue;
|
|
193
|
+
}
|
|
194
|
+
if (newInit !== void 0) {
|
|
195
|
+
if (init === void 0) {
|
|
196
|
+
init = newInit;
|
|
197
|
+
} else if (typeof init === "function") {
|
|
198
|
+
init = [
|
|
199
|
+
init,
|
|
200
|
+
newInit
|
|
201
|
+
];
|
|
202
|
+
} else {
|
|
203
|
+
init.push(newInit);
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
if (kind === 0 || kind === 1) {
|
|
210
|
+
if (init === void 0) {
|
|
211
|
+
init = function(instance, init) {
|
|
212
|
+
return init;
|
|
213
|
+
};
|
|
214
|
+
} else if (typeof init !== "function") {
|
|
215
|
+
var ownInitializers = init;
|
|
216
|
+
init = function(instance, init) {
|
|
217
|
+
var value = init;
|
|
218
|
+
for(var i = 0; i < ownInitializers.length; i++){
|
|
219
|
+
value = ownInitializers[i].call(instance, value);
|
|
220
|
+
}
|
|
221
|
+
return value;
|
|
222
|
+
};
|
|
223
|
+
} else {
|
|
224
|
+
var originalInitializer = init;
|
|
225
|
+
init = function(instance, init) {
|
|
226
|
+
return originalInitializer.call(instance, init);
|
|
227
|
+
};
|
|
228
|
+
}
|
|
229
|
+
ret.push(init);
|
|
230
|
+
}
|
|
231
|
+
if (kind !== 0) {
|
|
232
|
+
if (kind === 1) {
|
|
233
|
+
desc.get = value.get;
|
|
234
|
+
desc.set = value.set;
|
|
235
|
+
} else if (kind === 2) {
|
|
236
|
+
desc.value = value;
|
|
237
|
+
} else if (kind === 3) {
|
|
238
|
+
desc.get = value;
|
|
239
|
+
} else if (kind === 4) {
|
|
240
|
+
desc.set = value;
|
|
241
|
+
}
|
|
242
|
+
if (isPrivate) {
|
|
243
|
+
if (kind === 1) {
|
|
244
|
+
ret.push(function(instance, args) {
|
|
245
|
+
return value.get.call(instance, args);
|
|
246
|
+
});
|
|
247
|
+
ret.push(function(instance, args) {
|
|
248
|
+
return value.set.call(instance, args);
|
|
249
|
+
});
|
|
250
|
+
} else if (kind === 2) {
|
|
251
|
+
ret.push(value);
|
|
252
|
+
} else {
|
|
253
|
+
ret.push(function(instance, args) {
|
|
254
|
+
return value.call(instance, args);
|
|
255
|
+
});
|
|
256
|
+
}
|
|
257
|
+
} else {
|
|
258
|
+
Object.defineProperty(base, name, desc);
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
function applyMemberDecs(Class, decInfos, metadata) {
|
|
263
|
+
var ret = [];
|
|
264
|
+
var protoInitializers;
|
|
265
|
+
var staticInitializers;
|
|
266
|
+
var existingProtoNonFields = new Map();
|
|
267
|
+
var existingStaticNonFields = new Map();
|
|
268
|
+
for(var i = 0; i < decInfos.length; i++){
|
|
269
|
+
var decInfo = decInfos[i];
|
|
270
|
+
if (!Array.isArray(decInfo)) continue;
|
|
271
|
+
var kind = decInfo[1];
|
|
272
|
+
var name = decInfo[2];
|
|
273
|
+
var isPrivate = decInfo.length > 3;
|
|
274
|
+
var isStatic = kind >= 5;
|
|
275
|
+
var base;
|
|
276
|
+
var initializers;
|
|
277
|
+
if (isStatic) {
|
|
278
|
+
base = Class;
|
|
279
|
+
kind = kind - 5;
|
|
280
|
+
staticInitializers = staticInitializers || [];
|
|
281
|
+
initializers = staticInitializers;
|
|
282
|
+
} else {
|
|
283
|
+
base = Class.prototype;
|
|
284
|
+
protoInitializers = protoInitializers || [];
|
|
285
|
+
initializers = protoInitializers;
|
|
286
|
+
}
|
|
287
|
+
if (kind !== 0 && !isPrivate) {
|
|
288
|
+
var existingNonFields = isStatic ? existingStaticNonFields : existingProtoNonFields;
|
|
289
|
+
var existingKind = existingNonFields.get(name) || 0;
|
|
290
|
+
if (existingKind === true || existingKind === 3 && kind !== 4 || existingKind === 4 && kind !== 3) {
|
|
291
|
+
throw new Error("Attempted to decorate a public method/accessor that has the same name as a previously decorated public method/accessor. This is not currently supported by the decorators plugin. Property name was: " + name);
|
|
292
|
+
} else if (!existingKind && kind > 2) {
|
|
293
|
+
existingNonFields.set(name, kind);
|
|
294
|
+
} else {
|
|
295
|
+
existingNonFields.set(name, true);
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
applyMemberDec(ret, base, decInfo, name, kind, isStatic, isPrivate, initializers, metadata);
|
|
299
|
+
}
|
|
300
|
+
pushInitializers(ret, protoInitializers);
|
|
301
|
+
pushInitializers(ret, staticInitializers);
|
|
302
|
+
return ret;
|
|
303
|
+
}
|
|
304
|
+
function pushInitializers(ret, initializers) {
|
|
305
|
+
if (initializers) {
|
|
306
|
+
ret.push(function(instance) {
|
|
307
|
+
for(var i = 0; i < initializers.length; i++){
|
|
308
|
+
initializers[i].call(instance);
|
|
309
|
+
}
|
|
310
|
+
return instance;
|
|
311
|
+
});
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
function applyClassDecs(targetClass, classDecs, metadata) {
|
|
315
|
+
if (classDecs.length > 0) {
|
|
316
|
+
var initializers = [];
|
|
317
|
+
var newClass = targetClass;
|
|
318
|
+
var name = targetClass.name;
|
|
319
|
+
for(var i = classDecs.length - 1; i >= 0; i--){
|
|
320
|
+
var decoratorFinishedRef = {
|
|
321
|
+
v: false
|
|
322
|
+
};
|
|
323
|
+
try {
|
|
324
|
+
var nextNewClass = classDecs[i](newClass, {
|
|
325
|
+
kind: "class",
|
|
326
|
+
name: name,
|
|
327
|
+
addInitializer: createAddInitializerMethod(initializers, decoratorFinishedRef),
|
|
328
|
+
metadata
|
|
329
|
+
});
|
|
330
|
+
} finally{
|
|
331
|
+
decoratorFinishedRef.v = true;
|
|
332
|
+
}
|
|
333
|
+
if (nextNewClass !== undefined) {
|
|
334
|
+
assertValidReturnValue(10, nextNewClass);
|
|
335
|
+
newClass = nextNewClass;
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
return [
|
|
339
|
+
defineMetadata(newClass, metadata),
|
|
340
|
+
function() {
|
|
341
|
+
for(var i = 0; i < initializers.length; i++){
|
|
342
|
+
initializers[i].call(newClass);
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
];
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
function defineMetadata(Class, metadata) {
|
|
349
|
+
return Object.defineProperty(Class, Symbol.metadata || Symbol.for("Symbol.metadata"), {
|
|
350
|
+
configurable: true,
|
|
351
|
+
enumerable: true,
|
|
352
|
+
value: metadata
|
|
353
|
+
});
|
|
354
|
+
}
|
|
355
|
+
return function applyDecs2203R(targetClass, memberDecs, classDecs, parentClass) {
|
|
356
|
+
if (parentClass !== void 0) {
|
|
357
|
+
var parentMetadata = parentClass[Symbol.metadata || Symbol.for("Symbol.metadata")];
|
|
358
|
+
}
|
|
359
|
+
var metadata = Object.create(parentMetadata === void 0 ? null : parentMetadata);
|
|
360
|
+
var e = applyMemberDecs(targetClass, memberDecs, metadata);
|
|
361
|
+
if (!classDecs.length) defineMetadata(targetClass, metadata);
|
|
362
|
+
return {
|
|
363
|
+
e: e,
|
|
364
|
+
get c () {
|
|
365
|
+
return applyClassDecs(targetClass, classDecs, metadata);
|
|
366
|
+
}
|
|
367
|
+
};
|
|
368
|
+
};
|
|
6
369
|
}
|
|
7
|
-
|
|
8
|
-
|
|
370
|
+
function _apply_decs_2203_r(targetClass, memberDecs, classDecs, parentClass) {
|
|
371
|
+
return (_apply_decs_2203_r = applyDecs2203RFactory())(targetClass, memberDecs, classDecs, parentClass);
|
|
372
|
+
}
|
|
373
|
+
var _dec, _dec1, _dec2, _dec3, _dec4, _dec5, _dec6, _dec7, _dec8, _initProto;
|
|
374
|
+
import { BlockNumber } from '@aztec/foundation/branded-types';
|
|
375
|
+
import { randomInt } from '@aztec/foundation/crypto/random';
|
|
376
|
+
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
9
377
|
import { createLibp2pComponentLogger, createLogger } from '@aztec/foundation/log';
|
|
10
|
-
import { SerialQueue } from '@aztec/foundation/queue';
|
|
11
378
|
import { RunningPromise } from '@aztec/foundation/running-promise';
|
|
12
379
|
import { Timer } from '@aztec/foundation/timer';
|
|
13
380
|
import { getVKTreeRoot } from '@aztec/noir-protocol-circuits-types/vk-tree';
|
|
14
381
|
import { protocolContractsHash } from '@aztec/protocol-contracts';
|
|
15
382
|
import { GasFees } from '@aztec/stdlib/gas';
|
|
16
|
-
import {
|
|
383
|
+
import { BlockProposal, CheckpointAttestation, CheckpointProposal, P2PClientType, P2PMessage, PeerErrorSeverity, TopicType, createTopicString, getTopicsForClientAndConfig, metricsTopicStrToLabels } from '@aztec/stdlib/p2p';
|
|
17
384
|
import { MerkleTreeId } from '@aztec/stdlib/trees';
|
|
18
385
|
import { Tx } from '@aztec/stdlib/tx';
|
|
19
386
|
import { compressComponentVersions } from '@aztec/stdlib/versioning';
|
|
20
|
-
import { Attributes, OtelMetricsAdapter, WithTracer, trackSpan } from '@aztec/telemetry-client';
|
|
387
|
+
import { Attributes, OtelMetricsAdapter, SpanStatusCode, WithTracer, trackSpan } from '@aztec/telemetry-client';
|
|
21
388
|
import { gossipsub } from '@chainsafe/libp2p-gossipsub';
|
|
22
389
|
import { createPeerScoreParams, createTopicScoreParams } from '@chainsafe/libp2p-gossipsub/score';
|
|
23
390
|
import { SignaturePolicy } from '@chainsafe/libp2p-gossipsub/types';
|
|
@@ -30,7 +397,8 @@ import { mplex } from '@libp2p/mplex';
|
|
|
30
397
|
import { tcp } from '@libp2p/tcp';
|
|
31
398
|
import { ENR } from '@nethermindeth/enr';
|
|
32
399
|
import { createLibp2p } from 'libp2p';
|
|
33
|
-
import {
|
|
400
|
+
import { ProposalSlotCapExceededError } from '../../errors/attestation-pool.error.js';
|
|
401
|
+
import { BlockProposalValidator, CheckpointAttestationValidator, CheckpointProposalValidator, FishermanAttestationValidator, SizeTxValidator } from '../../msg_validators/index.js';
|
|
34
402
|
import { MessageSeenValidator } from '../../msg_validators/msg_seen_validator/msg_seen_validator.js';
|
|
35
403
|
import { getDefaultAllowedSetupFunctions } from '../../msg_validators/tx_validator/allowed_public_setup.js';
|
|
36
404
|
import { createTxMessageValidators } from '../../msg_validators/tx_validator/factory.js';
|
|
@@ -50,6 +418,31 @@ import { reqGoodbyeHandler } from '../reqresp/protocols/goodbye.js';
|
|
|
50
418
|
import { pingHandler, reqRespBlockHandler, reqRespStatusHandler, reqRespTxHandler } from '../reqresp/protocols/index.js';
|
|
51
419
|
import { ReqResp } from '../reqresp/reqresp.js';
|
|
52
420
|
import { P2PInstrumentation } from './instrumentation.js';
|
|
421
|
+
_dec = trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
|
|
422
|
+
[Attributes.SLOT_NUMBER]: block.slotNumber,
|
|
423
|
+
[Attributes.BLOCK_ARCHIVE]: block.archive.toString(),
|
|
424
|
+
[Attributes.P2P_ID]: await block.p2pMessageLoggingIdentifier().then((i)=>i.toString())
|
|
425
|
+
})), _dec1 = trackSpan('Libp2pService.processValidCheckpointProposal', async (checkpoint)=>({
|
|
426
|
+
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber,
|
|
427
|
+
[Attributes.BLOCK_ARCHIVE]: checkpoint.archive.toString(),
|
|
428
|
+
[Attributes.P2P_ID]: await checkpoint.p2pMessageLoggingIdentifier().then((i)=>i.toString())
|
|
429
|
+
})), _dec2 = trackSpan('Libp2pService.validateRequestedBlockTxs', (request)=>({
|
|
430
|
+
[Attributes.BLOCK_HASH]: request.blockHash.toString()
|
|
431
|
+
})), _dec3 = trackSpan('Libp2pService.validateRequestedTx', (requestedTxHash, _responseTx)=>({
|
|
432
|
+
[Attributes.TX_HASH]: requestedTxHash.toString()
|
|
433
|
+
})), _dec4 = trackSpan('Libp2pService.validateRequestedBlock', (requestedBlockNumber, _responseBlock)=>({
|
|
434
|
+
[Attributes.BLOCK_NUMBER]: requestedBlockNumber.toString()
|
|
435
|
+
})), _dec5 = trackSpan('Libp2pService.validatePropagatedTx', (tx)=>({
|
|
436
|
+
[Attributes.TX_HASH]: tx.getTxHash().toString()
|
|
437
|
+
})), _dec6 = trackSpan('Libp2pService.validateCheckpointAttestation', async (_, attestation)=>({
|
|
438
|
+
[Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber,
|
|
439
|
+
[Attributes.BLOCK_ARCHIVE]: attestation.archive.toString(),
|
|
440
|
+
[Attributes.P2P_ID]: await attestation.p2pMessageLoggingIdentifier().then((i)=>i.toString())
|
|
441
|
+
})), _dec7 = trackSpan('Libp2pService.validateBlockProposal', (_peerId, block)=>({
|
|
442
|
+
[Attributes.SLOT_NUMBER]: block.slotNumber.toString()
|
|
443
|
+
})), _dec8 = trackSpan('Libp2pService.validateCheckpointProposal', (_peerId, checkpoint)=>({
|
|
444
|
+
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString()
|
|
445
|
+
}));
|
|
53
446
|
/**
|
|
54
447
|
* Lib P2P implementation of the P2PService interface.
|
|
55
448
|
*/ export class LibP2PService extends WithTracer {
|
|
@@ -64,13 +457,61 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
64
457
|
epochCache;
|
|
65
458
|
proofVerifier;
|
|
66
459
|
worldStateSynchronizer;
|
|
67
|
-
|
|
68
|
-
|
|
460
|
+
static{
|
|
461
|
+
({ e: [_initProto] } = _apply_decs_2203_r(this, [
|
|
462
|
+
[
|
|
463
|
+
_dec,
|
|
464
|
+
2,
|
|
465
|
+
"processValidBlockProposal"
|
|
466
|
+
],
|
|
467
|
+
[
|
|
468
|
+
_dec1,
|
|
469
|
+
2,
|
|
470
|
+
"processValidCheckpointProposal"
|
|
471
|
+
],
|
|
472
|
+
[
|
|
473
|
+
_dec2,
|
|
474
|
+
2,
|
|
475
|
+
"validateRequestedBlockTxs"
|
|
476
|
+
],
|
|
477
|
+
[
|
|
478
|
+
_dec3,
|
|
479
|
+
2,
|
|
480
|
+
"validateRequestedTxs"
|
|
481
|
+
],
|
|
482
|
+
[
|
|
483
|
+
_dec4,
|
|
484
|
+
2,
|
|
485
|
+
"validateRequestedBlock"
|
|
486
|
+
],
|
|
487
|
+
[
|
|
488
|
+
_dec5,
|
|
489
|
+
2,
|
|
490
|
+
"validatePropagatedTx"
|
|
491
|
+
],
|
|
492
|
+
[
|
|
493
|
+
_dec6,
|
|
494
|
+
2,
|
|
495
|
+
"validateCheckpointAttestation"
|
|
496
|
+
],
|
|
497
|
+
[
|
|
498
|
+
_dec7,
|
|
499
|
+
2,
|
|
500
|
+
"validateBlockProposal"
|
|
501
|
+
],
|
|
502
|
+
[
|
|
503
|
+
_dec8,
|
|
504
|
+
2,
|
|
505
|
+
"validateCheckpointProposal"
|
|
506
|
+
]
|
|
507
|
+
], []));
|
|
508
|
+
}
|
|
69
509
|
discoveryRunningPromise;
|
|
70
510
|
msgIdSeenValidators;
|
|
71
511
|
// Message validators
|
|
72
|
-
attestationValidator;
|
|
73
512
|
blockProposalValidator;
|
|
513
|
+
checkpointProposalValidator;
|
|
514
|
+
checkpointAttestationValidator;
|
|
74
515
|
protocolVersion;
|
|
75
516
|
topicStrings;
|
|
76
517
|
feesCache;
|
|
@@ -79,30 +520,49 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
79
520
|
* @param block - The block received from the peer.
|
|
80
521
|
* @returns The attestation for the block, if any.
|
|
81
522
|
*/ blockReceivedCallback;
|
|
523
|
+
/**
|
|
524
|
+
* Callback for when a checkpoint proposal is received from a peer.
|
|
525
|
+
* @param checkpoint - The checkpoint proposal received from the peer.
|
|
526
|
+
* @returns The attestations for the checkpoint, if any.
|
|
527
|
+
*/ checkpointReceivedCallback;
|
|
82
528
|
gossipSubEventHandler;
|
|
83
529
|
instrumentation;
|
|
530
|
+
telemetry;
|
|
531
|
+
logger;
|
|
84
532
|
constructor(clientType, config, node, peerDiscoveryService, reqresp, peerManager, mempools, archiver, epochCache, proofVerifier, worldStateSynchronizer, telemetry, logger = createLogger('p2p:libp2p_service')){
|
|
85
|
-
super(telemetry, 'LibP2PService'), this.clientType = clientType, this.config = config, this.node = node, this.peerDiscoveryService = peerDiscoveryService, this.reqresp = reqresp, this.peerManager = peerManager, this.mempools = mempools, this.archiver = archiver, this.epochCache = epochCache, this.proofVerifier = proofVerifier, this.worldStateSynchronizer = worldStateSynchronizer, this.
|
|
533
|
+
super(telemetry, 'LibP2PService'), this.clientType = clientType, this.config = config, this.node = node, this.peerDiscoveryService = peerDiscoveryService, this.reqresp = reqresp, this.peerManager = peerManager, this.mempools = mempools, this.archiver = archiver, this.epochCache = epochCache, this.proofVerifier = proofVerifier, this.worldStateSynchronizer = worldStateSynchronizer, this.msgIdSeenValidators = (_initProto(this), {}), this.protocolVersion = '', this.topicStrings = {};
|
|
534
|
+
this.telemetry = telemetry;
|
|
535
|
+
// Create child logger with fisherman prefix if in fisherman mode
|
|
536
|
+
this.logger = config.fishermanMode ? logger.createChild('[FISHERMAN]') : logger;
|
|
86
537
|
this.instrumentation = new P2PInstrumentation(telemetry, 'LibP2PService');
|
|
87
538
|
this.msgIdSeenValidators[TopicType.tx] = new MessageSeenValidator(config.seenMessageCacheSize);
|
|
88
539
|
this.msgIdSeenValidators[TopicType.block_proposal] = new MessageSeenValidator(config.seenMessageCacheSize);
|
|
89
|
-
this.msgIdSeenValidators[TopicType.
|
|
540
|
+
this.msgIdSeenValidators[TopicType.checkpoint_proposal] = new MessageSeenValidator(config.seenMessageCacheSize);
|
|
541
|
+
this.msgIdSeenValidators[TopicType.checkpoint_attestation] = new MessageSeenValidator(config.seenMessageCacheSize);
|
|
90
542
|
const versions = getVersions(config);
|
|
91
543
|
this.protocolVersion = compressComponentVersions(versions);
|
|
92
544
|
logger.info(`Started libp2p service with protocol version ${this.protocolVersion}`);
|
|
93
545
|
this.topicStrings[TopicType.tx] = createTopicString(TopicType.tx, this.protocolVersion);
|
|
94
546
|
this.topicStrings[TopicType.block_proposal] = createTopicString(TopicType.block_proposal, this.protocolVersion);
|
|
95
|
-
this.topicStrings[TopicType.
|
|
96
|
-
this.
|
|
547
|
+
this.topicStrings[TopicType.checkpoint_proposal] = createTopicString(TopicType.checkpoint_proposal, this.protocolVersion);
|
|
548
|
+
this.topicStrings[TopicType.checkpoint_attestation] = createTopicString(TopicType.checkpoint_attestation, this.protocolVersion);
|
|
97
549
|
this.blockProposalValidator = new BlockProposalValidator(epochCache, {
|
|
98
550
|
txsPermitted: !config.disableTransactions
|
|
99
551
|
});
|
|
552
|
+
this.checkpointProposalValidator = new CheckpointProposalValidator(epochCache, {
|
|
553
|
+
txsPermitted: !config.disableTransactions
|
|
554
|
+
});
|
|
555
|
+
this.checkpointAttestationValidator = config.fishermanMode ? new FishermanAttestationValidator(epochCache, mempools.attestationPool, telemetry) : new CheckpointAttestationValidator(epochCache);
|
|
100
556
|
this.gossipSubEventHandler = this.handleGossipSubEvent.bind(this);
|
|
101
557
|
this.blockReceivedCallback = async (block)=>{
|
|
102
|
-
this.logger.debug(`Handler not yet registered: Block received callback not set. Received block for slot ${block.slotNumber
|
|
103
|
-
p2pMessageIdentifier: await block.
|
|
558
|
+
this.logger.debug(`Handler not yet registered: Block received callback not set. Received block for slot ${block.slotNumber} from peer.`, {
|
|
559
|
+
p2pMessageIdentifier: await block.p2pMessageLoggingIdentifier()
|
|
104
560
|
});
|
|
105
|
-
return
|
|
561
|
+
return false;
|
|
562
|
+
};
|
|
563
|
+
this.checkpointReceivedCallback = (checkpoint)=>{
|
|
564
|
+
this.logger.debug(`Handler not yet registered: Checkpoint received callback not set. Received checkpoint for slot ${checkpoint.slotNumber} from peer.`);
|
|
565
|
+
return Promise.resolve(undefined);
|
|
106
566
|
};
|
|
107
567
|
}
|
|
108
568
|
updateConfig(config) {
|
|
@@ -135,7 +595,8 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
135
595
|
const protocolVersion = compressComponentVersions(versions);
|
|
136
596
|
const txTopic = createTopicString(TopicType.tx, protocolVersion);
|
|
137
597
|
const blockProposalTopic = createTopicString(TopicType.block_proposal, protocolVersion);
|
|
138
|
-
const
|
|
598
|
+
const checkpointProposalTopic = createTopicString(TopicType.checkpoint_proposal, protocolVersion);
|
|
599
|
+
const checkpointAttestationTopic = createTopicString(TopicType.checkpoint_attestation, protocolVersion);
|
|
139
600
|
const preferredPeersEnrs = config.preferredPeers.map((enr)=>ENR.decodeTxt(enr));
|
|
140
601
|
const directPeers = (await Promise.all(preferredPeersEnrs.map(async (enr)=>{
|
|
141
602
|
const peerId = await enr.peerId();
|
|
@@ -259,12 +720,17 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
259
720
|
invalidMessageDeliveriesWeight: -20,
|
|
260
721
|
invalidMessageDeliveriesDecay: 0.5
|
|
261
722
|
}),
|
|
262
|
-
[
|
|
723
|
+
[blockProposalTopic]: createTopicScoreParams({
|
|
263
724
|
topicWeight: 1,
|
|
264
725
|
invalidMessageDeliveriesWeight: -20,
|
|
265
726
|
invalidMessageDeliveriesDecay: 0.5
|
|
266
727
|
}),
|
|
267
|
-
[
|
|
728
|
+
[checkpointProposalTopic]: createTopicScoreParams({
|
|
729
|
+
topicWeight: 1,
|
|
730
|
+
invalidMessageDeliveriesWeight: -20,
|
|
731
|
+
invalidMessageDeliveriesDecay: 0.5
|
|
732
|
+
}),
|
|
733
|
+
[checkpointAttestationTopic]: createTopicScoreParams({
|
|
268
734
|
topicWeight: 1,
|
|
269
735
|
invalidMessageDeliveriesWeight: -20,
|
|
270
736
|
invalidMessageDeliveriesDecay: 0.5
|
|
@@ -278,7 +744,7 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
278
744
|
},
|
|
279
745
|
logger: createLibp2pComponentLogger(logger.module)
|
|
280
746
|
});
|
|
281
|
-
const peerScoring = new PeerScoring(config);
|
|
747
|
+
const peerScoring = new PeerScoring(config, telemetry);
|
|
282
748
|
const reqresp = new ReqResp(config, node, peerScoring, createLogger(`${logger.module}:reqresp`));
|
|
283
749
|
const peerManager = new PeerManager(node, peerDiscoveryService, config, telemetry, createLogger(`${logger.module}:peer_manager`), peerScoring, reqresp, worldStateSynchronizer, protocolVersion, epochCache);
|
|
284
750
|
// Update gossipsub score params
|
|
@@ -300,17 +766,6 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
300
766
|
throw new Error('Announce address not provided.');
|
|
301
767
|
}
|
|
302
768
|
const announceTcpMultiaddr = convertToMultiaddr(p2pIp, p2pPort, 'tcp');
|
|
303
|
-
// Start job queue, peer discovery service and libp2p node
|
|
304
|
-
this.jobQueue.start();
|
|
305
|
-
await this.peerManager.initializePeers();
|
|
306
|
-
if (!this.config.p2pDiscoveryDisabled) {
|
|
307
|
-
await this.peerDiscoveryService.start();
|
|
308
|
-
}
|
|
309
|
-
await this.node.start();
|
|
310
|
-
// Subscribe to standard GossipSub topics by default
|
|
311
|
-
for (const topic of getTopicsForClientAndConfig(this.clientType, this.config.disableTransactions)){
|
|
312
|
-
this.subscribeToTopic(this.topicStrings[topic]);
|
|
313
|
-
}
|
|
314
769
|
// Create request response protocol handlers
|
|
315
770
|
const txHandler = reqRespTxHandler(this.mempools);
|
|
316
771
|
const goodbyeHandler = reqGoodbyeHandler(this.peerManager);
|
|
@@ -322,27 +777,37 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
322
777
|
[ReqRespSubProtocol.GOODBYE]: goodbyeHandler.bind(this),
|
|
323
778
|
[ReqRespSubProtocol.BLOCK]: blockHandler.bind(this)
|
|
324
779
|
};
|
|
325
|
-
|
|
326
|
-
if (this.mempools.attestationPool && !this.config.disableTransactions) {
|
|
780
|
+
if (!this.config.disableTransactions) {
|
|
327
781
|
const blockTxsHandler = reqRespBlockTxsHandler(this.mempools.attestationPool, this.mempools.txPool);
|
|
328
782
|
requestResponseHandlers[ReqRespSubProtocol.BLOCK_TXS] = blockTxsHandler.bind(this);
|
|
329
783
|
}
|
|
330
784
|
if (!this.config.disableTransactions) {
|
|
331
785
|
requestResponseHandlers[ReqRespSubProtocol.TX] = txHandler.bind(this);
|
|
332
786
|
}
|
|
333
|
-
// add GossipSub listener
|
|
334
|
-
this.node.services.pubsub.addEventListener(GossipSubEvent.MESSAGE, this.gossipSubEventHandler);
|
|
335
|
-
// Start running promise for peer discovery
|
|
336
|
-
this.discoveryRunningPromise = new RunningPromise(()=>this.peerManager.heartbeat(), this.logger, this.config.peerCheckIntervalMS);
|
|
337
|
-
this.discoveryRunningPromise.start();
|
|
338
787
|
// Define the sub protocol validators - This is done within this start() method to gain a callback to the existing validateTx function
|
|
339
788
|
const reqrespSubProtocolValidators = {
|
|
340
789
|
...DEFAULT_SUB_PROTOCOL_VALIDATORS,
|
|
341
|
-
// TODO(#11336): A request validator for blocks
|
|
342
790
|
[ReqRespSubProtocol.TX]: this.validateRequestedTxs.bind(this),
|
|
343
|
-
[ReqRespSubProtocol.BLOCK_TXS]: this.validateRequestedBlockTxs.bind(this)
|
|
791
|
+
[ReqRespSubProtocol.BLOCK_TXS]: this.validateRequestedBlockTxs.bind(this),
|
|
792
|
+
[ReqRespSubProtocol.BLOCK]: this.validateRequestedBlock.bind(this)
|
|
344
793
|
};
|
|
794
|
+
await this.peerManager.initializePeers();
|
|
345
795
|
await this.reqresp.start(requestResponseHandlers, reqrespSubProtocolValidators);
|
|
796
|
+
await this.node.start();
|
|
797
|
+
// Subscribe to standard GossipSub topics by default
|
|
798
|
+
for (const topic of getTopicsForClientAndConfig(this.clientType, this.config.disableTransactions)){
|
|
799
|
+
this.subscribeToTopic(this.topicStrings[topic]);
|
|
800
|
+
}
|
|
801
|
+
// add GossipSub listener
|
|
802
|
+
this.node.services.pubsub.addEventListener(GossipSubEvent.MESSAGE, this.gossipSubEventHandler);
|
|
803
|
+
// Start running promise for peer discovery and metrics collection
|
|
804
|
+
if (!this.config.p2pDiscoveryDisabled) {
|
|
805
|
+
await this.peerDiscoveryService.start();
|
|
806
|
+
}
|
|
807
|
+
this.discoveryRunningPromise = new RunningPromise(async ()=>{
|
|
808
|
+
await this.peerManager.heartbeat();
|
|
809
|
+
}, this.logger, this.config.peerCheckIntervalMS);
|
|
810
|
+
this.discoveryRunningPromise.start();
|
|
346
811
|
this.logger.info(`Started P2P service`, {
|
|
347
812
|
listen: this.config.listenAddress,
|
|
348
813
|
port: this.config.p2pPort,
|
|
@@ -359,8 +824,6 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
359
824
|
// Stop peer manager
|
|
360
825
|
this.logger.debug('Stopping peer manager...');
|
|
361
826
|
await this.peerManager.stop();
|
|
362
|
-
this.logger.debug('Stopping job queue...');
|
|
363
|
-
await this.jobQueue.end();
|
|
364
827
|
this.logger.debug('Stopping running promise...');
|
|
365
828
|
await this.discoveryRunningPromise?.stop();
|
|
366
829
|
this.logger.debug('Stopping peer discovery service...');
|
|
@@ -408,6 +871,9 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
408
871
|
registerBlockReceivedCallback(callback) {
|
|
409
872
|
this.blockReceivedCallback = callback;
|
|
410
873
|
}
|
|
874
|
+
registerCheckpointReceivedCallback(callback) {
|
|
875
|
+
this.checkpointReceivedCallback = callback;
|
|
876
|
+
}
|
|
411
877
|
/**
|
|
412
878
|
* Subscribes to a topic.
|
|
413
879
|
* @param topic - The topic to subscribe to.
|
|
@@ -426,7 +892,9 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
426
892
|
if (!this.node.services.pubsub) {
|
|
427
893
|
throw new Error('Pubsub service not available.');
|
|
428
894
|
}
|
|
429
|
-
const
|
|
895
|
+
const isBlockProposal = topic === this.topicStrings[TopicType.block_proposal];
|
|
896
|
+
const traceContext = this.config.debugP2PInstrumentMessages && isBlockProposal ? this.telemetry.getTraceContext() : undefined;
|
|
897
|
+
const p2pMessage = P2PMessage.fromGossipable(message, this.config.debugP2PInstrumentMessages, traceContext);
|
|
430
898
|
const result = await this.node.services.pubsub.publish(topic, p2pMessage.toMessageData());
|
|
431
899
|
return result.recipients.length;
|
|
432
900
|
}
|
|
@@ -440,12 +908,15 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
440
908
|
case this.topicStrings[TopicType.tx]:
|
|
441
909
|
topicType = TopicType.tx;
|
|
442
910
|
break;
|
|
443
|
-
case this.topicStrings[TopicType.block_attestation]:
|
|
444
|
-
topicType = TopicType.block_attestation;
|
|
445
|
-
break;
|
|
446
911
|
case this.topicStrings[TopicType.block_proposal]:
|
|
447
912
|
topicType = TopicType.block_proposal;
|
|
448
913
|
break;
|
|
914
|
+
case this.topicStrings[TopicType.checkpoint_proposal]:
|
|
915
|
+
topicType = TopicType.checkpoint_proposal;
|
|
916
|
+
break;
|
|
917
|
+
case this.topicStrings[TopicType.checkpoint_attestation]:
|
|
918
|
+
topicType = TopicType.checkpoint_attestation;
|
|
919
|
+
break;
|
|
449
920
|
default:
|
|
450
921
|
this.logger.error(`Received message on unknown topic: ${msg.topic}`);
|
|
451
922
|
break;
|
|
@@ -466,23 +937,97 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
466
937
|
};
|
|
467
938
|
}
|
|
468
939
|
/**
|
|
940
|
+
* Safely deserializes a P2PMessage from raw message data.
|
|
941
|
+
* @param msgId - The message ID.
|
|
942
|
+
* @param source - The peer ID of the message source.
|
|
943
|
+
* @param data - The raw message data.
|
|
944
|
+
* @returns The deserialized P2PMessage or undefined if deserialization fails.
|
|
945
|
+
*/ safelyDeserializeP2PMessage(msgId, source, data) {
|
|
946
|
+
try {
|
|
947
|
+
return P2PMessage.fromMessageData(Buffer.from(data), this.config.debugP2PInstrumentMessages);
|
|
948
|
+
} catch (err) {
|
|
949
|
+
this.logger.error(`Error deserializing P2PMessage`, err, {
|
|
950
|
+
msgId,
|
|
951
|
+
source: source.toString()
|
|
952
|
+
});
|
|
953
|
+
this.node.services.pubsub.reportMessageValidationResult(msgId, source.toString(), TopicValidatorResult.Reject);
|
|
954
|
+
this.peerManager.penalizePeer(source, PeerErrorSeverity.LowToleranceError);
|
|
955
|
+
return undefined;
|
|
956
|
+
}
|
|
957
|
+
}
|
|
958
|
+
/**
|
|
469
959
|
* Handles a new gossip message that was received by the client.
|
|
470
960
|
* @param topic - The message's topic.
|
|
471
961
|
* @param data - The message data
|
|
472
962
|
*/ async handleNewGossipMessage(msg, msgId, source) {
|
|
473
|
-
const
|
|
963
|
+
const msgReceivedTime = Date.now();
|
|
964
|
+
let topicType;
|
|
965
|
+
const p2pMessage = this.safelyDeserializeP2PMessage(msgId, source, msg.data);
|
|
966
|
+
if (!p2pMessage) {
|
|
967
|
+
return;
|
|
968
|
+
}
|
|
474
969
|
const preValidationResult = this.preValidateReceivedMessage(msg, msgId, source);
|
|
475
970
|
if (!preValidationResult.result) {
|
|
476
971
|
return;
|
|
477
972
|
}
|
|
973
|
+
// Determine topic type for attributes
|
|
478
974
|
if (msg.topic === this.topicStrings[TopicType.tx]) {
|
|
479
|
-
|
|
975
|
+
topicType = TopicType.tx;
|
|
976
|
+
} else if (msg.topic === this.topicStrings[TopicType.checkpoint_attestation]) {
|
|
977
|
+
topicType = TopicType.checkpoint_attestation;
|
|
978
|
+
} else if (msg.topic === this.topicStrings[TopicType.block_proposal]) {
|
|
979
|
+
topicType = TopicType.block_proposal;
|
|
980
|
+
} else if (msg.topic === this.topicStrings[TopicType.checkpoint_proposal]) {
|
|
981
|
+
topicType = TopicType.checkpoint_proposal;
|
|
480
982
|
}
|
|
481
|
-
|
|
482
|
-
|
|
983
|
+
// Process the message, optionally within a linked span for trace propagation
|
|
984
|
+
const processMessage = async ()=>{
|
|
985
|
+
if (msg.topic === this.topicStrings[TopicType.tx]) {
|
|
986
|
+
await this.handleGossipedTx(p2pMessage.payload, msgId, source);
|
|
987
|
+
} else if (msg.topic === this.topicStrings[TopicType.checkpoint_attestation]) {
|
|
988
|
+
if (this.clientType === P2PClientType.Full) {
|
|
989
|
+
await this.processCheckpointAttestationFromPeer(p2pMessage.payload, msgId, source);
|
|
990
|
+
}
|
|
991
|
+
} else if (msg.topic === this.topicStrings[TopicType.block_proposal]) {
|
|
992
|
+
await this.processBlockFromPeer(p2pMessage.payload, msgId, source);
|
|
993
|
+
} else if (msg.topic === this.topicStrings[TopicType.checkpoint_proposal]) {
|
|
994
|
+
await this.handleGossipedCheckpointProposal(p2pMessage.payload, msgId, source);
|
|
995
|
+
} else {
|
|
996
|
+
this.logger.error(`Received message on unknown topic: ${msg.topic}`);
|
|
997
|
+
}
|
|
998
|
+
};
|
|
999
|
+
const latency = p2pMessage.timestamp !== undefined ? msgReceivedTime - p2pMessage.timestamp.getTime() : undefined;
|
|
1000
|
+
const propagatedContext = p2pMessage.traceContext ? this.telemetry.extractPropagatedContext(p2pMessage.traceContext) : undefined;
|
|
1001
|
+
if (propagatedContext) {
|
|
1002
|
+
await this.tracer.startActiveSpan('LibP2PService.processMessage', {
|
|
1003
|
+
attributes: {
|
|
1004
|
+
[Attributes.TOPIC_NAME]: topicType,
|
|
1005
|
+
[Attributes.PEER_ID]: source.toString()
|
|
1006
|
+
}
|
|
1007
|
+
}, propagatedContext, async (span)=>{
|
|
1008
|
+
try {
|
|
1009
|
+
await processMessage();
|
|
1010
|
+
span.setStatus({
|
|
1011
|
+
code: SpanStatusCode.OK
|
|
1012
|
+
});
|
|
1013
|
+
} catch (err) {
|
|
1014
|
+
span.setStatus({
|
|
1015
|
+
code: SpanStatusCode.ERROR,
|
|
1016
|
+
message: String(err)
|
|
1017
|
+
});
|
|
1018
|
+
if (typeof err === 'string' || err && err instanceof Error) {
|
|
1019
|
+
span.recordException(err);
|
|
1020
|
+
}
|
|
1021
|
+
throw err;
|
|
1022
|
+
} finally{
|
|
1023
|
+
span.end();
|
|
1024
|
+
}
|
|
1025
|
+
});
|
|
1026
|
+
} else {
|
|
1027
|
+
await processMessage();
|
|
483
1028
|
}
|
|
484
|
-
if (
|
|
485
|
-
|
|
1029
|
+
if (latency !== undefined && topicType !== undefined) {
|
|
1030
|
+
this.instrumentation.recordMessageLatency(topicType, latency);
|
|
486
1031
|
}
|
|
487
1032
|
return;
|
|
488
1033
|
}
|
|
@@ -494,6 +1039,7 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
494
1039
|
try {
|
|
495
1040
|
resultAndObj = await validationFunc();
|
|
496
1041
|
} catch (err) {
|
|
1042
|
+
this.peerManager.penalizePeer(source, PeerErrorSeverity.LowToleranceError);
|
|
497
1043
|
this.logger.error(`Error deserializing and validating gossipsub message`, err, {
|
|
498
1044
|
msgId,
|
|
499
1045
|
source: source.toString(),
|
|
@@ -546,31 +1092,50 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
546
1092
|
this.logger.warn(`Intentionally dropping tx ${txHashString} (probability rule)`);
|
|
547
1093
|
return;
|
|
548
1094
|
}
|
|
1095
|
+
this.instrumentation.incrementTxReceived(1);
|
|
549
1096
|
await this.mempools.txPool.addTxs([
|
|
550
1097
|
tx
|
|
551
1098
|
]);
|
|
552
1099
|
}
|
|
553
1100
|
/**
|
|
554
|
-
* Process
|
|
555
|
-
*
|
|
556
|
-
|
|
557
|
-
* @param attestation - The attestation to process.
|
|
558
|
-
*/ async processAttestationFromPeer(payloadData, msgId, source) {
|
|
1101
|
+
* Process a checkpoint attestation from a peer.
|
|
1102
|
+
* Validates the attestation and adds it to the pool.
|
|
1103
|
+
*/ async processCheckpointAttestationFromPeer(payloadData, msgId, source) {
|
|
559
1104
|
const validationFunc = async ()=>{
|
|
560
|
-
const attestation =
|
|
561
|
-
const
|
|
562
|
-
const
|
|
563
|
-
|
|
1105
|
+
const attestation = CheckpointAttestation.fromBuffer(payloadData);
|
|
1106
|
+
const pool = this.mempools.attestationPool;
|
|
1107
|
+
const validationResult = await this.validateCheckpointAttestation(source, attestation);
|
|
1108
|
+
const isValid = validationResult.result === 'accept';
|
|
1109
|
+
const exists = isValid && await pool.hasCheckpointAttestation(attestation);
|
|
1110
|
+
let canAdd = true;
|
|
1111
|
+
if (isValid && !exists) {
|
|
1112
|
+
const slot = attestation.payload.header.slotNumber;
|
|
1113
|
+
const { committee } = await this.epochCache.getCommittee(slot);
|
|
1114
|
+
const committeeSize = committee?.length ?? 0;
|
|
1115
|
+
canAdd = await pool.canAddCheckpointAttestation(attestation, committeeSize);
|
|
1116
|
+
}
|
|
1117
|
+
this.logger.trace(`Validate propagated checkpoint attestation`, {
|
|
564
1118
|
isValid,
|
|
565
1119
|
exists,
|
|
1120
|
+
canAdd,
|
|
566
1121
|
[Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber.toString(),
|
|
567
1122
|
[Attributes.P2P_ID]: source.toString()
|
|
568
1123
|
});
|
|
569
|
-
if (
|
|
1124
|
+
if (validationResult.result === 'reject') {
|
|
570
1125
|
return {
|
|
571
1126
|
result: TopicValidatorResult.Reject
|
|
572
1127
|
};
|
|
573
|
-
} else if (exists) {
|
|
1128
|
+
} else if (validationResult.result === 'ignore' || exists) {
|
|
1129
|
+
return {
|
|
1130
|
+
result: TopicValidatorResult.Ignore,
|
|
1131
|
+
obj: attestation
|
|
1132
|
+
};
|
|
1133
|
+
} else if (!canAdd) {
|
|
1134
|
+
this.logger.warn(`Dropping checkpoint attestation due to per-(slot, proposalId) attestation cap`, {
|
|
1135
|
+
slot: attestation.payload.header.slotNumber.toString(),
|
|
1136
|
+
archive: attestation.archive.toString(),
|
|
1137
|
+
source: source.toString()
|
|
1138
|
+
});
|
|
574
1139
|
return {
|
|
575
1140
|
result: TopicValidatorResult.Ignore,
|
|
576
1141
|
obj: attestation
|
|
@@ -582,42 +1147,54 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
582
1147
|
};
|
|
583
1148
|
}
|
|
584
1149
|
};
|
|
585
|
-
const { result, obj: attestation } = await this.validateReceivedMessage(validationFunc, msgId, source, TopicType.
|
|
1150
|
+
const { result, obj: attestation } = await this.validateReceivedMessage(validationFunc, msgId, source, TopicType.checkpoint_attestation);
|
|
586
1151
|
if (result !== TopicValidatorResult.Accept || !attestation) {
|
|
587
1152
|
return;
|
|
588
1153
|
}
|
|
589
|
-
this.logger.debug(`Received attestation for slot ${attestation.slotNumber
|
|
590
|
-
p2pMessageIdentifier: await attestation.
|
|
591
|
-
slot: attestation.slotNumber
|
|
1154
|
+
this.logger.debug(`Received checkpoint attestation for slot ${attestation.slotNumber} from external peer ${source.toString()}`, {
|
|
1155
|
+
p2pMessageIdentifier: await attestation.p2pMessageLoggingIdentifier(),
|
|
1156
|
+
slot: attestation.slotNumber,
|
|
592
1157
|
archive: attestation.archive.toString(),
|
|
593
1158
|
source: source.toString()
|
|
594
1159
|
});
|
|
595
|
-
await this.mempools.attestationPool.
|
|
1160
|
+
await this.mempools.attestationPool.addCheckpointAttestations([
|
|
596
1161
|
attestation
|
|
597
1162
|
]);
|
|
598
1163
|
}
|
|
599
1164
|
async processBlockFromPeer(payloadData, msgId, source) {
|
|
600
1165
|
const validationFunc = async ()=>{
|
|
601
1166
|
const block = BlockProposal.fromBuffer(payloadData);
|
|
602
|
-
const
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
const exists = isValid && await
|
|
1167
|
+
const validationResult = await this.validateBlockProposal(source, block);
|
|
1168
|
+
const isValid = validationResult.result === 'accept';
|
|
1169
|
+
const pool = this.mempools.attestationPool;
|
|
1170
|
+
const exists = isValid && await pool.hasBlockProposal(block);
|
|
1171
|
+
const canAdd = isValid && await pool.canAddProposal(block);
|
|
606
1172
|
this.logger.trace(`Validate propagated block proposal`, {
|
|
607
1173
|
isValid,
|
|
608
1174
|
exists,
|
|
609
|
-
|
|
1175
|
+
canAdd,
|
|
1176
|
+
[Attributes.SLOT_NUMBER]: block.slotNumber.toString(),
|
|
610
1177
|
[Attributes.P2P_ID]: source.toString()
|
|
611
1178
|
});
|
|
612
|
-
if (
|
|
1179
|
+
if (validationResult.result === 'reject') {
|
|
613
1180
|
return {
|
|
614
1181
|
result: TopicValidatorResult.Reject
|
|
615
1182
|
};
|
|
616
|
-
} else if (exists) {
|
|
1183
|
+
} else if (validationResult.result === 'ignore' || exists) {
|
|
617
1184
|
return {
|
|
618
1185
|
result: TopicValidatorResult.Ignore,
|
|
619
1186
|
obj: block
|
|
620
1187
|
};
|
|
1188
|
+
} else if (!canAdd) {
|
|
1189
|
+
this.peerManager.penalizePeer(source, PeerErrorSeverity.MidToleranceError);
|
|
1190
|
+
this.logger.warn(`Penalizing peer for block proposal exceeding per-slot cap`, {
|
|
1191
|
+
slot: block.slotNumber.toString(),
|
|
1192
|
+
archive: block.archive.toString(),
|
|
1193
|
+
source: source.toString()
|
|
1194
|
+
});
|
|
1195
|
+
return {
|
|
1196
|
+
result: TopicValidatorResult.Reject
|
|
1197
|
+
};
|
|
621
1198
|
} else {
|
|
622
1199
|
return {
|
|
623
1200
|
result: TopicValidatorResult.Accept,
|
|
@@ -632,68 +1209,204 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
632
1209
|
await this.processValidBlockProposal(block, source);
|
|
633
1210
|
}
|
|
634
1211
|
// REVIEW: callback pattern https://github.com/AztecProtocol/aztec-packages/issues/7963
|
|
1212
|
+
// REFACTOR(palla): This method should be moved to the p2p_client or to a separate component,
|
|
1213
|
+
// should not be here as it does not deal with p2p networking.
|
|
635
1214
|
async processValidBlockProposal(block, sender) {
|
|
636
|
-
const slot = block.slotNumber
|
|
637
|
-
const previousSlot = slot - 1n;
|
|
1215
|
+
const slot = block.slotNumber;
|
|
638
1216
|
this.logger.verbose(`Received block proposal for slot ${slot} from external peer ${sender.toString()}.`, {
|
|
639
|
-
p2pMessageIdentifier: await block.
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
source: sender.toString()
|
|
1217
|
+
p2pMessageIdentifier: await block.p2pMessageLoggingIdentifier(),
|
|
1218
|
+
source: sender.toString(),
|
|
1219
|
+
...block.toBlockInfo()
|
|
643
1220
|
});
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
this.
|
|
1221
|
+
// Attempt to add proposal
|
|
1222
|
+
try {
|
|
1223
|
+
await this.mempools.attestationPool.addBlockProposal(block);
|
|
1224
|
+
} catch (err) {
|
|
1225
|
+
// Drop proposals if we hit per-slot cap in the attestation pool; rethrow unknown errors
|
|
1226
|
+
if (err instanceof ProposalSlotCapExceededError) {
|
|
1227
|
+
this.logger.warn(`Dropping block proposal due to per-slot proposal cap`, {
|
|
1228
|
+
slot: String(slot),
|
|
1229
|
+
archive: block.archive.toString(),
|
|
1230
|
+
error: err.message
|
|
1231
|
+
});
|
|
1232
|
+
return;
|
|
1233
|
+
}
|
|
1234
|
+
throw err;
|
|
647
1235
|
}
|
|
648
1236
|
// Mark the txs in this proposal as non-evictable
|
|
649
1237
|
await this.mempools.txPool.markTxsAsNonEvictable(block.txHashes);
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
654
|
-
|
|
655
|
-
|
|
656
|
-
|
|
657
|
-
|
|
658
|
-
|
|
659
|
-
|
|
1238
|
+
// Call the block received callback to validate the proposal.
|
|
1239
|
+
// Note: Validators do NOT attest to individual blocks, only to checkpoint proposals.
|
|
1240
|
+
const isValid = await this.blockReceivedCallback(block, sender);
|
|
1241
|
+
if (!isValid) {
|
|
1242
|
+
this.logger.warn(`Block proposal validation failed for block ${block.blockNumber}`, block.toBlockInfo());
|
|
1243
|
+
}
|
|
1244
|
+
}
|
|
1245
|
+
/**
|
|
1246
|
+
* Handle a gossiped checkpoint proposal.
|
|
1247
|
+
* Validates and processes the checkpoint proposal, then triggers the callback for attestation.
|
|
1248
|
+
*/ async handleGossipedCheckpointProposal(payloadData, msgId, source) {
|
|
1249
|
+
// TODO(palla/mbps): This pattern is repeated across multiple message handlers, consider abstracting it.
|
|
1250
|
+
const validationFunc = async ()=>{
|
|
1251
|
+
const checkpoint = CheckpointProposal.fromBuffer(payloadData);
|
|
1252
|
+
const validationResult = await this.validateCheckpointProposal(source, checkpoint);
|
|
1253
|
+
const isValid = validationResult.result === 'accept';
|
|
1254
|
+
const pool = this.mempools.attestationPool;
|
|
1255
|
+
const exists = isValid && await pool.hasCheckpointProposal(checkpoint);
|
|
1256
|
+
const canAdd = isValid && await pool.canAddCheckpointProposal(checkpoint);
|
|
1257
|
+
this.logger.trace(`Validate propagated checkpoint proposal`, {
|
|
1258
|
+
isValid,
|
|
1259
|
+
exists,
|
|
1260
|
+
canAdd,
|
|
1261
|
+
[Attributes.SLOT_NUMBER]: checkpoint.slotNumber.toString(),
|
|
1262
|
+
[Attributes.P2P_ID]: source.toString()
|
|
1263
|
+
});
|
|
1264
|
+
if (validationResult.result === 'reject') {
|
|
1265
|
+
return {
|
|
1266
|
+
result: TopicValidatorResult.Reject
|
|
1267
|
+
};
|
|
1268
|
+
} else if (validationResult.result === 'ignore' || exists) {
|
|
1269
|
+
return {
|
|
1270
|
+
result: TopicValidatorResult.Ignore,
|
|
1271
|
+
obj: checkpoint
|
|
1272
|
+
};
|
|
1273
|
+
} else if (!canAdd) {
|
|
1274
|
+
this.peerManager.penalizePeer(source, PeerErrorSeverity.MidToleranceError);
|
|
1275
|
+
this.logger.warn(`Penalizing peer for checkpoint proposal exceeding per-slot cap`, {
|
|
1276
|
+
slot: checkpoint.slotNumber.toString(),
|
|
1277
|
+
archive: checkpoint.archive.toString(),
|
|
1278
|
+
source: source.toString()
|
|
660
1279
|
});
|
|
661
|
-
|
|
1280
|
+
return {
|
|
1281
|
+
result: TopicValidatorResult.Reject
|
|
1282
|
+
};
|
|
1283
|
+
} else {
|
|
1284
|
+
return {
|
|
1285
|
+
result: TopicValidatorResult.Accept,
|
|
1286
|
+
obj: checkpoint
|
|
1287
|
+
};
|
|
662
1288
|
}
|
|
1289
|
+
};
|
|
1290
|
+
const { result, obj: checkpoint } = await this.validateReceivedMessage(validationFunc, msgId, source, TopicType.checkpoint_proposal);
|
|
1291
|
+
if (result !== TopicValidatorResult.Accept || !checkpoint) {
|
|
1292
|
+
return;
|
|
663
1293
|
}
|
|
1294
|
+
await this.processValidCheckpointProposal(checkpoint, source);
|
|
664
1295
|
}
|
|
665
1296
|
/**
|
|
666
|
-
*
|
|
667
|
-
*
|
|
668
|
-
|
|
669
|
-
|
|
1297
|
+
* Process a validated checkpoint proposal.
|
|
1298
|
+
* Extracts and processes the last block proposal (if present) first, then processes the checkpoint.
|
|
1299
|
+
* The block callback is invoked before the checkpoint callback.
|
|
1300
|
+
*/ async processValidCheckpointProposal(checkpoint, sender) {
|
|
1301
|
+
const slot = checkpoint.slotNumber;
|
|
1302
|
+
this.logger.verbose(`Received checkpoint proposal for slot ${slot} from external peer ${sender.toString()}.`, {
|
|
1303
|
+
p2pMessageIdentifier: await checkpoint.p2pMessageLoggingIdentifier(),
|
|
1304
|
+
slot: checkpoint.slotNumber,
|
|
1305
|
+
archive: checkpoint.archive.toString(),
|
|
1306
|
+
source: sender.toString()
|
|
1307
|
+
});
|
|
1308
|
+
// Extract block proposal before adding to pool (pool stores them separately)
|
|
1309
|
+
const blockProposal = checkpoint.getBlockProposal();
|
|
1310
|
+
// Add proposal to the pool (this extracts and stores block proposal separately)
|
|
1311
|
+
await this.mempools.attestationPool.addCheckpointProposal(checkpoint);
|
|
1312
|
+
// Mark txs as non-evictable if present (from the last block)
|
|
1313
|
+
if (checkpoint.txHashes.length > 0) {
|
|
1314
|
+
await this.mempools.txPool.markTxsAsNonEvictable(checkpoint.txHashes);
|
|
1315
|
+
}
|
|
1316
|
+
// If there was a last block proposal, invoke the block callback first for validation.
|
|
1317
|
+
// Note: The block proposal is already stored in the pool by addCheckpointProposal.
|
|
1318
|
+
if (blockProposal) {
|
|
1319
|
+
const isValid = await this.blockReceivedCallback(blockProposal, sender);
|
|
1320
|
+
if (!isValid) {
|
|
1321
|
+
this.logger.warn(`Block proposal from checkpoint failed validation`, {
|
|
1322
|
+
slot: slot.toString(),
|
|
1323
|
+
archive: checkpoint.archive.toString(),
|
|
1324
|
+
blockNumber: blockProposal.blockNumber.toString()
|
|
1325
|
+
});
|
|
1326
|
+
return;
|
|
1327
|
+
}
|
|
1328
|
+
}
|
|
1329
|
+
// Call the checkpoint received callback with the core version (without lastBlock)
|
|
1330
|
+
// to validate and potentially generate attestations
|
|
1331
|
+
const attestations = await this.checkpointReceivedCallback(checkpoint.toCore(), sender);
|
|
1332
|
+
if (attestations && attestations.length > 0) {
|
|
1333
|
+
// If the callback returned attestations, add them to the pool and propagate them
|
|
1334
|
+
await this.mempools.attestationPool.addCheckpointAttestations(attestations);
|
|
1335
|
+
for (const attestation of attestations){
|
|
1336
|
+
await this.propagate(attestation);
|
|
1337
|
+
}
|
|
1338
|
+
}
|
|
670
1339
|
}
|
|
671
1340
|
/**
|
|
672
1341
|
* Propagates provided message to peers.
|
|
673
1342
|
* @param message - The message to propagate.
|
|
674
1343
|
*/ async propagate(message) {
|
|
675
|
-
const p2pMessageIdentifier = await message.
|
|
1344
|
+
const p2pMessageIdentifier = await message.p2pMessageLoggingIdentifier();
|
|
676
1345
|
this.logger.trace(`Message ${p2pMessageIdentifier} queued`, {
|
|
677
1346
|
p2pMessageIdentifier
|
|
678
1347
|
});
|
|
679
|
-
void this.
|
|
680
|
-
await this.sendToPeers(message);
|
|
681
|
-
}).catch((error)=>{
|
|
1348
|
+
void this.sendToPeers(message).catch((error)=>{
|
|
682
1349
|
this.logger.error(`Error propagating message ${p2pMessageIdentifier}`, {
|
|
683
1350
|
error
|
|
684
1351
|
});
|
|
685
1352
|
});
|
|
686
1353
|
}
|
|
687
1354
|
/**
|
|
688
|
-
* Validate the requested block transactions.
|
|
1355
|
+
* Validate the requested block transactions. Allow partial returns.
|
|
689
1356
|
* @param request - The block transactions request.
|
|
690
1357
|
* @param response - The block transactions response.
|
|
691
1358
|
* @param peerId - The ID of the peer that made the request.
|
|
692
1359
|
* @returns True if the requested block transactions are valid, false otherwise.
|
|
693
|
-
*/ async validateRequestedBlockTxs(
|
|
1360
|
+
*/ async validateRequestedBlockTxs(request, response, peerId) {
|
|
694
1361
|
const requestedTxValidator = this.createRequestedTxValidator();
|
|
695
1362
|
try {
|
|
696
|
-
|
|
1363
|
+
if (!response.blockHash.equals(request.blockHash)) {
|
|
1364
|
+
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
|
|
1365
|
+
throw new ValidationError(`Received block txs for unexpected block: expected ${request.blockHash.toString()}, got ${response.blockHash.toString()}`);
|
|
1366
|
+
}
|
|
1367
|
+
if (response.txIndices.getLength() !== request.txIndices.getLength()) {
|
|
1368
|
+
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
|
|
1369
|
+
throw new ValidationError(`Received block txs with mismatched bitvector length: expected ${request.txIndices.getLength()}, got ${response.txIndices.getLength()}`);
|
|
1370
|
+
}
|
|
1371
|
+
// Check no duplicates and not exceeding returnable count
|
|
1372
|
+
const requestedIndices = new Set(request.txIndices.getTrueIndices());
|
|
1373
|
+
const availableIndices = new Set(response.txIndices.getTrueIndices());
|
|
1374
|
+
const maxReturnable = [
|
|
1375
|
+
...requestedIndices
|
|
1376
|
+
].filter((i)=>availableIndices.has(i)).length;
|
|
1377
|
+
const returnedHashes = await Promise.all(response.txs.map((tx)=>tx.getTxHash().toString()));
|
|
1378
|
+
const uniqueReturned = new Set(returnedHashes.map((h)=>h.toString()));
|
|
1379
|
+
if (uniqueReturned.size !== returnedHashes.length) {
|
|
1380
|
+
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
|
|
1381
|
+
throw new ValidationError(`Received duplicate txs in block txs response`);
|
|
1382
|
+
}
|
|
1383
|
+
if (response.txs.length > maxReturnable) {
|
|
1384
|
+
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
|
|
1385
|
+
throw new ValidationError(`Received more txs (${response.txs.length}) than requested-and-available (${maxReturnable})`);
|
|
1386
|
+
}
|
|
1387
|
+
// Given proposal (should have locally), ensure returned txs are valid subset and match request indices
|
|
1388
|
+
const proposal = await this.mempools.attestationPool.getBlockProposal(request.blockHash.toString());
|
|
1389
|
+
if (proposal) {
|
|
1390
|
+
// Build intersected indices
|
|
1391
|
+
const intersectIdx = request.txIndices.getTrueIndices().filter((i)=>response.txIndices.isSet(i));
|
|
1392
|
+
// Enforce subset membership and preserve increasing order by index.
|
|
1393
|
+
const hashToIndexInProposal = new Map(proposal.txHashes.map((h, i)=>[
|
|
1394
|
+
h.toString(),
|
|
1395
|
+
i
|
|
1396
|
+
]));
|
|
1397
|
+
const allowedIndexSet = new Set(intersectIdx);
|
|
1398
|
+
const indices = returnedHashes.map((h)=>hashToIndexInProposal.get(h));
|
|
1399
|
+
const allAllowed = indices.every((idx)=>idx !== undefined && allowedIndexSet.has(idx));
|
|
1400
|
+
const strictlyIncreasing = indices.every((idx, i)=>i === 0 ? idx !== undefined : idx > indices[i - 1]);
|
|
1401
|
+
if (!allAllowed || !strictlyIncreasing) {
|
|
1402
|
+
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.LowToleranceError);
|
|
1403
|
+
throw new ValidationError('Returned txs do not match expected subset/order for requested indices');
|
|
1404
|
+
}
|
|
1405
|
+
} else {
|
|
1406
|
+
// No local proposal, cannot check the membership/order of the returned txs
|
|
1407
|
+
this.logger.warn(`Block proposal not found for block hash ${request.blockHash.toString()}; cannot validate membership/order of returned txs`);
|
|
1408
|
+
return false;
|
|
1409
|
+
}
|
|
697
1410
|
await Promise.all(response.txs.map((tx)=>this.validateRequestedTx(tx, peerId, requestedTxValidator)));
|
|
698
1411
|
return true;
|
|
699
1412
|
} catch (e) {
|
|
@@ -715,7 +1428,7 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
715
1428
|
* ReqRespSubProtocol.TX subprotocol validation.
|
|
716
1429
|
*
|
|
717
1430
|
* @param requestedTxHash - The collection of the txs that was requested.
|
|
718
|
-
* @param responseTx - The
|
|
1431
|
+
* @param responseTx - The collection of txs that was received as a response to the request.
|
|
719
1432
|
* @param peerId - The peer ID of the peer that sent the tx.
|
|
720
1433
|
* @returns True if the whole collection of txs is valid, false otherwise.
|
|
721
1434
|
*/ async validateRequestedTxs(requestedTxHash, responseTx, peerId) {
|
|
@@ -735,8 +1448,47 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
735
1448
|
return false;
|
|
736
1449
|
}
|
|
737
1450
|
}
|
|
1451
|
+
/**
|
|
1452
|
+
* Validates a BLOCK response.
|
|
1453
|
+
*
|
|
1454
|
+
* If a local copy exists, enforces hash equality. If missing, rejects (no penalty) since the hash cannot be verified.
|
|
1455
|
+
* Penalizes on block number mismatch or hash mismatch.
|
|
1456
|
+
*
|
|
1457
|
+
* @param requestedBlockNumber - The requested block number.
|
|
1458
|
+
* @param responseBlock - The block returned by the peer.
|
|
1459
|
+
* @param peerId - The peer that returned the block.
|
|
1460
|
+
* @returns True if the response is valid, false otherwise.
|
|
1461
|
+
*/ async validateRequestedBlock(requestedBlockNumber, responseBlock, peerId) {
|
|
1462
|
+
try {
|
|
1463
|
+
const reqNum = Number(requestedBlockNumber.toString());
|
|
1464
|
+
if (responseBlock.number !== reqNum) {
|
|
1465
|
+
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.LowToleranceError);
|
|
1466
|
+
return false;
|
|
1467
|
+
}
|
|
1468
|
+
const local = await this.archiver.getBlock(BlockNumber(reqNum));
|
|
1469
|
+
if (!local) {
|
|
1470
|
+
// We are missing the local block; we cannot verify the hash yet. Reject without penalizing.
|
|
1471
|
+
// TODO: Consider extending this validator to accept an expected hash or
|
|
1472
|
+
// performing quorum-based checks when using P2P syncing prior to L1 sync.
|
|
1473
|
+
this.logger.warn(`Local block ${reqNum} not found; rejecting BLOCK response without hash verification`);
|
|
1474
|
+
return false;
|
|
1475
|
+
}
|
|
1476
|
+
const [localHash, respHash] = await Promise.all([
|
|
1477
|
+
local.hash(),
|
|
1478
|
+
responseBlock.hash()
|
|
1479
|
+
]);
|
|
1480
|
+
if (!localHash.equals(respHash)) {
|
|
1481
|
+
this.peerManager.penalizePeer(peerId, PeerErrorSeverity.MidToleranceError);
|
|
1482
|
+
return false;
|
|
1483
|
+
}
|
|
1484
|
+
return true;
|
|
1485
|
+
} catch (e) {
|
|
1486
|
+
this.logger.warn(`Error validating requested block`, e);
|
|
1487
|
+
return false;
|
|
1488
|
+
}
|
|
1489
|
+
}
|
|
738
1490
|
createRequestedTxValidator() {
|
|
739
|
-
return new AggregateTxValidator(new DataTxValidator(), new MetadataTxValidator({
|
|
1491
|
+
return new AggregateTxValidator(new DataTxValidator(), new SizeTxValidator(), new MetadataTxValidator({
|
|
740
1492
|
l1ChainId: new Fr(this.config.l1ChainId),
|
|
741
1493
|
rollupVersion: new Fr(this.config.rollupVersion),
|
|
742
1494
|
protocolContractsHash,
|
|
@@ -744,17 +1496,18 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
744
1496
|
}), new TxProofValidator(this.proofVerifier));
|
|
745
1497
|
}
|
|
746
1498
|
async validateRequestedTx(tx, peerId, txValidator, requested) {
|
|
1499
|
+
const penalize = (severity)=>this.peerManager.penalizePeer(peerId, severity);
|
|
747
1500
|
if (!await tx.validateTxHash()) {
|
|
748
|
-
|
|
1501
|
+
penalize(PeerErrorSeverity.MidToleranceError);
|
|
749
1502
|
throw new ValidationError(`Received tx with invalid hash ${tx.getTxHash().toString()}.`);
|
|
750
1503
|
}
|
|
751
1504
|
if (requested && !requested.has(tx.getTxHash().toString())) {
|
|
752
|
-
|
|
1505
|
+
penalize(PeerErrorSeverity.MidToleranceError);
|
|
753
1506
|
throw new ValidationError(`Received tx with hash ${tx.getTxHash().toString()} that was not requested.`);
|
|
754
1507
|
}
|
|
755
1508
|
const { result } = await txValidator.validateTx(tx);
|
|
756
1509
|
if (result === 'invalid') {
|
|
757
|
-
|
|
1510
|
+
penalize(PeerErrorSeverity.LowToleranceError);
|
|
758
1511
|
throw new ValidationError(`Received tx with hash ${tx.getTxHash().toString()} that is invalid.`);
|
|
759
1512
|
}
|
|
760
1513
|
}
|
|
@@ -772,7 +1525,7 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
772
1525
|
let { severity } = outcome.failure;
|
|
773
1526
|
// Double spend validator has a special case handler
|
|
774
1527
|
if (name === 'doubleSpendValidator') {
|
|
775
|
-
const txBlockNumber = currentBlockNumber + 1; // tx is expected to be in the next block
|
|
1528
|
+
const txBlockNumber = BlockNumber(currentBlockNumber + 1); // tx is expected to be in the next block
|
|
776
1529
|
severity = await this.handleDoubleSpendFailure(tx, txBlockNumber);
|
|
777
1530
|
}
|
|
778
1531
|
this.peerManager.penalizePeer(peerId, severity);
|
|
@@ -822,7 +1575,7 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
822
1575
|
*/ async createMessageValidators(currentBlockNumber, nextSlotTimestamp) {
|
|
823
1576
|
const gasFees = await this.getGasFees(currentBlockNumber);
|
|
824
1577
|
const allowedInSetup = this.config.txPublicSetupAllowList ?? await getDefaultAllowedSetupFunctions();
|
|
825
|
-
const blockNumberInWhichTheTxIsConsideredToBeIncluded = currentBlockNumber + 1;
|
|
1578
|
+
const blockNumberInWhichTheTxIsConsideredToBeIncluded = BlockNumber(currentBlockNumber + 1);
|
|
826
1579
|
return createTxMessageValidators(nextSlotTimestamp, blockNumberInWhichTheTxIsConsideredToBeIncluded, this.worldStateSynchronizer, gasFees, this.config.l1ChainId, this.config.rollupVersion, protocolContractsHash, this.archiver, this.proofVerifier, !this.config.disableTransactions, allowedInSetup);
|
|
827
1580
|
}
|
|
828
1581
|
/**
|
|
@@ -878,7 +1631,7 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
878
1631
|
}
|
|
879
1632
|
const snapshotValidator = new DoubleSpendTxValidator({
|
|
880
1633
|
nullifiersExist: async (nullifiers)=>{
|
|
881
|
-
const merkleTree = this.worldStateSynchronizer.getSnapshot(blockNumber - this.config.doubleSpendSeverePeerPenaltyWindow);
|
|
1634
|
+
const merkleTree = this.worldStateSynchronizer.getSnapshot(BlockNumber(blockNumber - this.config.doubleSpendSeverePeerPenaltyWindow));
|
|
882
1635
|
const indices = await merkleTree.findLeafIndices(MerkleTreeId.NULLIFIER_TREE, nullifiers);
|
|
883
1636
|
return indices.map((index)=>index !== undefined);
|
|
884
1637
|
}
|
|
@@ -890,17 +1643,17 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
890
1643
|
return PeerErrorSeverity.HighToleranceError;
|
|
891
1644
|
}
|
|
892
1645
|
/**
|
|
893
|
-
* Validate
|
|
1646
|
+
* Validate a checkpoint attestation.
|
|
894
1647
|
*
|
|
895
|
-
* @param attestation - The attestation to validate.
|
|
896
|
-
* @returns True if the attestation is valid, false otherwise.
|
|
897
|
-
*/ async
|
|
898
|
-
const
|
|
899
|
-
if (
|
|
900
|
-
this.
|
|
901
|
-
|
|
1648
|
+
* @param attestation - The checkpoint attestation to validate.
|
|
1649
|
+
* @returns True if the checkpoint attestation is valid, false otherwise.
|
|
1650
|
+
*/ async validateCheckpointAttestation(peerId, attestation) {
|
|
1651
|
+
const result = await this.checkpointAttestationValidator.validate(attestation);
|
|
1652
|
+
if (result.result === 'reject') {
|
|
1653
|
+
this.logger.debug(`Penalizing peer ${peerId} for checkpoint attestation validation failure`);
|
|
1654
|
+
this.peerManager.penalizePeer(peerId, result.severity);
|
|
902
1655
|
}
|
|
903
|
-
return
|
|
1656
|
+
return result;
|
|
904
1657
|
}
|
|
905
1658
|
/**
|
|
906
1659
|
* Validate a block proposal.
|
|
@@ -908,13 +1661,25 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
908
1661
|
* @param block - The block proposal to validate.
|
|
909
1662
|
* @returns True if the block proposal is valid, false otherwise.
|
|
910
1663
|
*/ async validateBlockProposal(peerId, block) {
|
|
911
|
-
const
|
|
912
|
-
if (
|
|
1664
|
+
const result = await this.blockProposalValidator.validate(block);
|
|
1665
|
+
if (result.result === 'reject') {
|
|
913
1666
|
this.logger.debug(`Penalizing peer ${peerId} for block proposal validation failure`);
|
|
914
|
-
this.peerManager.penalizePeer(peerId, severity);
|
|
915
|
-
return false;
|
|
1667
|
+
this.peerManager.penalizePeer(peerId, result.severity);
|
|
916
1668
|
}
|
|
917
|
-
return
|
|
1669
|
+
return result;
|
|
1670
|
+
}
|
|
1671
|
+
/**
|
|
1672
|
+
* Validate a checkpoint proposal.
|
|
1673
|
+
*
|
|
1674
|
+
* @param checkpoint - The checkpoint proposal to validate.
|
|
1675
|
+
* @returns True if the checkpoint proposal is valid, false otherwise.
|
|
1676
|
+
*/ async validateCheckpointProposal(peerId, checkpoint) {
|
|
1677
|
+
const result = await this.checkpointProposalValidator.validate(checkpoint);
|
|
1678
|
+
if (result.result === 'reject') {
|
|
1679
|
+
this.logger.debug(`Penalizing peer ${peerId} for checkpoint proposal validation failure`);
|
|
1680
|
+
this.peerManager.penalizePeer(peerId, result.severity);
|
|
1681
|
+
}
|
|
1682
|
+
return result;
|
|
918
1683
|
}
|
|
919
1684
|
getPeerScore(peerId) {
|
|
920
1685
|
return this.node.services.pubsub.score.score(peerId.toString());
|
|
@@ -924,7 +1689,7 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
924
1689
|
}
|
|
925
1690
|
async sendToPeers(message) {
|
|
926
1691
|
const parent = message.constructor;
|
|
927
|
-
const identifier = await message.
|
|
1692
|
+
const identifier = await message.p2pMessageLoggingIdentifier().then((i)=>i.toString());
|
|
928
1693
|
this.logger.trace(`Sending message ${identifier}`, {
|
|
929
1694
|
p2pMessageIdentifier: identifier
|
|
930
1695
|
});
|
|
@@ -951,44 +1716,3 @@ import { P2PInstrumentation } from './instrumentation.js';
|
|
|
951
1716
|
}
|
|
952
1717
|
}
|
|
953
1718
|
}
|
|
954
|
-
_ts_decorate([
|
|
955
|
-
trackSpan('Libp2pService.processValidBlockProposal', async (block)=>({
|
|
956
|
-
[Attributes.SLOT_NUMBER]: block.slotNumber.toNumber(),
|
|
957
|
-
[Attributes.BLOCK_ARCHIVE]: block.archive.toString(),
|
|
958
|
-
[Attributes.P2P_ID]: await block.p2pMessageIdentifier().then((i)=>i.toString())
|
|
959
|
-
}))
|
|
960
|
-
], LibP2PService.prototype, "processValidBlockProposal", null);
|
|
961
|
-
_ts_decorate([
|
|
962
|
-
trackSpan('Libp2pService.broadcastAttestation', async (attestation)=>({
|
|
963
|
-
[Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber.toNumber(),
|
|
964
|
-
[Attributes.BLOCK_ARCHIVE]: attestation.archive.toString(),
|
|
965
|
-
[Attributes.P2P_ID]: await attestation.p2pMessageIdentifier().then((i)=>i.toString())
|
|
966
|
-
}))
|
|
967
|
-
], LibP2PService.prototype, "broadcastAttestation", null);
|
|
968
|
-
_ts_decorate([
|
|
969
|
-
trackSpan('Libp2pService.validateRequestedBlockTxs', (request)=>({
|
|
970
|
-
[Attributes.BLOCK_HASH]: request.blockHash.toString()
|
|
971
|
-
}))
|
|
972
|
-
], LibP2PService.prototype, "validateRequestedBlockTxs", null);
|
|
973
|
-
_ts_decorate([
|
|
974
|
-
trackSpan('Libp2pService.validateRequestedTx', (requestedTxHash, _responseTx)=>({
|
|
975
|
-
[Attributes.TX_HASH]: requestedTxHash.toString()
|
|
976
|
-
}))
|
|
977
|
-
], LibP2PService.prototype, "validateRequestedTxs", null);
|
|
978
|
-
_ts_decorate([
|
|
979
|
-
trackSpan('Libp2pService.validatePropagatedTx', (tx)=>({
|
|
980
|
-
[Attributes.TX_HASH]: tx.getTxHash().toString()
|
|
981
|
-
}))
|
|
982
|
-
], LibP2PService.prototype, "validatePropagatedTx", null);
|
|
983
|
-
_ts_decorate([
|
|
984
|
-
trackSpan('Libp2pService.validateAttestation', async (_, attestation)=>({
|
|
985
|
-
[Attributes.SLOT_NUMBER]: attestation.payload.header.slotNumber.toNumber(),
|
|
986
|
-
[Attributes.BLOCK_ARCHIVE]: attestation.archive.toString(),
|
|
987
|
-
[Attributes.P2P_ID]: await attestation.p2pMessageIdentifier().then((i)=>i.toString())
|
|
988
|
-
}))
|
|
989
|
-
], LibP2PService.prototype, "validateAttestation", null);
|
|
990
|
-
_ts_decorate([
|
|
991
|
-
trackSpan('Libp2pService.validateBlockProposal', (_peerId, block)=>({
|
|
992
|
-
[Attributes.SLOT_NUMBER]: block.payload.header.slotNumber.toString()
|
|
993
|
-
}))
|
|
994
|
-
], LibP2PService.prototype, "validateBlockProposal", null);
|