@aztec/p2p 0.0.1-commit.f2ce05ee → 0.0.1-commit.f504929
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/client/factory.d.ts +8 -8
- package/dest/client/factory.d.ts.map +1 -1
- package/dest/client/factory.js +37 -13
- package/dest/client/interface.d.ts +39 -33
- package/dest/client/interface.d.ts.map +1 -1
- package/dest/client/p2p_client.d.ts +37 -50
- package/dest/client/p2p_client.d.ts.map +1 -1
- package/dest/client/p2p_client.js +138 -199
- package/dest/client/test/tx_proposal_collector/proposal_tx_collector_worker.js +6 -7
- package/dest/config.d.ts +29 -15
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +39 -35
- package/dest/errors/tx-pool.error.d.ts +8 -0
- package/dest/errors/tx-pool.error.d.ts.map +1 -0
- package/dest/errors/tx-pool.error.js +9 -0
- package/dest/index.d.ts +2 -1
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +1 -0
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +21 -12
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.js +72 -38
- 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 +53 -53
- package/dest/mem_pools/attestation_pool/index.d.ts +2 -2
- package/dest/mem_pools/attestation_pool/index.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/index.js +1 -1
- package/dest/mem_pools/attestation_pool/mocks.d.ts +2 -2
- package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/mocks.js +2 -2
- package/dest/mem_pools/index.d.ts +2 -1
- package/dest/mem_pools/index.d.ts.map +1 -1
- package/dest/mem_pools/instrumentation.d.ts +4 -2
- package/dest/mem_pools/instrumentation.d.ts.map +1 -1
- package/dest/mem_pools/instrumentation.js +16 -14
- package/dest/mem_pools/interface.d.ts +3 -3
- package/dest/mem_pools/interface.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.js +3 -3
- package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts +104 -0
- package/dest/mem_pools/tx_pool_v2/deleted_pool.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/deleted_pool.js +251 -0
- package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts +3 -3
- package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/eviction_manager.js +18 -9
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.js +5 -2
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts +3 -3
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.js +12 -4
- package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts +2 -2
- package/dest/mem_pools/tx_pool_v2/eviction/index.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/index.js +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts +48 -5
- package/dest/mem_pools/tx_pool_v2/eviction/interfaces.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/interfaces.js +8 -0
- package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.js +7 -5
- package/dest/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.js +7 -5
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts +2 -2
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.js +14 -6
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts +4 -4
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.js +14 -4
- package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts +3 -3
- package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.js +2 -2
- package/dest/mem_pools/tx_pool_v2/index.d.ts +3 -2
- package/dest/mem_pools/tx_pool_v2/index.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/index.js +2 -1
- package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts +15 -0
- package/dest/mem_pools/tx_pool_v2/instrumentation.d.ts.map +1 -0
- package/dest/mem_pools/tx_pool_v2/instrumentation.js +43 -0
- package/dest/mem_pools/tx_pool_v2/interfaces.d.ts +22 -8
- package/dest/mem_pools/tx_pool_v2/interfaces.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/interfaces.js +4 -1
- package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts +56 -8
- package/dest/mem_pools/tx_pool_v2/tx_metadata.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/tx_metadata.js +108 -10
- package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts +12 -3
- package/dest/mem_pools/tx_pool_v2/tx_pool_indices.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/tx_pool_indices.js +36 -14
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts +9 -4
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2.js +14 -6
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts +14 -5
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool_v2/tx_pool_v2_impl.js +344 -184
- package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts +2 -1
- package/dest/msg_validators/proposal_validator/block_proposal_validator.d.ts.map +1 -1
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts +2 -1
- package/dest/msg_validators/proposal_validator/checkpoint_proposal_validator.d.ts.map +1 -1
- package/dest/msg_validators/proposal_validator/proposal_validator.d.ts +3 -1
- package/dest/msg_validators/proposal_validator/proposal_validator.d.ts.map +1 -1
- package/dest/msg_validators/proposal_validator/proposal_validator.js +10 -0
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts +2 -1
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.d.ts.map +1 -1
- package/dest/msg_validators/proposal_validator/proposal_validator_test_suite.js +166 -0
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts +4 -4
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.js +3 -3
- package/dest/msg_validators/tx_validator/allowed_public_setup.d.ts +2 -1
- package/dest/msg_validators/tx_validator/allowed_public_setup.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/allowed_public_setup.js +25 -10
- package/dest/msg_validators/tx_validator/block_header_validator.d.ts +16 -3
- package/dest/msg_validators/tx_validator/block_header_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/block_header_validator.js +1 -1
- package/dest/msg_validators/tx_validator/double_spend_validator.d.ts +13 -3
- package/dest/msg_validators/tx_validator/double_spend_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/double_spend_validator.js +4 -4
- package/dest/msg_validators/tx_validator/factory.d.ts +114 -6
- package/dest/msg_validators/tx_validator/factory.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/factory.js +219 -58
- package/dest/msg_validators/tx_validator/gas_validator.d.ts +58 -3
- package/dest/msg_validators/tx_validator/gas_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/gas_validator.js +73 -36
- 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/nullifier_cache.d.ts +14 -0
- package/dest/msg_validators/tx_validator/nullifier_cache.d.ts.map +1 -0
- package/dest/msg_validators/tx_validator/nullifier_cache.js +24 -0
- package/dest/msg_validators/tx_validator/phases_validator.d.ts +2 -2
- package/dest/msg_validators/tx_validator/phases_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/phases_validator.js +25 -24
- package/dest/msg_validators/tx_validator/timestamp_validator.d.ts +20 -4
- package/dest/msg_validators/tx_validator/timestamp_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/timestamp_validator.js +6 -6
- package/dest/services/dummy_service.d.ts +9 -5
- package/dest/services/dummy_service.d.ts.map +1 -1
- package/dest/services/dummy_service.js +7 -4
- package/dest/services/encoding.d.ts +3 -3
- package/dest/services/encoding.d.ts.map +1 -1
- package/dest/services/encoding.js +11 -10
- package/dest/services/gossipsub/topic_score_params.d.ts +18 -6
- package/dest/services/gossipsub/topic_score_params.d.ts.map +1 -1
- package/dest/services/gossipsub/topic_score_params.js +32 -10
- package/dest/services/libp2p/libp2p_service.d.ts +25 -14
- package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
- package/dest/services/libp2p/libp2p_service.js +143 -114
- package/dest/services/peer-manager/metrics.d.ts +3 -1
- package/dest/services/peer-manager/metrics.d.ts.map +1 -1
- package/dest/services/peer-manager/metrics.js +6 -0
- package/dest/services/peer-manager/peer_manager.d.ts +1 -1
- package/dest/services/peer-manager/peer_manager.d.ts.map +1 -1
- package/dest/services/peer-manager/peer_manager.js +2 -1
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts +4 -3
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.d.ts.map +1 -1
- package/dest/services/reqresp/batch-tx-requester/batch_tx_requester.js +19 -46
- package/dest/services/reqresp/batch-tx-requester/interface.d.ts +2 -6
- package/dest/services/reqresp/batch-tx-requester/interface.d.ts.map +1 -1
- package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts +10 -13
- package/dest/services/reqresp/batch-tx-requester/missing_txs.d.ts.map +1 -1
- package/dest/services/reqresp/batch-tx-requester/missing_txs.js +25 -46
- package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts +17 -11
- package/dest/services/reqresp/batch-tx-requester/peer_collection.d.ts.map +1 -1
- package/dest/services/reqresp/batch-tx-requester/peer_collection.js +49 -15
- package/dest/services/reqresp/batch-tx-requester/tx_validator.js +2 -2
- package/dest/services/reqresp/interface.d.ts +10 -1
- package/dest/services/reqresp/interface.d.ts.map +1 -1
- package/dest/services/reqresp/interface.js +15 -1
- package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts +3 -3
- package/dest/services/reqresp/protocols/block_txs/block_txs_handler.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/block_txs/block_txs_reqresp.d.ts +7 -1
- 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 +15 -0
- package/dest/services/reqresp/protocols/tx.d.ts +7 -1
- package/dest/services/reqresp/protocols/tx.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/tx.js +20 -0
- package/dest/services/reqresp/reqresp.d.ts +1 -1
- package/dest/services/reqresp/reqresp.d.ts.map +1 -1
- package/dest/services/reqresp/reqresp.js +13 -5
- package/dest/services/service.d.ts +22 -3
- package/dest/services/service.d.ts.map +1 -1
- package/dest/services/tx_collection/config.d.ts +19 -1
- package/dest/services/tx_collection/config.d.ts.map +1 -1
- package/dest/services/tx_collection/config.js +46 -0
- package/dest/services/tx_collection/fast_tx_collection.d.ts +3 -1
- package/dest/services/tx_collection/fast_tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/fast_tx_collection.js +56 -36
- package/dest/services/tx_collection/file_store_tx_collection.d.ts +53 -0
- package/dest/services/tx_collection/file_store_tx_collection.d.ts.map +1 -0
- package/dest/services/tx_collection/file_store_tx_collection.js +167 -0
- package/dest/services/tx_collection/file_store_tx_source.d.ts +37 -0
- package/dest/services/tx_collection/file_store_tx_source.d.ts.map +1 -0
- package/dest/services/tx_collection/file_store_tx_source.js +90 -0
- package/dest/services/tx_collection/index.d.ts +2 -1
- package/dest/services/tx_collection/index.d.ts.map +1 -1
- package/dest/services/tx_collection/index.js +1 -0
- package/dest/services/tx_collection/instrumentation.d.ts +1 -1
- package/dest/services/tx_collection/instrumentation.d.ts.map +1 -1
- package/dest/services/tx_collection/instrumentation.js +2 -1
- package/dest/services/tx_collection/missing_txs_tracker.d.ts +32 -0
- package/dest/services/tx_collection/missing_txs_tracker.d.ts.map +1 -0
- package/dest/services/tx_collection/missing_txs_tracker.js +27 -0
- package/dest/services/tx_collection/proposal_tx_collector.d.ts +7 -6
- package/dest/services/tx_collection/proposal_tx_collector.d.ts.map +1 -1
- package/dest/services/tx_collection/proposal_tx_collector.js +5 -4
- package/dest/services/tx_collection/slow_tx_collection.d.ts +7 -3
- package/dest/services/tx_collection/slow_tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/slow_tx_collection.js +60 -26
- package/dest/services/tx_collection/tx_collection.d.ts +23 -10
- package/dest/services/tx_collection/tx_collection.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_collection.js +75 -3
- package/dest/services/tx_collection/tx_collection_sink.d.ts +18 -8
- package/dest/services/tx_collection/tx_collection_sink.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_collection_sink.js +26 -29
- package/dest/services/tx_collection/tx_source.d.ts +8 -3
- package/dest/services/tx_collection/tx_source.d.ts.map +1 -1
- package/dest/services/tx_collection/tx_source.js +19 -2
- package/dest/services/tx_file_store/config.d.ts +1 -3
- package/dest/services/tx_file_store/config.d.ts.map +1 -1
- package/dest/services/tx_file_store/config.js +0 -4
- package/dest/services/tx_file_store/tx_file_store.d.ts +4 -3
- package/dest/services/tx_file_store/tx_file_store.d.ts.map +1 -1
- package/dest/services/tx_file_store/tx_file_store.js +9 -6
- package/dest/services/tx_provider.d.ts +4 -4
- package/dest/services/tx_provider.d.ts.map +1 -1
- package/dest/services/tx_provider.js +9 -8
- package/dest/test-helpers/make-test-p2p-clients.d.ts +7 -8
- package/dest/test-helpers/make-test-p2p-clients.d.ts.map +1 -1
- package/dest/test-helpers/make-test-p2p-clients.js +1 -2
- package/dest/test-helpers/mock-pubsub.d.ts +30 -4
- package/dest/test-helpers/mock-pubsub.d.ts.map +1 -1
- package/dest/test-helpers/mock-pubsub.js +105 -4
- package/dest/test-helpers/reqresp-nodes.d.ts +2 -3
- package/dest/test-helpers/reqresp-nodes.d.ts.map +1 -1
- package/dest/test-helpers/reqresp-nodes.js +2 -2
- package/dest/test-helpers/testbench-utils.d.ts +35 -24
- package/dest/test-helpers/testbench-utils.d.ts.map +1 -1
- package/dest/test-helpers/testbench-utils.js +93 -35
- package/dest/testbench/p2p_client_testbench_worker.d.ts +2 -2
- package/dest/testbench/p2p_client_testbench_worker.d.ts.map +1 -1
- package/dest/testbench/p2p_client_testbench_worker.js +14 -14
- package/dest/util.d.ts +2 -2
- package/dest/util.d.ts.map +1 -1
- package/package.json +14 -14
- package/src/client/factory.ts +71 -23
- package/src/client/interface.ts +43 -33
- package/src/client/p2p_client.ts +164 -243
- package/src/client/test/tx_proposal_collector/proposal_tx_collector_worker.ts +19 -10
- package/src/config.ts +60 -42
- package/src/errors/tx-pool.error.ts +12 -0
- package/src/index.ts +1 -0
- package/src/mem_pools/attestation_pool/attestation_pool.ts +100 -48
- package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +57 -53
- package/src/mem_pools/attestation_pool/index.ts +3 -3
- package/src/mem_pools/attestation_pool/mocks.ts +2 -1
- package/src/mem_pools/index.ts +3 -0
- package/src/mem_pools/instrumentation.ts +17 -13
- package/src/mem_pools/interface.ts +2 -2
- package/src/mem_pools/tx_pool/README.md +1 -1
- package/src/mem_pools/tx_pool/eviction/invalid_txs_after_mining_rule.ts +3 -3
- package/src/mem_pools/tx_pool_v2/README.md +76 -10
- package/src/mem_pools/tx_pool_v2/deleted_pool.ts +321 -0
- package/src/mem_pools/tx_pool_v2/eviction/eviction_manager.ts +21 -8
- package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_eviction_rule.ts +5 -2
- package/src/mem_pools/tx_pool_v2/eviction/fee_payer_balance_pre_add_rule.ts +18 -4
- package/src/mem_pools/tx_pool_v2/eviction/index.ts +4 -0
- package/src/mem_pools/tx_pool_v2/eviction/interfaces.ts +49 -4
- package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_mining_rule.ts +5 -5
- package/src/mem_pools/tx_pool_v2/eviction/invalid_txs_after_reorg_rule.ts +5 -5
- package/src/mem_pools/tx_pool_v2/eviction/low_priority_eviction_rule.ts +14 -9
- package/src/mem_pools/tx_pool_v2/eviction/low_priority_pre_add_rule.ts +24 -6
- package/src/mem_pools/tx_pool_v2/eviction/nullifier_conflict_rule.ts +3 -3
- package/src/mem_pools/tx_pool_v2/index.ts +2 -1
- package/src/mem_pools/tx_pool_v2/instrumentation.ts +69 -0
- package/src/mem_pools/tx_pool_v2/interfaces.ts +23 -8
- package/src/mem_pools/tx_pool_v2/tx_metadata.ts +153 -17
- package/src/mem_pools/tx_pool_v2/tx_pool_indices.ts +43 -16
- package/src/mem_pools/tx_pool_v2/tx_pool_v2.ts +21 -7
- package/src/mem_pools/tx_pool_v2/tx_pool_v2_impl.ts +388 -182
- package/src/msg_validators/proposal_validator/block_proposal_validator.ts +1 -1
- package/src/msg_validators/proposal_validator/checkpoint_proposal_validator.ts +1 -1
- package/src/msg_validators/proposal_validator/proposal_validator.ts +15 -1
- package/src/msg_validators/proposal_validator/proposal_validator_test_suite.ts +144 -1
- package/src/msg_validators/tx_validator/README.md +115 -0
- package/src/msg_validators/tx_validator/aggregate_tx_validator.ts +5 -5
- package/src/msg_validators/tx_validator/allowed_public_setup.ts +27 -13
- package/src/msg_validators/tx_validator/block_header_validator.ts +15 -3
- package/src/msg_validators/tx_validator/double_spend_validator.ts +11 -6
- package/src/msg_validators/tx_validator/factory.ts +353 -77
- package/src/msg_validators/tx_validator/gas_validator.ts +90 -27
- package/src/msg_validators/tx_validator/index.ts +1 -0
- package/src/msg_validators/tx_validator/nullifier_cache.ts +30 -0
- package/src/msg_validators/tx_validator/phases_validator.ts +25 -29
- package/src/msg_validators/tx_validator/timestamp_validator.ts +23 -18
- package/src/services/dummy_service.ts +12 -6
- package/src/services/encoding.ts +9 -9
- package/src/services/gossipsub/README.md +29 -14
- package/src/services/gossipsub/topic_score_params.ts +49 -13
- package/src/services/libp2p/libp2p_service.ts +153 -123
- package/src/services/peer-manager/metrics.ts +7 -0
- package/src/services/peer-manager/peer_manager.ts +2 -1
- package/src/services/reqresp/batch-tx-requester/batch_tx_requester.ts +20 -48
- package/src/services/reqresp/batch-tx-requester/interface.ts +1 -5
- package/src/services/reqresp/batch-tx-requester/missing_txs.ts +23 -71
- package/src/services/reqresp/batch-tx-requester/peer_collection.ts +63 -24
- package/src/services/reqresp/batch-tx-requester/tx_validator.ts +2 -2
- package/src/services/reqresp/interface.ts +26 -1
- package/src/services/reqresp/protocols/block_txs/block_txs_handler.ts +2 -2
- package/src/services/reqresp/protocols/block_txs/block_txs_reqresp.ts +17 -0
- package/src/services/reqresp/protocols/tx.ts +22 -0
- package/src/services/reqresp/reqresp.ts +16 -4
- package/src/services/service.ts +31 -2
- package/src/services/tx_collection/config.ts +68 -0
- package/src/services/tx_collection/fast_tx_collection.ts +65 -32
- package/src/services/tx_collection/file_store_tx_collection.ts +202 -0
- package/src/services/tx_collection/file_store_tx_source.ts +117 -0
- package/src/services/tx_collection/index.ts +1 -0
- package/src/services/tx_collection/instrumentation.ts +7 -1
- package/src/services/tx_collection/missing_txs_tracker.ts +52 -0
- package/src/services/tx_collection/proposal_tx_collector.ts +8 -7
- package/src/services/tx_collection/slow_tx_collection.ts +66 -33
- package/src/services/tx_collection/tx_collection.ts +113 -16
- package/src/services/tx_collection/tx_collection_sink.ts +30 -34
- package/src/services/tx_collection/tx_source.ts +22 -3
- package/src/services/tx_file_store/config.ts +0 -6
- package/src/services/tx_file_store/tx_file_store.ts +10 -8
- package/src/services/tx_provider.ts +10 -9
- package/src/test-helpers/make-test-p2p-clients.ts +3 -5
- package/src/test-helpers/mock-pubsub.ts +146 -9
- package/src/test-helpers/reqresp-nodes.ts +2 -5
- package/src/test-helpers/testbench-utils.ts +108 -40
- package/src/testbench/p2p_client_testbench_worker.ts +23 -20
- package/src/util.ts +7 -1
|
@@ -1,10 +1,13 @@
|
|
|
1
|
+
import { BlockNumber } from '@aztec/foundation/branded-types';
|
|
1
2
|
import { ProtocolContractAddress } from '@aztec/protocol-contracts';
|
|
2
3
|
import { computeFeePayerBalanceStorageSlot } from '@aztec/protocol-contracts/fee-juice';
|
|
3
4
|
import { AztecAddress } from '@aztec/stdlib/aztec-address';
|
|
4
5
|
import { DatabasePublicStateSource } from '@aztec/stdlib/trees';
|
|
5
6
|
import { Tx, TxHash } from '@aztec/stdlib/tx';
|
|
6
7
|
import { TxArchive } from './archive/index.js';
|
|
7
|
-
import {
|
|
8
|
+
import { DeletedPool } from './deleted_pool.js';
|
|
9
|
+
import { EvictionManager, FeePayerBalanceEvictionRule, FeePayerBalancePreAddRule, InvalidTxsAfterMiningRule, InvalidTxsAfterReorgRule, LowPriorityEvictionRule, LowPriorityPreAddRule, NullifierConflictRule, TxPoolRejectionCode } from './eviction/index.js';
|
|
10
|
+
import { TxPoolV2Instrumentation } from './instrumentation.js';
|
|
8
11
|
import { DEFAULT_TX_POOL_V2_CONFIG } from './interfaces.js';
|
|
9
12
|
import { buildTxMetaData, checkNullifierConflict } from './tx_metadata.js';
|
|
10
13
|
import { TxPoolIndices } from './tx_pool_indices.js';
|
|
@@ -19,26 +22,33 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
19
22
|
// === Dependencies ===
|
|
20
23
|
#l2BlockSource;
|
|
21
24
|
#worldStateSynchronizer;
|
|
22
|
-
#
|
|
25
|
+
#createTxValidator;
|
|
23
26
|
// === In-Memory Indices ===
|
|
24
27
|
#indices = new TxPoolIndices();
|
|
25
28
|
// === Config & Services ===
|
|
26
29
|
#config;
|
|
27
30
|
#archive;
|
|
31
|
+
#deletedPool;
|
|
28
32
|
#evictionManager;
|
|
33
|
+
#dateProvider;
|
|
34
|
+
#instrumentation;
|
|
35
|
+
#evictedTxHashes = new Set();
|
|
29
36
|
#log;
|
|
30
37
|
#callbacks;
|
|
31
|
-
constructor(store, archiveStore, deps, callbacks, config = {}, log){
|
|
38
|
+
constructor(store, archiveStore, deps, callbacks, telemetry, config = {}, dateProvider, log){
|
|
32
39
|
this.#store = store;
|
|
33
40
|
this.#txsDB = store.openMap('txs');
|
|
34
41
|
this.#l2BlockSource = deps.l2BlockSource;
|
|
35
42
|
this.#worldStateSynchronizer = deps.worldStateSynchronizer;
|
|
36
|
-
this.#
|
|
43
|
+
this.#createTxValidator = deps.createTxValidator;
|
|
37
44
|
this.#config = {
|
|
38
45
|
...DEFAULT_TX_POOL_V2_CONFIG,
|
|
39
46
|
...config
|
|
40
47
|
};
|
|
41
48
|
this.#archive = new TxArchive(archiveStore, this.#config.archivedTxLimit, log);
|
|
49
|
+
this.#deletedPool = new DeletedPool(store, this.#txsDB, log);
|
|
50
|
+
this.#dateProvider = dateProvider;
|
|
51
|
+
this.#instrumentation = new TxPoolV2Instrumentation(telemetry, ()=>this.#indices.getTotalMetadataBytes());
|
|
42
52
|
this.#log = log;
|
|
43
53
|
this.#callbacks = callbacks;
|
|
44
54
|
// Setup eviction manager with rules
|
|
@@ -69,7 +79,9 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
69
79
|
* Note: Protected status is lost on restart. All non-mined txs are rebuilt as pending
|
|
70
80
|
* by running pre-add rules to resolve nullifier conflicts, balance checks, and pool size limits.
|
|
71
81
|
*/ async hydrateFromDatabase() {
|
|
72
|
-
// Step
|
|
82
|
+
// Step 0: Hydrate deleted pool state
|
|
83
|
+
await this.#deletedPool.hydrateFromDatabase();
|
|
84
|
+
// Step 1: Load all transactions from DB (excluding soft-deleted)
|
|
73
85
|
const { loaded, errors: deserializationErrors } = await this.#loadAllTxsFromDb();
|
|
74
86
|
// Step 2: Check mined status for each tx
|
|
75
87
|
await this.#markMinedStatusBatch(loaded.map((l)=>l.meta));
|
|
@@ -84,7 +96,7 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
84
96
|
}
|
|
85
97
|
}
|
|
86
98
|
// Step 4: Validate non-mined transactions
|
|
87
|
-
const { valid, invalid } = await this.#
|
|
99
|
+
const { valid, invalid } = await this.#revalidateMetadata(nonMined.map((e)=>e.meta), 'on startup');
|
|
88
100
|
// Step 5: Populate mined indices (these don't need conflict resolution)
|
|
89
101
|
for (const meta of mined){
|
|
90
102
|
this.#indices.addMined(meta);
|
|
@@ -106,14 +118,42 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
106
118
|
await this.#txsDB.delete(txHashStr);
|
|
107
119
|
}
|
|
108
120
|
});
|
|
109
|
-
this.#log.info(`Deleted ${toDelete.length} invalid/rejected transactions on startup
|
|
121
|
+
this.#log.info(`Deleted ${toDelete.length} invalid/rejected transactions on startup`, {
|
|
122
|
+
txHashes: toDelete
|
|
123
|
+
});
|
|
110
124
|
}
|
|
111
125
|
async addPendingTxs(txs, opts) {
|
|
112
126
|
const accepted = [];
|
|
113
127
|
const ignored = [];
|
|
114
128
|
const rejected = [];
|
|
129
|
+
const errors = new Map();
|
|
115
130
|
const acceptedPending = new Set();
|
|
131
|
+
// Phase 1: Pre-compute all throwable I/O outside the transaction.
|
|
132
|
+
// If any pre-computation throws, the entire call fails before mutations happen.
|
|
133
|
+
const precomputed = new Map();
|
|
134
|
+
const validator = await this.#createTxValidator();
|
|
135
|
+
for (const tx of txs){
|
|
136
|
+
const txHash = tx.getTxHash();
|
|
137
|
+
const txHashStr = txHash.toString();
|
|
138
|
+
const meta = await buildTxMetaData(tx);
|
|
139
|
+
const minedBlockId = await this.#getMinedBlockId(txHash);
|
|
140
|
+
// Validate non-mined txs (mined and pre-protected txs bypass validation inside the transaction)
|
|
141
|
+
let isValid = true;
|
|
142
|
+
if (!minedBlockId) {
|
|
143
|
+
isValid = await this.#validateMeta(meta, validator);
|
|
144
|
+
}
|
|
145
|
+
precomputed.set(txHashStr, {
|
|
146
|
+
meta,
|
|
147
|
+
minedBlockId,
|
|
148
|
+
isValid
|
|
149
|
+
});
|
|
150
|
+
}
|
|
151
|
+
// Phase 2: Apply mutations inside the transaction using only pre-computed results,
|
|
152
|
+
// in-memory reads, and buffered DB writes. Nothing here can throw an unhandled exception.
|
|
116
153
|
const poolAccess = this.#createPreAddPoolAccess();
|
|
154
|
+
const preAddContext = opts.feeComparisonOnly !== undefined ? {
|
|
155
|
+
feeComparisonOnly: opts.feeComparisonOnly
|
|
156
|
+
} : undefined;
|
|
117
157
|
await this.#store.transactionAsync(async ()=>{
|
|
118
158
|
for (const tx of txs){
|
|
119
159
|
const txHash = tx.getTxHash();
|
|
@@ -123,82 +163,112 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
123
163
|
ignored.push(txHash);
|
|
124
164
|
continue;
|
|
125
165
|
}
|
|
126
|
-
|
|
127
|
-
const minedBlockId = await this.#getMinedBlockId(txHash);
|
|
166
|
+
const { meta, minedBlockId, isValid } = precomputed.get(txHashStr);
|
|
128
167
|
const preProtectedSlot = this.#indices.getProtectionSlot(txHashStr);
|
|
129
168
|
if (minedBlockId) {
|
|
130
169
|
// Already mined - add directly (protection already set if pre-protected)
|
|
131
170
|
await this.#addTx(tx, {
|
|
132
171
|
mined: minedBlockId
|
|
133
|
-
}, opts);
|
|
172
|
+
}, opts, meta);
|
|
134
173
|
accepted.push(txHash);
|
|
135
174
|
} else if (preProtectedSlot !== undefined) {
|
|
136
175
|
// Pre-protected and not mined - add as protected (bypass validation)
|
|
137
176
|
await this.#addTx(tx, {
|
|
138
177
|
protected: preProtectedSlot
|
|
139
|
-
}, opts);
|
|
178
|
+
}, opts, meta);
|
|
140
179
|
accepted.push(txHash);
|
|
180
|
+
} else if (!isValid) {
|
|
181
|
+
// Failed pre-computed validation
|
|
182
|
+
rejected.push(txHash);
|
|
141
183
|
} else {
|
|
142
|
-
// Regular pending tx -
|
|
143
|
-
const result = await this.#tryAddRegularPendingTx(tx, opts, poolAccess, acceptedPending, ignored);
|
|
184
|
+
// Regular pending tx - run pre-add rules using pre-computed metadata
|
|
185
|
+
const result = await this.#tryAddRegularPendingTx(tx, meta, opts, poolAccess, acceptedPending, ignored, errors, preAddContext);
|
|
144
186
|
if (result.status === 'accepted') {
|
|
145
187
|
acceptedPending.add(txHashStr);
|
|
146
|
-
} else if (result.status === 'rejected') {
|
|
147
|
-
rejected.push(txHash);
|
|
148
188
|
} else {
|
|
149
189
|
ignored.push(txHash);
|
|
150
190
|
}
|
|
151
191
|
}
|
|
152
192
|
}
|
|
193
|
+
// Run post-add eviction rules for pending txs (inside transaction for atomicity)
|
|
194
|
+
if (acceptedPending.size > 0) {
|
|
195
|
+
const feePayers = Array.from(acceptedPending).map((txHash)=>this.#indices.getMetadata(txHash).feePayer);
|
|
196
|
+
const uniqueFeePayers = new Set(feePayers);
|
|
197
|
+
await this.#evictionManager.evictAfterNewTxs(Array.from(acceptedPending), [
|
|
198
|
+
...uniqueFeePayers
|
|
199
|
+
]);
|
|
200
|
+
}
|
|
153
201
|
});
|
|
154
202
|
// Build final accepted list for pending txs (excludes intra-batch evictions)
|
|
155
203
|
for (const txHashStr of acceptedPending){
|
|
156
204
|
accepted.push(TxHash.fromString(txHashStr));
|
|
157
205
|
}
|
|
158
|
-
//
|
|
159
|
-
if (
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
]);
|
|
206
|
+
// Record metrics
|
|
207
|
+
if (ignored.length > 0) {
|
|
208
|
+
this.#instrumentation.recordIgnored(ignored.length);
|
|
209
|
+
}
|
|
210
|
+
if (rejected.length > 0) {
|
|
211
|
+
this.#instrumentation.recordRejected(rejected.length);
|
|
165
212
|
}
|
|
166
213
|
return {
|
|
167
214
|
accepted,
|
|
168
215
|
ignored,
|
|
169
|
-
rejected
|
|
216
|
+
rejected,
|
|
217
|
+
...errors.size > 0 ? {
|
|
218
|
+
errors
|
|
219
|
+
} : {}
|
|
170
220
|
};
|
|
171
221
|
}
|
|
172
|
-
/**
|
|
173
|
-
const
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
if (!await this.#validateTx(tx)) {
|
|
177
|
-
return {
|
|
178
|
-
status: 'rejected'
|
|
179
|
-
};
|
|
180
|
-
}
|
|
181
|
-
// Build metadata and run pre-add rules
|
|
182
|
-
const meta = await buildTxMetaData(tx);
|
|
183
|
-
const preAddResult = await this.#evictionManager.runPreAddRules(meta, poolAccess);
|
|
222
|
+
/** Adds a validated pending tx, running pre-add rules and evicting conflicts. */ async #tryAddRegularPendingTx(tx, precomputedMeta, opts, poolAccess, acceptedPending, ignored, errors, preAddContext) {
|
|
223
|
+
const txHashStr = tx.getTxHash().toString();
|
|
224
|
+
// Run pre-add rules
|
|
225
|
+
const preAddResult = await this.#evictionManager.runPreAddRules(precomputedMeta, poolAccess, preAddContext);
|
|
184
226
|
if (preAddResult.shouldIgnore) {
|
|
185
|
-
this.#log.debug(`Ignoring tx ${txHashStr}: ${preAddResult.reason}`);
|
|
227
|
+
this.#log.debug(`Ignoring tx ${txHashStr}: ${preAddResult.reason?.message ?? 'unknown reason'}`);
|
|
228
|
+
if (preAddResult.reason && preAddResult.reason.code !== TxPoolRejectionCode.INTERNAL_ERROR) {
|
|
229
|
+
errors.set(txHashStr, preAddResult.reason);
|
|
230
|
+
}
|
|
186
231
|
return {
|
|
187
232
|
status: 'ignored'
|
|
188
233
|
};
|
|
189
234
|
}
|
|
190
|
-
// Evict conflicts
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
235
|
+
// Evict conflicts, grouped by rule name for metrics
|
|
236
|
+
if (preAddResult.evictions && preAddResult.evictions.length > 0) {
|
|
237
|
+
const byReason = new Map();
|
|
238
|
+
for (const { txHash: evictHash, reason } of preAddResult.evictions){
|
|
239
|
+
const group = byReason.get(reason);
|
|
240
|
+
if (group) {
|
|
241
|
+
group.push(evictHash);
|
|
242
|
+
} else {
|
|
243
|
+
byReason.set(reason, [
|
|
244
|
+
evictHash
|
|
245
|
+
]);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
for (const [reason, hashes] of byReason){
|
|
249
|
+
await this.#evictTxs(hashes, reason);
|
|
198
250
|
}
|
|
251
|
+
for (const evictHashStr of preAddResult.txHashesToEvict){
|
|
252
|
+
this.#log.debug(`Evicted tx ${evictHashStr} due to higher-fee tx ${txHashStr}`, {
|
|
253
|
+
evictedTxHash: evictHashStr,
|
|
254
|
+
replacementTxHash: txHashStr
|
|
255
|
+
});
|
|
256
|
+
if (acceptedPending.has(evictHashStr)) {
|
|
257
|
+
// Evicted tx was from this batch - mark as ignored in result
|
|
258
|
+
acceptedPending.delete(evictHashStr);
|
|
259
|
+
ignored.push(TxHash.fromString(evictHashStr));
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
// Randomly drop the transaction for testing purposes (report as accepted so it propagates)
|
|
264
|
+
if (this.#config.dropTransactionsProbability > 0 && Math.random() < this.#config.dropTransactionsProbability) {
|
|
265
|
+
this.#log.debug(`Dropping tx ${txHashStr} (simulated drop for testing)`);
|
|
266
|
+
return {
|
|
267
|
+
status: 'accepted'
|
|
268
|
+
};
|
|
199
269
|
}
|
|
200
270
|
// Add the transaction
|
|
201
|
-
await this.#addTx(tx, 'pending', opts);
|
|
271
|
+
await this.#addTx(tx, 'pending', opts, precomputedMeta);
|
|
202
272
|
return {
|
|
203
273
|
status: 'accepted'
|
|
204
274
|
};
|
|
@@ -209,12 +279,7 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
209
279
|
if (this.#indices.has(txHashStr)) {
|
|
210
280
|
return 'ignored';
|
|
211
281
|
}
|
|
212
|
-
//
|
|
213
|
-
const validationResult = await this.#pendingTxValidator.validateTx(tx);
|
|
214
|
-
if (validationResult.result !== 'valid') {
|
|
215
|
-
return 'rejected';
|
|
216
|
-
}
|
|
217
|
-
// Build metadata and use pre-add rules
|
|
282
|
+
// Build metadata and check pre-add rules
|
|
218
283
|
const meta = await buildTxMetaData(tx);
|
|
219
284
|
const poolAccess = this.#createPreAddPoolAccess();
|
|
220
285
|
const preAddResult = await this.#evictionManager.runPreAddRules(meta, poolAccess);
|
|
@@ -251,20 +316,53 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
251
316
|
}
|
|
252
317
|
});
|
|
253
318
|
}
|
|
254
|
-
protectTxs(txHashes, block) {
|
|
319
|
+
async protectTxs(txHashes, block) {
|
|
255
320
|
const slotNumber = block.globalVariables.slotNumber;
|
|
256
321
|
const missing = [];
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
322
|
+
let softDeletedHits = 0;
|
|
323
|
+
let missingPreviouslyEvicted = 0;
|
|
324
|
+
await this.#store.transactionAsync(async ()=>{
|
|
325
|
+
for (const txHash of txHashes){
|
|
326
|
+
const txHashStr = txHash.toString();
|
|
327
|
+
if (this.#indices.has(txHashStr)) {
|
|
328
|
+
// Update protection for existing tx
|
|
329
|
+
this.#indices.updateProtection(txHashStr, slotNumber);
|
|
330
|
+
} else if (this.#deletedPool.isSoftDeleted(txHashStr)) {
|
|
331
|
+
// Resurrect soft-deleted tx as protected
|
|
332
|
+
const buffer = await this.#txsDB.getAsync(txHashStr);
|
|
333
|
+
if (buffer) {
|
|
334
|
+
const tx = Tx.fromBuffer(buffer);
|
|
335
|
+
await this.#addTx(tx, {
|
|
336
|
+
protected: slotNumber
|
|
337
|
+
});
|
|
338
|
+
softDeletedHits++;
|
|
339
|
+
} else {
|
|
340
|
+
// Data missing despite soft-delete flag — treat as truly missing
|
|
341
|
+
this.#indices.setProtection(txHashStr, slotNumber);
|
|
342
|
+
missing.push(txHash);
|
|
343
|
+
}
|
|
344
|
+
} else {
|
|
345
|
+
// Truly missing — pre-record protection for tx we don't have yet
|
|
346
|
+
this.#indices.setProtection(txHashStr, slotNumber);
|
|
347
|
+
missing.push(txHash);
|
|
348
|
+
if (this.#evictedTxHashes.has(txHashStr)) {
|
|
349
|
+
missingPreviouslyEvicted++;
|
|
350
|
+
}
|
|
351
|
+
}
|
|
266
352
|
}
|
|
353
|
+
});
|
|
354
|
+
// Record metrics
|
|
355
|
+
if (softDeletedHits > 0) {
|
|
356
|
+
this.#instrumentation.recordSoftDeletedHits(softDeletedHits);
|
|
357
|
+
}
|
|
358
|
+
if (missing.length > 0) {
|
|
359
|
+
this.#log.debug(`protectTxs missing tx hashes: ${missing.map((h)=>h.toString()).join(', ')}`);
|
|
360
|
+
this.#instrumentation.recordMissingOnProtect(missing.length);
|
|
361
|
+
}
|
|
362
|
+
if (missingPreviouslyEvicted > 0) {
|
|
363
|
+
this.#instrumentation.recordMissingPreviouslyEvicted(missingPreviouslyEvicted);
|
|
267
364
|
}
|
|
365
|
+
this.#log.info(`Protected ${txHashes.length} txs, missing: ${missing.length}, soft-deleted hits: ${softDeletedHits}`);
|
|
268
366
|
return missing;
|
|
269
367
|
}
|
|
270
368
|
async addMinedTxs(txs, block, opts) {
|
|
@@ -283,6 +381,7 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
283
381
|
mined: blockId
|
|
284
382
|
}, opts);
|
|
285
383
|
}
|
|
384
|
+
await this.#deletedPool.clearIfMinedHigher(txHashStr, blockId.number);
|
|
286
385
|
}
|
|
287
386
|
});
|
|
288
387
|
}
|
|
@@ -302,44 +401,53 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
302
401
|
found.push(meta);
|
|
303
402
|
}
|
|
304
403
|
}
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
404
|
+
await this.#store.transactionAsync(async ()=>{
|
|
405
|
+
// Step 4: Mark txs as mined (only those we have in the pool)
|
|
406
|
+
for (const meta of found){
|
|
407
|
+
this.#indices.markAsMined(meta, blockId);
|
|
408
|
+
await this.#deletedPool.clearIfMinedHigher(meta.txHash, blockId.number);
|
|
409
|
+
}
|
|
410
|
+
// Step 5: Run post-event eviction rules (inside transaction for atomicity)
|
|
411
|
+
await this.#evictionManager.evictAfterNewBlock(block.header, nullifiers, feePayers);
|
|
412
|
+
});
|
|
413
|
+
if (found.length > 0) {
|
|
414
|
+
this.#callbacks.onTxsMined(found.map((m)=>m.txHash));
|
|
308
415
|
}
|
|
309
|
-
// Step 5: Run eviction rules (remove pending txs with conflicting nullifiers/expired timestamps)
|
|
310
|
-
await this.#evictionManager.evictAfterNewBlock(block.header, nullifiers, feePayers);
|
|
311
416
|
this.#log.info(`Marked ${found.length} txs as mined in block ${blockId.number}`);
|
|
312
417
|
}
|
|
313
418
|
async prepareForSlot(slotNumber) {
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
419
|
+
await this.#store.transactionAsync(async ()=>{
|
|
420
|
+
// Step 0: Clean up slot-deleted txs from previous slots
|
|
421
|
+
await this.#deletedPool.cleanupSlotDeleted(slotNumber);
|
|
422
|
+
// Step 1: Find expired protected txs
|
|
423
|
+
const expiredProtected = this.#indices.findExpiredProtectedTxs(slotNumber);
|
|
424
|
+
// Step 2: Clear protection for all expired entries (including those without metadata)
|
|
425
|
+
this.#indices.clearProtection(expiredProtected);
|
|
426
|
+
// Step 3: Filter to only txs that have metadata and are not mined
|
|
427
|
+
const txsToRestore = this.#indices.filterRestorable(expiredProtected);
|
|
428
|
+
if (txsToRestore.length === 0) {
|
|
429
|
+
this.#log.debug(`Preparing for slot ${slotNumber}, no txs to unprotect`);
|
|
430
|
+
return;
|
|
431
|
+
}
|
|
432
|
+
this.#log.info(`Preparing for slot ${slotNumber}: unprotecting ${txsToRestore.length} txs`);
|
|
433
|
+
// Step 4: Validate for pending pool
|
|
434
|
+
const { valid, invalid } = await this.#revalidateMetadata(txsToRestore, 'during prepareForSlot');
|
|
435
|
+
// Step 5: Resolve nullifier conflicts and add winners to pending indices
|
|
436
|
+
const { added, toEvict } = this.#applyNullifierConflictResolution(valid);
|
|
437
|
+
// Step 6: Delete invalid txs and evict conflict losers
|
|
438
|
+
await this.#deleteTxsBatch(invalid);
|
|
439
|
+
await this.#evictTxs(toEvict, 'NullifierConflict');
|
|
440
|
+
// Step 7: Run eviction rules (enforce pool size limit)
|
|
441
|
+
if (added.length > 0) {
|
|
442
|
+
const feePayers = added.map((meta)=>meta.feePayer);
|
|
443
|
+
const uniqueFeePayers = new Set(feePayers);
|
|
444
|
+
await this.#evictionManager.evictAfterNewTxs(added.map((m)=>m.txHash), [
|
|
445
|
+
...uniqueFeePayers
|
|
446
|
+
]);
|
|
447
|
+
}
|
|
448
|
+
});
|
|
341
449
|
}
|
|
342
|
-
async handlePrunedBlocks(latestBlock) {
|
|
450
|
+
async handlePrunedBlocks(latestBlock, options) {
|
|
343
451
|
// Step 1: Find transactions mined after the prune point
|
|
344
452
|
const txsToUnmine = this.#indices.findTxsMinedAfter(latestBlock.number);
|
|
345
453
|
if (txsToUnmine.length === 0) {
|
|
@@ -347,54 +455,81 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
347
455
|
return;
|
|
348
456
|
}
|
|
349
457
|
this.#log.info(`Handling prune to block ${latestBlock.number}: un-mining ${txsToUnmine.length} txs`);
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
458
|
+
await this.#store.transactionAsync(async ()=>{
|
|
459
|
+
// Step 2: Mark ALL un-mined txs with their original mined block number
|
|
460
|
+
// This ensures they get soft-deleted if removed later, and only hard-deleted
|
|
461
|
+
// when their original mined block is finalized
|
|
462
|
+
await this.#deletedPool.markFromPrunedBlock(txsToUnmine.map((m)=>({
|
|
463
|
+
txHash: m.txHash,
|
|
464
|
+
minedAtBlock: BlockNumber(m.minedL2BlockId.number)
|
|
465
|
+
})));
|
|
466
|
+
// Step 3: Unmine - clear mined status from metadata
|
|
467
|
+
for (const meta of txsToUnmine){
|
|
468
|
+
this.#indices.markAsUnmined(meta);
|
|
469
|
+
}
|
|
470
|
+
// If deleteAllTxs is set (epoch prune), delete all un-mined txs and return early
|
|
471
|
+
if (options?.deleteAllTxs) {
|
|
472
|
+
const allTxHashes = txsToUnmine.map((m)=>m.txHash);
|
|
473
|
+
await this.#deleteTxsBatch(allTxHashes);
|
|
474
|
+
this.#log.info(`Handled prune to block ${latestBlock.number} with deleteAllTxs: deleted ${allTxHashes.length} txs`);
|
|
475
|
+
return;
|
|
476
|
+
}
|
|
477
|
+
// Step 4: Filter out protected txs (they'll be handled by prepareForSlot)
|
|
478
|
+
const unprotectedTxs = this.#indices.filterUnprotected(txsToUnmine);
|
|
479
|
+
// Step 5: Validate for pending pool
|
|
480
|
+
const { valid, invalid } = await this.#revalidateMetadata(unprotectedTxs, 'during handlePrunedBlocks');
|
|
481
|
+
// Step 6: Resolve nullifier conflicts and add winners to pending indices
|
|
482
|
+
const { toEvict } = this.#applyNullifierConflictResolution(valid);
|
|
483
|
+
// Step 7: Delete invalid txs and evict conflict losers
|
|
484
|
+
await this.#deleteTxsBatch(invalid);
|
|
485
|
+
await this.#evictTxs(toEvict, 'NullifierConflict');
|
|
486
|
+
this.#log.info(`Handled prune to block ${latestBlock.number}: ${valid.length} txs restored to pending, ${invalid.length} invalid, ${toEvict.length} evicted due to nullifier conflicts`, {
|
|
487
|
+
txHashesRestored: valid.map((m)=>m.txHash),
|
|
488
|
+
txHashesInvalid: invalid,
|
|
489
|
+
txHashesEvicted: toEvict
|
|
490
|
+
});
|
|
491
|
+
// Step 8: Run eviction rules for ALL pending txs (not just restored ones)
|
|
492
|
+
// This handles cases like existing pending txs with invalid fee payer balances
|
|
493
|
+
await this.#evictionManager.evictAfterChainPrune(latestBlock.number);
|
|
494
|
+
});
|
|
368
495
|
}
|
|
369
496
|
async handleFailedExecution(txHashes) {
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
497
|
+
await this.#store.transactionAsync(async ()=>{
|
|
498
|
+
await this.#deleteTxsBatch(txHashes.map((h)=>h.toString()));
|
|
499
|
+
});
|
|
500
|
+
this.#log.info(`Deleted ${txHashes.length} failed txs`, {
|
|
501
|
+
txHashes: txHashes.map((h)=>h.toString())
|
|
502
|
+
});
|
|
373
503
|
}
|
|
374
504
|
async handleFinalizedBlock(block) {
|
|
375
505
|
const blockNumber = block.globalVariables.blockNumber;
|
|
376
|
-
// Step 1: Find txs
|
|
377
|
-
const
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
txsToArchive.push(Tx.fromBuffer(buffer));
|
|
506
|
+
// Step 1: Find mined txs at or before finalized block
|
|
507
|
+
const minedTxsToFinalize = this.#indices.findTxsMinedAtOrBefore(blockNumber);
|
|
508
|
+
await this.#store.transactionAsync(async ()=>{
|
|
509
|
+
// Step 2: Collect mined txs for archiving (before deletion)
|
|
510
|
+
const txsToArchive = [];
|
|
511
|
+
if (this.#archive.isEnabled()) {
|
|
512
|
+
for (const txHashStr of minedTxsToFinalize){
|
|
513
|
+
const buffer = await this.#txsDB.getAsync(txHashStr);
|
|
514
|
+
if (buffer) {
|
|
515
|
+
txsToArchive.push(Tx.fromBuffer(buffer));
|
|
516
|
+
}
|
|
388
517
|
}
|
|
389
518
|
}
|
|
519
|
+
// Step 3: Delete mined txs from active pool
|
|
520
|
+
await this.#deleteTxsBatch(minedTxsToFinalize);
|
|
521
|
+
// Step 4: Finalize soft-deleted txs
|
|
522
|
+
await this.#deletedPool.finalizeBlock(blockNumber);
|
|
523
|
+
// Step 5: Archive mined txs
|
|
524
|
+
if (txsToArchive.length > 0) {
|
|
525
|
+
await this.#archive.archiveTxs(txsToArchive);
|
|
526
|
+
}
|
|
527
|
+
});
|
|
528
|
+
if (minedTxsToFinalize.length > 0) {
|
|
529
|
+
this.#log.info(`Finalized ${minedTxsToFinalize.length} mined txs from blocks up to ${blockNumber}`, {
|
|
530
|
+
txHashes: minedTxsToFinalize
|
|
531
|
+
});
|
|
390
532
|
}
|
|
391
|
-
// Step 3: Delete from active pool
|
|
392
|
-
await this.#deleteTxsBatch(txsToFinalize);
|
|
393
|
-
// Step 4: Archive
|
|
394
|
-
if (txsToArchive.length > 0) {
|
|
395
|
-
await this.#archive.archiveTxs(txsToArchive);
|
|
396
|
-
}
|
|
397
|
-
this.#log.info(`Finalized ${txsToFinalize.length} txs from blocks up to ${blockNumber}`);
|
|
398
533
|
}
|
|
399
534
|
// === Query Methods ===
|
|
400
535
|
async getTxByHash(txHash) {
|
|
@@ -410,20 +545,34 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
410
545
|
return results;
|
|
411
546
|
}
|
|
412
547
|
hasTxs(txHashes) {
|
|
413
|
-
return txHashes.map((h)=>
|
|
548
|
+
return txHashes.map((h)=>{
|
|
549
|
+
const hashStr = h.toString();
|
|
550
|
+
return this.#indices.has(hashStr) || this.#deletedPool.isSoftDeleted(hashStr);
|
|
551
|
+
});
|
|
414
552
|
}
|
|
415
553
|
getTxStatus(txHash) {
|
|
416
|
-
const
|
|
417
|
-
|
|
418
|
-
|
|
554
|
+
const txHashStr = txHash.toString();
|
|
555
|
+
const meta = this.#indices.getMetadata(txHashStr);
|
|
556
|
+
if (meta) {
|
|
557
|
+
return this.#indices.getTxState(meta);
|
|
419
558
|
}
|
|
420
|
-
|
|
559
|
+
// Check if soft-deleted
|
|
560
|
+
if (this.#deletedPool.isSoftDeleted(txHashStr)) {
|
|
561
|
+
return 'deleted';
|
|
562
|
+
}
|
|
563
|
+
return undefined;
|
|
421
564
|
}
|
|
422
565
|
getPendingTxHashes() {
|
|
423
566
|
return [
|
|
424
567
|
...this.#indices.iteratePendingByPriority('desc')
|
|
425
568
|
].map((hash)=>TxHash.fromString(hash));
|
|
426
569
|
}
|
|
570
|
+
getEligiblePendingTxHashes() {
|
|
571
|
+
const maxReceivedAt = this.#dateProvider.now() - this.#config.minTxPoolAgeMs;
|
|
572
|
+
return [
|
|
573
|
+
...this.#indices.iterateEligiblePendingByPriority('desc', maxReceivedAt)
|
|
574
|
+
].map((hash)=>TxHash.fromString(hash));
|
|
575
|
+
}
|
|
427
576
|
getPendingTxCount() {
|
|
428
577
|
return this.#indices.getPendingTxCount();
|
|
429
578
|
}
|
|
@@ -463,6 +612,9 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
463
612
|
this.#config.archivedTxLimit = config.archivedTxLimit;
|
|
464
613
|
this.#archive.updateLimit(config.archivedTxLimit);
|
|
465
614
|
}
|
|
615
|
+
if (config.minTxPoolAgeMs !== undefined) {
|
|
616
|
+
this.#config.minTxPoolAgeMs = config.minTxPoolAgeMs;
|
|
617
|
+
}
|
|
466
618
|
// Update eviction rules with new config
|
|
467
619
|
this.#evictionManager.updateConfig(config);
|
|
468
620
|
}
|
|
@@ -477,7 +629,10 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
477
629
|
}
|
|
478
630
|
// === Metrics ===
|
|
479
631
|
countTxs() {
|
|
480
|
-
return
|
|
632
|
+
return {
|
|
633
|
+
...this.#indices.countTxs(),
|
|
634
|
+
softDeleted: this.#deletedPool.getSoftDeletedCount()
|
|
635
|
+
};
|
|
481
636
|
}
|
|
482
637
|
// ============================================================================
|
|
483
638
|
// PRIVATE HELPERS - Transaction Management
|
|
@@ -485,10 +640,12 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
485
640
|
/**
|
|
486
641
|
* Adds a new transaction to the pool with the specified state.
|
|
487
642
|
* Emits onTxsAdded callback immediately after DB write.
|
|
488
|
-
*/ async #addTx(tx, state, opts = {}) {
|
|
643
|
+
*/ async #addTx(tx, state, opts = {}, precomputedMeta) {
|
|
489
644
|
const txHashStr = tx.getTxHash().toString();
|
|
490
|
-
const meta = await buildTxMetaData(tx);
|
|
645
|
+
const meta = precomputedMeta ?? await buildTxMetaData(tx);
|
|
646
|
+
meta.receivedAt = this.#dateProvider.now();
|
|
491
647
|
await this.#txsDB.set(txHashStr, tx.toBuffer());
|
|
648
|
+
await this.#deletedPool.clearSoftDeleted(txHashStr);
|
|
492
649
|
this.#callbacks.onTxsAdded([
|
|
493
650
|
tx
|
|
494
651
|
], opts);
|
|
@@ -501,64 +658,74 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
501
658
|
this.#indices.addMined(meta);
|
|
502
659
|
}
|
|
503
660
|
const stateStr = typeof state === 'string' ? state : Object.keys(state)[0];
|
|
504
|
-
this.#log.
|
|
661
|
+
this.#log.debug(`Added tx ${txHashStr} as ${stateStr}`, {
|
|
505
662
|
eventName: 'tx-added-to-pool',
|
|
506
|
-
|
|
663
|
+
txHash: txHashStr,
|
|
664
|
+
state: stateStr,
|
|
665
|
+
source: opts.source
|
|
507
666
|
});
|
|
508
667
|
return meta;
|
|
509
668
|
}
|
|
510
669
|
/**
|
|
511
670
|
* Deletes a transaction from both indices and DB.
|
|
512
671
|
* Emits onTxsRemoved callback immediately after DB delete.
|
|
672
|
+
*/ /**
|
|
673
|
+
* Deletes a transaction from the pool.
|
|
674
|
+
* Delegates to DeletedPool which decides soft vs hard delete based on whether
|
|
675
|
+
* the tx is from a pruned block.
|
|
513
676
|
*/ async #deleteTx(txHashStr) {
|
|
514
677
|
this.#indices.remove(txHashStr);
|
|
515
|
-
await this.#txsDB.delete(txHashStr);
|
|
516
678
|
this.#callbacks.onTxsRemoved([
|
|
517
679
|
txHashStr
|
|
518
680
|
]);
|
|
681
|
+
await this.#deletedPool.deleteTx(txHashStr);
|
|
519
682
|
}
|
|
520
683
|
/** Deletes a batch of transactions, emitting callbacks individually for each. */ async #deleteTxsBatch(txHashes) {
|
|
521
684
|
for (const txHashStr of txHashes){
|
|
522
685
|
await this.#deleteTx(txHashStr);
|
|
523
686
|
}
|
|
524
687
|
}
|
|
688
|
+
/** Evicts transactions: records eviction metric with reason, caches hashes, then deletes. */ async #evictTxs(txHashes, reason) {
|
|
689
|
+
if (txHashes.length === 0) {
|
|
690
|
+
return;
|
|
691
|
+
}
|
|
692
|
+
this.#instrumentation.recordEvictions(txHashes.length, reason);
|
|
693
|
+
for (const txHashStr of txHashes){
|
|
694
|
+
this.#log.debug(`Evicting tx ${txHashStr}`, {
|
|
695
|
+
txHash: txHashStr,
|
|
696
|
+
reason
|
|
697
|
+
});
|
|
698
|
+
this.#addToEvictedCache(txHashStr);
|
|
699
|
+
}
|
|
700
|
+
await this.#deleteTxsBatch(txHashes);
|
|
701
|
+
}
|
|
702
|
+
/** Adds a tx hash to the bounded evicted cache, evicting the oldest entry if at capacity. */ #addToEvictedCache(txHashStr) {
|
|
703
|
+
if (this.#evictedTxHashes.size >= this.#config.evictedTxCacheSize) {
|
|
704
|
+
// FIFO eviction: remove the first (oldest) entry
|
|
705
|
+
const oldest = this.#evictedTxHashes.values().next().value;
|
|
706
|
+
this.#evictedTxHashes.delete(oldest);
|
|
707
|
+
}
|
|
708
|
+
this.#evictedTxHashes.add(txHashStr);
|
|
709
|
+
}
|
|
525
710
|
// ============================================================================
|
|
526
711
|
// PRIVATE HELPERS - Validation & Conflict Resolution
|
|
527
712
|
// ============================================================================
|
|
528
|
-
/** Validates
|
|
529
|
-
const
|
|
713
|
+
/** Validates transaction metadata, returning true if valid */ async #validateMeta(meta, validator, context) {
|
|
714
|
+
const txValidator = validator ?? await this.#createTxValidator();
|
|
715
|
+
const result = await txValidator.validateTx(meta);
|
|
530
716
|
if (result.result !== 'valid') {
|
|
531
717
|
const contextStr = context ? ` ${context}` : '';
|
|
532
|
-
this.#log.info(`Tx ${
|
|
718
|
+
this.#log.info(`Tx ${meta.txHash}${contextStr} failed validation: ${result.reason?.join(', ')}`);
|
|
533
719
|
return false;
|
|
534
720
|
}
|
|
535
721
|
return true;
|
|
536
722
|
}
|
|
537
|
-
/**
|
|
538
|
-
const loaded = [];
|
|
539
|
-
const missing = [];
|
|
540
|
-
for (const meta of metas){
|
|
541
|
-
const buffer = await this.#txsDB.getAsync(meta.txHash);
|
|
542
|
-
if (!buffer) {
|
|
543
|
-
this.#log.warn(`Tx ${meta.txHash} not found in DB`);
|
|
544
|
-
missing.push(meta.txHash);
|
|
545
|
-
continue;
|
|
546
|
-
}
|
|
547
|
-
loaded.push({
|
|
548
|
-
tx: Tx.fromBuffer(buffer),
|
|
549
|
-
meta
|
|
550
|
-
});
|
|
551
|
-
}
|
|
552
|
-
return {
|
|
553
|
-
loaded,
|
|
554
|
-
missing
|
|
555
|
-
};
|
|
556
|
-
}
|
|
557
|
-
/** Validates a batch of transactions, returning valid and invalid groups */ async #validateTxBatch(txs, context) {
|
|
723
|
+
/** Validates metadata directly */ async #revalidateMetadata(metas, context) {
|
|
558
724
|
const valid = [];
|
|
559
725
|
const invalid = [];
|
|
560
|
-
|
|
561
|
-
|
|
726
|
+
const validator = await this.#createTxValidator();
|
|
727
|
+
for (const meta of metas){
|
|
728
|
+
if (await this.#validateMeta(meta, validator, context)) {
|
|
562
729
|
valid.push(meta);
|
|
563
730
|
} else {
|
|
564
731
|
invalid.push(meta.txHash);
|
|
@@ -569,17 +736,6 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
569
736
|
invalid
|
|
570
737
|
};
|
|
571
738
|
}
|
|
572
|
-
/** Loads transactions from DB and validates them */ async #loadAndValidateTxs(metas, context) {
|
|
573
|
-
const { loaded, missing } = await this.#loadTxsFromDb(metas);
|
|
574
|
-
const { valid, invalid } = await this.#validateTxBatch(loaded, context);
|
|
575
|
-
return {
|
|
576
|
-
valid,
|
|
577
|
-
invalid: [
|
|
578
|
-
...missing,
|
|
579
|
-
...invalid
|
|
580
|
-
]
|
|
581
|
-
};
|
|
582
|
-
}
|
|
583
739
|
/**
|
|
584
740
|
* Resolves nullifier conflicts between incoming txs and existing pending txs.
|
|
585
741
|
* Modifies the pending indices during iteration to maintain consistent state
|
|
@@ -635,6 +791,10 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
635
791
|
const loaded = [];
|
|
636
792
|
const errors = [];
|
|
637
793
|
for await (const [txHashStr, buffer] of this.#txsDB.entriesAsync()){
|
|
794
|
+
// Skip soft-deleted transactions - they stay in DB but not in indices
|
|
795
|
+
if (this.#deletedPool.isSoftDeleted(txHashStr)) {
|
|
796
|
+
continue;
|
|
797
|
+
}
|
|
638
798
|
try {
|
|
639
799
|
const tx = Tx.fromBuffer(buffer);
|
|
640
800
|
const meta = await buildTxMetaData(tx);
|
|
@@ -685,7 +845,7 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
685
845
|
if (preAddResult.shouldIgnore) {
|
|
686
846
|
// Transaction rejected - mark for deletion from DB
|
|
687
847
|
rejected.push(meta.txHash);
|
|
688
|
-
this.#log.debug(`Rejected tx ${meta.txHash} during rebuild: ${preAddResult.reason}`);
|
|
848
|
+
this.#log.debug(`Rejected tx ${meta.txHash} during rebuild: ${preAddResult.reason?.message ?? 'unknown reason'}`);
|
|
689
849
|
continue;
|
|
690
850
|
}
|
|
691
851
|
// Evict any conflicting txs identified by pre-add rules
|
|
@@ -721,7 +881,7 @@ import { TxPoolIndices } from './tx_pool_indices.js';
|
|
|
721
881
|
getFeePayerPendingTxs: (feePayer)=>this.#indices.getFeePayerPendingTxs(feePayer),
|
|
722
882
|
getPendingTxCount: ()=>this.#indices.getPendingTxCount(),
|
|
723
883
|
getLowestPriorityPending: (limit)=>this.#indices.getLowestPriorityPending(limit),
|
|
724
|
-
deleteTxs: (txHashes)=>this.#
|
|
884
|
+
deleteTxs: (txHashes, reason)=>this.#evictTxs(txHashes, reason ?? 'unknown')
|
|
725
885
|
};
|
|
726
886
|
}
|
|
727
887
|
#createPreAddPoolAccess() {
|