@aztec/p2p 0.76.4 → 0.77.0-testnet-ignition.21
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 +2 -2
- package/dest/bootstrap/bootstrap.d.ts.map +1 -1
- package/dest/bootstrap/bootstrap.js +55 -41
- package/dest/client/factory.d.ts +8 -6
- package/dest/client/factory.d.ts.map +1 -1
- package/dest/client/factory.js +8 -10
- package/dest/client/index.js +0 -1
- package/dest/client/p2p_client.d.ts +7 -4
- package/dest/client/p2p_client.d.ts.map +1 -1
- package/dest/client/p2p_client.js +492 -514
- package/dest/config.d.ts +8 -10
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +54 -47
- package/dest/enr/generate-enr.d.ts +9 -0
- package/dest/enr/generate-enr.d.ts.map +1 -0
- package/dest/enr/generate-enr.js +30 -0
- package/dest/enr/index.d.ts +2 -0
- package/dest/enr/index.d.ts.map +1 -0
- package/dest/enr/index.js +1 -0
- package/dest/errors/reqresp.error.js +6 -10
- package/dest/index.js +0 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts +1 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/attestation_pool.js +6 -2
- 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 +65 -33
- package/dest/mem_pools/attestation_pool/index.js +0 -1
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts +3 -3
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/kv_attestation_pool.js +23 -20
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts +2 -2
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/memory_attestation_pool.js +22 -26
- package/dest/mem_pools/attestation_pool/mocks.d.ts +3 -2
- package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -1
- package/dest/mem_pools/attestation_pool/mocks.js +12 -7
- package/dest/mem_pools/index.d.ts +2 -2
- package/dest/mem_pools/index.d.ts.map +1 -1
- package/dest/mem_pools/index.js +1 -2
- package/dest/mem_pools/instrumentation.d.ts +1 -1
- package/dest/mem_pools/instrumentation.d.ts.map +1 -1
- package/dest/mem_pools/instrumentation.js +35 -39
- package/dest/mem_pools/interface.d.ts +3 -3
- package/dest/mem_pools/interface.d.ts.map +1 -1
- package/dest/mem_pools/interface.js +3 -2
- package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts +2 -2
- 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 +129 -136
- package/dest/mem_pools/tx_pool/index.js +0 -1
- package/dest/mem_pools/tx_pool/memory_tx_pool.d.ts +2 -2
- package/dest/mem_pools/tx_pool/memory_tx_pool.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/memory_tx_pool.js +46 -44
- package/dest/mem_pools/tx_pool/priority.d.ts +1 -1
- package/dest/mem_pools/tx_pool/priority.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/priority.js +1 -3
- package/dest/mem_pools/tx_pool/tx_pool.d.ts +1 -1
- package/dest/mem_pools/tx_pool/tx_pool.d.ts.map +1 -1
- package/dest/mem_pools/tx_pool/tx_pool.js +3 -2
- 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 +109 -39
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts +2 -2
- package/dest/msg_validators/attestation_validator/attestation_validator.d.ts.map +1 -1
- package/dest/msg_validators/attestation_validator/attestation_validator.js +4 -4
- package/dest/msg_validators/attestation_validator/index.js +0 -1
- package/dest/msg_validators/block_proposal_validator/block_proposal_validator.d.ts +2 -2
- package/dest/msg_validators/block_proposal_validator/block_proposal_validator.d.ts.map +1 -1
- package/dest/msg_validators/block_proposal_validator/block_proposal_validator.js +3 -3
- package/dest/msg_validators/block_proposal_validator/index.js +0 -1
- package/dest/msg_validators/index.js +0 -1
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts +1 -1
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/aggregate_tx_validator.js +9 -11
- 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/block_header_validator.js +18 -13
- 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/data_validator.js +102 -33
- 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/double_spend_validator.js +34 -20
- package/dest/msg_validators/tx_validator/index.js +0 -1
- 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/metadata_validator.js +30 -27
- package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts +2 -1
- package/dest/msg_validators/tx_validator/tx_proof_validator.d.ts.map +1 -1
- package/dest/msg_validators/tx_validator/tx_proof_validator.js +17 -12
- package/dest/services/data_store.js +57 -57
- package/dest/services/discv5/discV5_service.d.ts +2 -0
- package/dest/services/discv5/discV5_service.d.ts.map +1 -1
- package/dest/services/discv5/discV5_service.js +64 -36
- package/dest/services/dummy_service.d.ts +4 -2
- package/dest/services/dummy_service.d.ts.map +1 -1
- package/dest/services/dummy_service.js +41 -59
- package/dest/services/encoding.d.ts +3 -3
- package/dest/services/encoding.d.ts.map +1 -1
- package/dest/services/encoding.js +10 -9
- package/dest/services/gossipsub/scoring.d.ts +7 -0
- package/dest/services/gossipsub/scoring.d.ts.map +1 -0
- package/dest/services/gossipsub/scoring.js +10 -0
- package/dest/services/index.js +0 -1
- package/dest/services/libp2p/libp2p_service.d.ts +10 -33
- package/dest/services/libp2p/libp2p_service.d.ts.map +1 -1
- package/dest/services/libp2p/libp2p_service.js +682 -673
- package/dest/services/peer-manager/metrics.js +14 -7
- package/dest/services/peer-manager/peer_manager.d.ts +24 -6
- package/dest/services/peer-manager/peer_manager.d.ts.map +1 -1
- package/dest/services/peer-manager/peer_manager.js +390 -340
- package/dest/services/peer-manager/peer_scoring.d.ts +3 -3
- package/dest/services/peer-manager/peer_scoring.d.ts.map +1 -1
- package/dest/services/peer-manager/peer_scoring.js +21 -19
- package/dest/services/reqresp/config.js +4 -5
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts +2 -2
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.d.ts.map +1 -1
- package/dest/services/reqresp/connection-sampler/batch_connection_sampler.js +35 -28
- package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts +1 -1
- package/dest/services/reqresp/connection-sampler/connection_sampler.d.ts.map +1 -1
- package/dest/services/reqresp/connection-sampler/connection_sampler.js +67 -61
- package/dest/services/reqresp/index.js +1 -3
- 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 +27 -31
- 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 +23 -10
- package/dest/services/reqresp/protocols/block.d.ts +2 -2
- package/dest/services/reqresp/protocols/block.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/block.js +1 -2
- package/dest/services/reqresp/protocols/goodbye.d.ts +5 -5
- package/dest/services/reqresp/protocols/goodbye.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/goodbye.js +36 -41
- package/dest/services/reqresp/protocols/index.js +1 -3
- package/dest/services/reqresp/protocols/ping.js +1 -3
- package/dest/services/reqresp/protocols/status.js +1 -3
- package/dest/services/reqresp/protocols/tx.d.ts +3 -3
- package/dest/services/reqresp/protocols/tx.d.ts.map +1 -1
- package/dest/services/reqresp/protocols/tx.js +6 -9
- package/dest/services/reqresp/rate-limiter/index.js +0 -1
- package/dest/services/reqresp/rate-limiter/rate_limiter.d.ts +9 -9
- package/dest/services/reqresp/rate-limiter/rate_limiter.d.ts.map +1 -1
- package/dest/services/reqresp/rate-limiter/rate_limiter.js +53 -46
- package/dest/services/reqresp/rate-limiter/rate_limits.js +16 -17
- package/dest/services/reqresp/reqresp.d.ts +4 -4
- package/dest/services/reqresp/reqresp.d.ts.map +1 -1
- package/dest/services/reqresp/reqresp.js +467 -464
- package/dest/services/reqresp/status.js +16 -17
- package/dest/services/service.d.ts +3 -2
- package/dest/services/service.d.ts.map +1 -1
- package/dest/services/service.js +3 -4
- package/dest/test-helpers/generate-peer-id-private-keys.js +2 -4
- package/dest/test-helpers/get-ports.js +3 -3
- package/dest/test-helpers/index.js +0 -1
- package/dest/test-helpers/make-enrs.d.ts +1 -1
- package/dest/test-helpers/make-enrs.d.ts.map +1 -1
- package/dest/test-helpers/make-enrs.js +3 -6
- package/dest/test-helpers/make-test-p2p-clients.d.ts +7 -6
- package/dest/test-helpers/make-test-p2p-clients.d.ts.map +1 -1
- package/dest/test-helpers/make-test-p2p-clients.js +10 -12
- package/dest/test-helpers/reqresp-nodes.d.ts +18 -7
- package/dest/test-helpers/reqresp-nodes.d.ts.map +1 -1
- package/dest/test-helpers/reqresp-nodes.js +64 -40
- package/dest/testbench/p2p_client_testbench_worker.js +61 -45
- package/dest/testbench/parse_log_file.d.ts +2 -0
- package/dest/testbench/parse_log_file.d.ts.map +1 -0
- package/dest/testbench/parse_log_file.js +131 -0
- package/dest/testbench/testbench.d.ts +2 -0
- package/dest/testbench/testbench.d.ts.map +1 -0
- package/dest/testbench/testbench.js +141 -0
- package/dest/{services/types.d.ts → types/index.d.ts} +1 -1
- package/dest/types/index.d.ts.map +1 -0
- package/dest/types/index.js +28 -0
- package/dest/util.d.ts +5 -5
- package/dest/util.d.ts.map +1 -1
- package/dest/util.js +23 -34
- package/dest/versioning.d.ts +3 -3
- package/dest/versioning.d.ts.map +1 -1
- package/dest/versioning.js +7 -12
- package/package.json +15 -13
- package/src/bootstrap/bootstrap.ts +30 -17
- package/src/client/factory.ts +9 -12
- package/src/client/p2p_client.ts +13 -24
- package/src/config.ts +14 -15
- package/src/enr/generate-enr.ts +39 -0
- package/src/enr/index.ts +1 -0
- package/src/mem_pools/attestation_pool/attestation_pool.ts +1 -1
- package/src/mem_pools/attestation_pool/attestation_pool_test_suite.ts +4 -3
- package/src/mem_pools/attestation_pool/kv_attestation_pool.ts +3 -3
- package/src/mem_pools/attestation_pool/memory_attestation_pool.ts +2 -2
- package/src/mem_pools/attestation_pool/mocks.ts +5 -5
- package/src/mem_pools/index.ts +2 -2
- package/src/mem_pools/instrumentation.ts +4 -3
- package/src/mem_pools/interface.ts +3 -3
- package/src/mem_pools/tx_pool/aztec_kv_tx_pool.ts +4 -4
- package/src/mem_pools/tx_pool/memory_tx_pool.ts +3 -3
- package/src/mem_pools/tx_pool/priority.ts +1 -1
- package/src/mem_pools/tx_pool/tx_pool.ts +1 -1
- package/src/mem_pools/tx_pool/tx_pool_test_suite.ts +4 -3
- package/src/msg_validators/attestation_validator/attestation_validator.ts +2 -2
- package/src/msg_validators/block_proposal_validator/block_proposal_validator.ts +2 -2
- package/src/msg_validators/tx_validator/aggregate_tx_validator.ts +1 -1
- package/src/msg_validators/tx_validator/block_header_validator.ts +2 -2
- package/src/msg_validators/tx_validator/data_validator.ts +57 -4
- package/src/msg_validators/tx_validator/double_spend_validator.ts +17 -12
- package/src/msg_validators/tx_validator/metadata_validator.ts +2 -2
- package/src/msg_validators/tx_validator/tx_proof_validator.ts +2 -6
- package/src/services/discv5/discV5_service.ts +33 -8
- package/src/services/dummy_service.ts +4 -2
- package/src/services/encoding.ts +3 -3
- package/src/services/gossipsub/scoring.ts +13 -0
- package/src/services/libp2p/libp2p_service.ts +124 -146
- package/src/services/peer-manager/peer_manager.ts +71 -13
- package/src/services/peer-manager/peer_scoring.ts +3 -3
- package/src/services/reqresp/connection-sampler/batch_connection_sampler.ts +2 -2
- package/src/services/reqresp/connection-sampler/connection_sampler.ts +9 -3
- package/src/services/reqresp/interface.ts +4 -3
- package/src/services/reqresp/metrics.ts +1 -1
- package/src/services/reqresp/protocols/block.ts +3 -3
- package/src/services/reqresp/protocols/goodbye.ts +7 -7
- package/src/services/reqresp/protocols/tx.ts +5 -5
- package/src/services/reqresp/rate-limiter/rate_limiter.ts +22 -18
- package/src/services/reqresp/reqresp.ts +18 -11
- package/src/services/service.ts +3 -2
- package/src/test-helpers/make-enrs.ts +1 -1
- package/src/test-helpers/make-test-p2p-clients.ts +9 -7
- package/src/test-helpers/reqresp-nodes.ts +32 -18
- package/src/testbench/p2p_client_testbench_worker.ts +16 -9
- package/src/testbench/parse_log_file.ts +175 -0
- package/src/testbench/testbench.ts +157 -0
- package/src/util.ts +5 -5
- package/src/versioning.ts +7 -7
- package/dest/services/libp2p/libp2p_logger.d.ts +0 -7
- package/dest/services/libp2p/libp2p_logger.d.ts.map +0 -1
- package/dest/services/libp2p/libp2p_logger.js +0 -67
- package/dest/services/types.d.ts.map +0 -1
- package/dest/services/types.js +0 -35
- package/src/services/libp2p/libp2p_logger.ts +0 -78
- package/src/testbench/scripts/run_testbench.sh +0 -7
- /package/src/{services/types.ts → types/index.ts} +0 -0
|
@@ -1,527 +1,505 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
1
|
+
function _ts_decorate(decorators, target, key, desc) {
|
|
2
|
+
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
|
|
3
|
+
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
|
|
4
|
+
else for(var i = decorators.length - 1; i >= 0; i--)if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
|
|
5
|
+
return c > 3 && r && Object.defineProperty(target, key, r), r;
|
|
6
|
+
}
|
|
7
|
+
import { INITIAL_L2_BLOCK_NUM } from '@aztec/constants';
|
|
3
8
|
import { createLogger } from '@aztec/foundation/log';
|
|
4
|
-
import { Attributes, TraceableL2BlockStream, WithTracer, getTelemetryClient, trackSpan
|
|
9
|
+
import { Attributes, TraceableL2BlockStream, WithTracer, getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
|
|
5
10
|
import { getP2PDefaultConfig } from '../config.js';
|
|
6
11
|
import { ReqRespSubProtocol } from '../services/reqresp/interface.js';
|
|
7
12
|
/**
|
|
8
13
|
* Enum defining the possible states of the p2p client.
|
|
9
|
-
*/
|
|
10
|
-
export var P2PClientState;
|
|
11
|
-
(function (P2PClientState) {
|
|
14
|
+
*/ export var P2PClientState = /*#__PURE__*/ function(P2PClientState) {
|
|
12
15
|
P2PClientState[P2PClientState["IDLE"] = 0] = "IDLE";
|
|
13
16
|
P2PClientState[P2PClientState["SYNCHING"] = 1] = "SYNCHING";
|
|
14
17
|
P2PClientState[P2PClientState["RUNNING"] = 2] = "RUNNING";
|
|
15
18
|
P2PClientState[P2PClientState["STOPPED"] = 3] = "STOPPED";
|
|
16
|
-
|
|
19
|
+
return P2PClientState;
|
|
20
|
+
}({});
|
|
17
21
|
/**
|
|
18
22
|
* The P2P client implementation.
|
|
19
|
-
*/
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
/**
|
|
123
|
-
* Starts the P2P client.
|
|
124
|
-
* @returns An empty promise signalling the synching process.
|
|
125
|
-
*/
|
|
126
|
-
async start() {
|
|
127
|
-
if (this.currentState === P2PClientState.STOPPED) {
|
|
128
|
-
throw new Error('P2P client already stopped');
|
|
129
|
-
}
|
|
130
|
-
if (this.currentState !== P2PClientState.IDLE) {
|
|
131
|
-
return this.syncPromise;
|
|
132
|
-
}
|
|
133
|
-
// get the current latest block numbers
|
|
134
|
-
this.latestBlockNumberAtStart = await this.l2BlockSource.getBlockNumber();
|
|
135
|
-
this.provenBlockNumberAtStart = await this.l2BlockSource.getProvenBlockNumber();
|
|
136
|
-
const syncedLatestBlock = (await this.getSyncedLatestBlockNum()) + 1;
|
|
137
|
-
const syncedProvenBlock = (await this.getSyncedProvenBlockNum()) + 1;
|
|
138
|
-
// if there are blocks to be retrieved, go to a synching state
|
|
139
|
-
if (syncedLatestBlock <= this.latestBlockNumberAtStart || syncedProvenBlock <= this.provenBlockNumberAtStart) {
|
|
140
|
-
this.setCurrentState(P2PClientState.SYNCHING);
|
|
141
|
-
this.syncPromise = new Promise(resolve => {
|
|
142
|
-
this.syncResolve = resolve;
|
|
143
|
-
});
|
|
144
|
-
this.log.verbose(`Starting sync from ${syncedLatestBlock} (last proven ${syncedProvenBlock})`);
|
|
145
|
-
}
|
|
146
|
-
else {
|
|
147
|
-
// if no blocks to be retrieved, go straight to running
|
|
148
|
-
this.setCurrentState(P2PClientState.RUNNING);
|
|
149
|
-
this.syncPromise = Promise.resolve();
|
|
150
|
-
await this.p2pService.start();
|
|
151
|
-
this.log.debug(`Block ${syncedLatestBlock} (proven ${syncedProvenBlock}) already beyond current block`);
|
|
152
|
-
}
|
|
153
|
-
this.blockStream.start();
|
|
154
|
-
this.log.verbose(`Started block downloader from block ${syncedLatestBlock}`);
|
|
155
|
-
return this.syncPromise;
|
|
156
|
-
}
|
|
157
|
-
/**
|
|
158
|
-
* Allows consumers to stop the instance of the P2P client.
|
|
159
|
-
* 'ready' will now return 'false' and the running promise that keeps the client synced is interrupted.
|
|
160
|
-
*/
|
|
161
|
-
async stop() {
|
|
162
|
-
this.log.debug('Stopping p2p client...');
|
|
163
|
-
await this.p2pService.stop();
|
|
164
|
-
this.log.debug('Stopped p2p service');
|
|
165
|
-
await this.blockStream.stop();
|
|
166
|
-
this.log.debug('Stopped block downloader');
|
|
167
|
-
await this.runningPromise;
|
|
168
|
-
this.setCurrentState(P2PClientState.STOPPED);
|
|
169
|
-
this.log.info('P2P client stopped.');
|
|
170
|
-
}
|
|
171
|
-
broadcastProposal(proposal) {
|
|
172
|
-
this.log.verbose(`Broadcasting proposal ${proposal.p2pMessageIdentifier()} to peers`);
|
|
173
|
-
return this.p2pService.propagate(proposal);
|
|
174
|
-
}
|
|
175
|
-
async getAttestationsForSlot(slot, proposalId) {
|
|
176
|
-
return (await this.attestationPool?.getAttestationsForSlot(slot, proposalId)) ?? [];
|
|
177
|
-
}
|
|
178
|
-
// REVIEW: https://github.com/AztecProtocol/aztec-packages/issues/7963
|
|
179
|
-
// ^ This pattern is not my favorite (md)
|
|
180
|
-
registerBlockProposalHandler(handler) {
|
|
181
|
-
this.p2pService.registerBlockReceivedCallback(handler);
|
|
182
|
-
}
|
|
183
|
-
/**
|
|
184
|
-
* Requests the transactions with the given hashes from the network.
|
|
185
|
-
*
|
|
186
|
-
* If a transaction can be retrieved, it will be returned, if not an undefined
|
|
187
|
-
* will be returned. In place.
|
|
188
|
-
*
|
|
189
|
-
* @param txHashes - The hashes of the transactions to request.
|
|
190
|
-
* @returns A promise that resolves to an array of transactions or undefined.
|
|
191
|
-
*/
|
|
192
|
-
async requestTxs(txHashes) {
|
|
193
|
-
const res = await this.p2pService.sendBatchRequest(ReqRespSubProtocol.TX, txHashes);
|
|
194
|
-
return Promise.resolve(res ?? []);
|
|
195
|
-
}
|
|
196
|
-
/**
|
|
197
|
-
* Uses the Request Response protocol to request a transaction from the network.
|
|
198
|
-
*
|
|
199
|
-
* If the underlying request response protocol fails, then we return undefined.
|
|
200
|
-
* If it succeeds then we add the transaction to our transaction pool and return.
|
|
201
|
-
*
|
|
202
|
-
* @param txHash - The hash of the transaction to request.
|
|
203
|
-
* @returns A promise that resolves to a transaction or undefined.
|
|
204
|
-
*/
|
|
205
|
-
async requestTxByHash(txHash) {
|
|
206
|
-
const tx = await this.p2pService.sendRequest(ReqRespSubProtocol.TX, txHash);
|
|
207
|
-
if (tx) {
|
|
208
|
-
this.log.debug(`Received tx ${txHash.toString()} from peer`);
|
|
209
|
-
await this.txPool.addTxs([tx]);
|
|
210
|
-
}
|
|
211
|
-
else {
|
|
212
|
-
this.log.debug(`Failed to receive tx ${txHash.toString()} from peer`);
|
|
213
|
-
}
|
|
214
|
-
return tx;
|
|
215
|
-
}
|
|
216
|
-
/**
|
|
217
|
-
* Uses the batched Request Response protocol to request a set of transactions from the network.
|
|
218
|
-
*/
|
|
219
|
-
async requestTxsByHash(txHashes) {
|
|
220
|
-
const txs = (await this.p2pService.sendBatchRequest(ReqRespSubProtocol.TX, txHashes)) ?? [];
|
|
221
|
-
await this.txPool.addTxs(txs);
|
|
222
|
-
const txHashesStr = txHashes.map(tx => tx.toString()).join(', ');
|
|
223
|
-
this.log.debug(`Received batched txs ${txHashesStr} (${txs.length} / ${txHashes.length}}) from peers`);
|
|
224
|
-
return txs;
|
|
225
|
-
}
|
|
226
|
-
getPendingTxs() {
|
|
227
|
-
return Promise.resolve(this.getTxs('pending'));
|
|
228
|
-
}
|
|
229
|
-
async getPendingTxCount() {
|
|
230
|
-
const pendingTxs = await this.txPool.getPendingTxHashes();
|
|
231
|
-
return pendingTxs.length;
|
|
232
|
-
}
|
|
233
|
-
async *iteratePendingTxs() {
|
|
234
|
-
for (const txHash of await this.txPool.getPendingTxHashes()) {
|
|
235
|
-
const tx = await this.txPool.getTxByHash(txHash);
|
|
236
|
-
if (tx) {
|
|
237
|
-
yield tx;
|
|
238
|
-
}
|
|
239
|
-
}
|
|
240
|
-
}
|
|
241
|
-
/**
|
|
242
|
-
* Returns all transactions in the transaction pool.
|
|
243
|
-
* @returns An array of Txs.
|
|
244
|
-
*/
|
|
245
|
-
async getTxs(filter) {
|
|
246
|
-
if (filter === 'all') {
|
|
247
|
-
return this.txPool.getAllTxs();
|
|
248
|
-
}
|
|
249
|
-
else if (filter === 'mined') {
|
|
250
|
-
const minedHashes = await this.txPool.getMinedTxHashes();
|
|
251
|
-
const minedTx = await Promise.all(minedHashes.map(([txHash]) => this.txPool.getTxByHash(txHash)));
|
|
252
|
-
return minedTx.filter((tx) => !!tx);
|
|
23
|
+
*/ export class P2PClient extends WithTracer {
|
|
24
|
+
l2BlockSource;
|
|
25
|
+
p2pService;
|
|
26
|
+
log;
|
|
27
|
+
/** The JS promise that will be running to keep the client's data in sync. Can be interrupted if the client is stopped. */ runningPromise;
|
|
28
|
+
currentState;
|
|
29
|
+
syncPromise;
|
|
30
|
+
syncResolve;
|
|
31
|
+
latestBlockNumberAtStart;
|
|
32
|
+
provenBlockNumberAtStart;
|
|
33
|
+
synchedBlockHashes;
|
|
34
|
+
synchedLatestBlockNumber;
|
|
35
|
+
synchedProvenBlockNumber;
|
|
36
|
+
txPool;
|
|
37
|
+
attestationPool;
|
|
38
|
+
/** How many slots to keep attestations for. */ keepAttestationsInPoolFor;
|
|
39
|
+
/** How many slots to keep proven txs for. */ keepProvenTxsFor;
|
|
40
|
+
blockStream;
|
|
41
|
+
/**
|
|
42
|
+
* In-memory P2P client constructor.
|
|
43
|
+
* @param store - The client's instance of the KV store.
|
|
44
|
+
* @param l2BlockSource - P2P client's source for fetching existing blocks.
|
|
45
|
+
* @param txPool - The client's instance of a transaction pool. Defaults to in-memory implementation.
|
|
46
|
+
* @param p2pService - The concrete instance of p2p networking to use.
|
|
47
|
+
* @param keepProvenTxsFor - How many blocks have to pass after a block is proven before its txs are deleted (zero to delete immediately once proven).
|
|
48
|
+
* @param log - A logger.
|
|
49
|
+
*/ constructor(_clientType, store, l2BlockSource, mempools, p2pService, config = {}, telemetry = getTelemetryClient(), log = createLogger('p2p')){
|
|
50
|
+
super(telemetry, 'P2PClient'), this.l2BlockSource = l2BlockSource, this.p2pService = p2pService, this.log = log, this.currentState = 0, this.syncPromise = Promise.resolve(), this.syncResolve = undefined, this.latestBlockNumberAtStart = -1, this.provenBlockNumberAtStart = -1;
|
|
51
|
+
const { keepProvenTxsInPoolFor, blockCheckIntervalMS, blockRequestBatchSize, keepAttestationsInPoolFor } = {
|
|
52
|
+
...getP2PDefaultConfig(),
|
|
53
|
+
...config
|
|
54
|
+
};
|
|
55
|
+
this.keepProvenTxsFor = keepProvenTxsInPoolFor;
|
|
56
|
+
this.keepAttestationsInPoolFor = keepAttestationsInPoolFor;
|
|
57
|
+
const tracer = telemetry.getTracer('P2PL2BlockStream');
|
|
58
|
+
const logger = createLogger('p2p:l2-block-stream');
|
|
59
|
+
this.blockStream = new TraceableL2BlockStream(l2BlockSource, this, this, tracer, 'P2PL2BlockStream', logger, {
|
|
60
|
+
batchSize: blockRequestBatchSize,
|
|
61
|
+
pollIntervalMS: blockCheckIntervalMS
|
|
62
|
+
});
|
|
63
|
+
this.synchedBlockHashes = store.openMap('p2p_pool_block_hashes');
|
|
64
|
+
this.synchedLatestBlockNumber = store.openSingleton('p2p_pool_last_l2_block');
|
|
65
|
+
this.synchedProvenBlockNumber = store.openSingleton('p2p_pool_last_proven_l2_block');
|
|
66
|
+
this.txPool = mempools.txPool;
|
|
67
|
+
this.attestationPool = mempools.attestationPool;
|
|
68
|
+
}
|
|
69
|
+
isP2PClient() {
|
|
70
|
+
return true;
|
|
71
|
+
}
|
|
72
|
+
getPeers(includePending) {
|
|
73
|
+
return Promise.resolve(this.p2pService.getPeers(includePending));
|
|
74
|
+
}
|
|
75
|
+
getL2BlockHash(number) {
|
|
76
|
+
return this.synchedBlockHashes.getAsync(number);
|
|
77
|
+
}
|
|
78
|
+
async getL2Tips() {
|
|
79
|
+
const latestBlockNumber = await this.getSyncedLatestBlockNum();
|
|
80
|
+
let latestBlockHash;
|
|
81
|
+
const provenBlockNumber = await this.getSyncedProvenBlockNum();
|
|
82
|
+
let provenBlockHash;
|
|
83
|
+
if (latestBlockNumber > 0) {
|
|
84
|
+
latestBlockHash = await this.synchedBlockHashes.getAsync(latestBlockNumber);
|
|
85
|
+
if (typeof latestBlockHash === 'undefined') {
|
|
86
|
+
this.log.warn(`Block hash for latest block ${latestBlockNumber} not found`);
|
|
87
|
+
throw new Error();
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
if (provenBlockNumber > 0) {
|
|
91
|
+
provenBlockHash = await this.synchedBlockHashes.getAsync(provenBlockNumber);
|
|
92
|
+
if (typeof provenBlockHash === 'undefined') {
|
|
93
|
+
this.log.warn(`Block hash for proven block ${provenBlockNumber} not found`);
|
|
94
|
+
throw new Error();
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
return Promise.resolve({
|
|
98
|
+
latest: {
|
|
99
|
+
hash: latestBlockHash,
|
|
100
|
+
number: latestBlockNumber
|
|
101
|
+
},
|
|
102
|
+
proven: {
|
|
103
|
+
hash: provenBlockHash,
|
|
104
|
+
number: provenBlockNumber
|
|
105
|
+
},
|
|
106
|
+
finalized: {
|
|
107
|
+
hash: provenBlockHash,
|
|
108
|
+
number: provenBlockNumber
|
|
109
|
+
}
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
async handleBlockStreamEvent(event) {
|
|
113
|
+
this.log.debug(`Handling block stream event ${event.type}`);
|
|
114
|
+
switch(event.type){
|
|
115
|
+
case 'blocks-added':
|
|
116
|
+
await this.handleLatestL2Blocks(event.blocks);
|
|
117
|
+
break;
|
|
118
|
+
case 'chain-finalized':
|
|
119
|
+
break;
|
|
120
|
+
case 'chain-proven':
|
|
121
|
+
{
|
|
122
|
+
const from = await this.getSyncedProvenBlockNum() + 1;
|
|
123
|
+
const limit = event.blockNumber - from + 1;
|
|
124
|
+
await this.handleProvenL2Blocks(await this.l2BlockSource.getBlocks(from, limit));
|
|
125
|
+
break;
|
|
253
126
|
}
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
throw new Error(`Unknown filter ${filter}`);
|
|
262
|
-
}
|
|
263
|
-
}
|
|
264
|
-
/**
|
|
265
|
-
* Returns a transaction in the transaction pool by its hash.
|
|
266
|
-
* @param txHash - Hash of the transaction to look for in the pool.
|
|
267
|
-
* @returns A single tx or undefined.
|
|
268
|
-
*/
|
|
269
|
-
getTxByHashFromPool(txHash) {
|
|
270
|
-
return this.txPool.getTxByHash(txHash);
|
|
271
|
-
}
|
|
272
|
-
/**
|
|
273
|
-
* Returns a transaction in the transaction pool by its hash.
|
|
274
|
-
* If the transaction is not in the pool, it will be requested from the network.
|
|
275
|
-
* @param txHash - Hash of the transaction to look for in the pool.
|
|
276
|
-
* @returns A single tx or undefined.
|
|
277
|
-
*/
|
|
278
|
-
async getTxByHash(txHash) {
|
|
279
|
-
const tx = await this.txPool.getTxByHash(txHash);
|
|
280
|
-
if (tx) {
|
|
281
|
-
return tx;
|
|
127
|
+
case 'chain-pruned':
|
|
128
|
+
await this.handlePruneL2Blocks(event.blockNumber);
|
|
129
|
+
break;
|
|
130
|
+
default:
|
|
131
|
+
{
|
|
132
|
+
const _ = event;
|
|
133
|
+
break;
|
|
282
134
|
}
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
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
|
-
|
|
341
|
-
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
380
|
-
|
|
381
|
-
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
|
|
385
|
-
|
|
386
|
-
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
|
|
421
|
-
}
|
|
422
|
-
|
|
423
|
-
|
|
424
|
-
|
|
425
|
-
|
|
426
|
-
|
|
427
|
-
|
|
428
|
-
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
475
|
-
|
|
476
|
-
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
506
|
-
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
|
|
524
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicDJwX2NsaWVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jbGllbnQvcDJwX2NsaWVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBZUEsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sOEJBQThCLENBQUM7QUFDcEUsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBRXJELE9BQU8sRUFDTCxVQUFVLEVBRVYsc0JBQXNCLEVBQ3RCLFVBQVUsRUFDVixrQkFBa0IsRUFDbEIsU0FBUyxHQUNWLE1BQU0seUJBQXlCLENBQUM7QUFJakMsT0FBTyxFQUFrQixtQkFBbUIsRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUluRSxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUd0RTs7R0FFRztBQUNILE1BQU0sQ0FBTixJQUFZLGNBS1g7QUFMRCxXQUFZLGNBQWM7SUFDeEIsbURBQUksQ0FBQTtJQUNKLDJEQUFRLENBQUE7SUFDUix5REFBTyxDQUFBO0lBQ1AseURBQU8sQ0FBQTtBQUNULENBQUMsRUFMVyxjQUFjLEtBQWQsY0FBYyxRQUt6QjtBQWtJRDs7R0FFRztJQUNVLFNBQVM7O3NCQUNaLFVBQVU7OztzQkFEUCxTQUNYLFNBQVEsV0FBVTtZQTBCbEI7Ozs7Ozs7O2VBUUc7WUFDSCxZQUNFLFdBQWMsRUFDZCxLQUF3QixFQUNoQixhQUE0QixFQUNwQyxRQUFxQixFQUNiLFVBQXNCLEVBQzlCLFNBQTZCLEVBQUUsRUFDL0IsWUFBNkIsa0JBQWtCLEVBQUUsRUFDekMsTUFBTSxZQUFZLENBQUMsS0FBSyxDQUFDO2dCQUVqQyxLQUFLLENBQUMsU0FBUyxFQUFFLFdBQVcsQ0FBQyxDQUFDOztnQkFQdEIsa0JBQWEsSUF2Q1osbURBQVMsRUF1Q1YsYUFBYSxFQUFlO2dCQUU1QixlQUFVLEdBQVYsVUFBVSxDQUFZO2dCQUd0QixRQUFHLEdBQUgsR0FBRyxDQUFzQjtnQkFyQzNCLGlCQUFZ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
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
#assertIsReady() {
|
|
138
|
+
// this.log.info('Checking if p2p client is ready, current state: ', this.currentState);
|
|
139
|
+
if (!this.isReady()) {
|
|
140
|
+
throw new Error('P2P client not ready');
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Starts the P2P client.
|
|
145
|
+
* @returns An empty promise signalling the synching process.
|
|
146
|
+
*/ async start() {
|
|
147
|
+
if (this.currentState === 3) {
|
|
148
|
+
throw new Error('P2P client already stopped');
|
|
149
|
+
}
|
|
150
|
+
if (this.currentState !== 0) {
|
|
151
|
+
return this.syncPromise;
|
|
152
|
+
}
|
|
153
|
+
// get the current latest block numbers
|
|
154
|
+
this.latestBlockNumberAtStart = await this.l2BlockSource.getBlockNumber();
|
|
155
|
+
this.provenBlockNumberAtStart = await this.l2BlockSource.getProvenBlockNumber();
|
|
156
|
+
const syncedLatestBlock = await this.getSyncedLatestBlockNum() + 1;
|
|
157
|
+
const syncedProvenBlock = await this.getSyncedProvenBlockNum() + 1;
|
|
158
|
+
// if there are blocks to be retrieved, go to a synching state
|
|
159
|
+
if (syncedLatestBlock <= this.latestBlockNumberAtStart || syncedProvenBlock <= this.provenBlockNumberAtStart) {
|
|
160
|
+
this.setCurrentState(1);
|
|
161
|
+
this.syncPromise = new Promise((resolve)=>{
|
|
162
|
+
this.syncResolve = resolve;
|
|
163
|
+
});
|
|
164
|
+
this.log.verbose(`Starting sync from ${syncedLatestBlock} (last proven ${syncedProvenBlock})`);
|
|
165
|
+
} else {
|
|
166
|
+
// if no blocks to be retrieved, go straight to running
|
|
167
|
+
this.setCurrentState(2);
|
|
168
|
+
this.syncPromise = Promise.resolve();
|
|
169
|
+
await this.p2pService.start();
|
|
170
|
+
this.log.debug(`Block ${syncedLatestBlock} (proven ${syncedProvenBlock}) already beyond current block`);
|
|
171
|
+
}
|
|
172
|
+
this.blockStream.start();
|
|
173
|
+
this.log.verbose(`Started block downloader from block ${syncedLatestBlock}`);
|
|
174
|
+
return this.syncPromise;
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* Allows consumers to stop the instance of the P2P client.
|
|
178
|
+
* 'ready' will now return 'false' and the running promise that keeps the client synced is interrupted.
|
|
179
|
+
*/ async stop() {
|
|
180
|
+
this.log.debug('Stopping p2p client...');
|
|
181
|
+
await this.p2pService.stop();
|
|
182
|
+
this.log.debug('Stopped p2p service');
|
|
183
|
+
await this.blockStream.stop();
|
|
184
|
+
this.log.debug('Stopped block downloader');
|
|
185
|
+
await this.runningPromise;
|
|
186
|
+
this.setCurrentState(3);
|
|
187
|
+
this.log.info('P2P client stopped.');
|
|
188
|
+
}
|
|
189
|
+
broadcastProposal(proposal) {
|
|
190
|
+
this.log.verbose(`Broadcasting proposal for slot ${proposal.slotNumber.toNumber()} to peers`);
|
|
191
|
+
return this.p2pService.propagate(proposal);
|
|
192
|
+
}
|
|
193
|
+
async getAttestationsForSlot(slot, proposalId) {
|
|
194
|
+
return await this.attestationPool?.getAttestationsForSlot(slot, proposalId) ?? [];
|
|
195
|
+
}
|
|
196
|
+
// REVIEW: https://github.com/AztecProtocol/aztec-packages/issues/7963
|
|
197
|
+
// ^ This pattern is not my favorite (md)
|
|
198
|
+
registerBlockProposalHandler(handler) {
|
|
199
|
+
this.p2pService.registerBlockReceivedCallback(handler);
|
|
200
|
+
}
|
|
201
|
+
/**
|
|
202
|
+
* Requests the transactions with the given hashes from the network.
|
|
203
|
+
*
|
|
204
|
+
* If a transaction can be retrieved, it will be returned, if not an undefined
|
|
205
|
+
* will be returned. In place.
|
|
206
|
+
*
|
|
207
|
+
* @param txHashes - The hashes of the transactions to request.
|
|
208
|
+
* @returns A promise that resolves to an array of transactions or undefined.
|
|
209
|
+
*/ async requestTxs(txHashes) {
|
|
210
|
+
const res = await this.p2pService.sendBatchRequest(ReqRespSubProtocol.TX, txHashes);
|
|
211
|
+
return Promise.resolve(res ?? []);
|
|
212
|
+
}
|
|
213
|
+
/**
|
|
214
|
+
* Uses the Request Response protocol to request a transaction from the network.
|
|
215
|
+
*
|
|
216
|
+
* If the underlying request response protocol fails, then we return undefined.
|
|
217
|
+
* If it succeeds then we add the transaction to our transaction pool and return.
|
|
218
|
+
*
|
|
219
|
+
* @param txHash - The hash of the transaction to request.
|
|
220
|
+
* @returns A promise that resolves to a transaction or undefined.
|
|
221
|
+
*/ async requestTxByHash(txHash) {
|
|
222
|
+
const tx = await this.p2pService.sendRequest(ReqRespSubProtocol.TX, txHash);
|
|
223
|
+
if (tx) {
|
|
224
|
+
this.log.debug(`Received tx ${txHash.toString()} from peer`);
|
|
225
|
+
await this.txPool.addTxs([
|
|
226
|
+
tx
|
|
227
|
+
]);
|
|
228
|
+
} else {
|
|
229
|
+
this.log.debug(`Failed to receive tx ${txHash.toString()} from peer`);
|
|
230
|
+
}
|
|
231
|
+
return tx;
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Uses the batched Request Response protocol to request a set of transactions from the network.
|
|
235
|
+
*/ async requestTxsByHash(txHashes) {
|
|
236
|
+
const txs = await this.p2pService.sendBatchRequest(ReqRespSubProtocol.TX, txHashes) ?? [];
|
|
237
|
+
await this.txPool.addTxs(txs);
|
|
238
|
+
const txHashesStr = txHashes.map((tx)=>tx.toString()).join(', ');
|
|
239
|
+
this.log.debug(`Received batched txs ${txHashesStr} (${txs.length} / ${txHashes.length}}) from peers`);
|
|
240
|
+
return txs;
|
|
241
|
+
}
|
|
242
|
+
getPendingTxs() {
|
|
243
|
+
return Promise.resolve(this.getTxs('pending'));
|
|
244
|
+
}
|
|
245
|
+
async getPendingTxCount() {
|
|
246
|
+
const pendingTxs = await this.txPool.getPendingTxHashes();
|
|
247
|
+
return pendingTxs.length;
|
|
248
|
+
}
|
|
249
|
+
async *iteratePendingTxs() {
|
|
250
|
+
for (const txHash of (await this.txPool.getPendingTxHashes())){
|
|
251
|
+
const tx = await this.txPool.getTxByHash(txHash);
|
|
252
|
+
if (tx) {
|
|
253
|
+
yield tx;
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
/**
|
|
258
|
+
* Returns all transactions in the transaction pool.
|
|
259
|
+
* @returns An array of Txs.
|
|
260
|
+
*/ async getTxs(filter) {
|
|
261
|
+
if (filter === 'all') {
|
|
262
|
+
return this.txPool.getAllTxs();
|
|
263
|
+
} else if (filter === 'mined') {
|
|
264
|
+
const minedHashes = await this.txPool.getMinedTxHashes();
|
|
265
|
+
const minedTx = await Promise.all(minedHashes.map(([txHash])=>this.txPool.getTxByHash(txHash)));
|
|
266
|
+
return minedTx.filter((tx)=>!!tx);
|
|
267
|
+
} else if (filter === 'pending') {
|
|
268
|
+
const pendingHashses = await this.txPool.getPendingTxHashes();
|
|
269
|
+
const pendingTxs = await Promise.all(pendingHashses.map((txHash)=>this.txPool.getTxByHash(txHash)));
|
|
270
|
+
return pendingTxs.filter((tx)=>!!tx);
|
|
271
|
+
} else {
|
|
272
|
+
const _ = filter;
|
|
273
|
+
throw new Error(`Unknown filter ${filter}`);
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
/**
|
|
277
|
+
* Returns a transaction in the transaction pool by its hash.
|
|
278
|
+
* @param txHash - Hash of the transaction to look for in the pool.
|
|
279
|
+
* @returns A single tx or undefined.
|
|
280
|
+
*/ getTxByHashFromPool(txHash) {
|
|
281
|
+
return this.txPool.getTxByHash(txHash);
|
|
282
|
+
}
|
|
283
|
+
/**
|
|
284
|
+
* Returns a transaction in the transaction pool by its hash.
|
|
285
|
+
* If the transaction is not in the pool, it will be requested from the network.
|
|
286
|
+
* @param txHash - Hash of the transaction to look for in the pool.
|
|
287
|
+
* @returns A single tx or undefined.
|
|
288
|
+
*/ async getTxByHash(txHash) {
|
|
289
|
+
const tx = await this.txPool.getTxByHash(txHash);
|
|
290
|
+
if (tx) {
|
|
291
|
+
return tx;
|
|
292
|
+
}
|
|
293
|
+
return this.requestTxByHash(txHash);
|
|
294
|
+
}
|
|
295
|
+
/**
|
|
296
|
+
* Returns transactions in the transaction pool by hash.
|
|
297
|
+
* If a transaction is not in the pool, it will be requested from the network.
|
|
298
|
+
* @param txHashes - Hashes of the transactions to look for.
|
|
299
|
+
* @returns The txs found, not necessarily on the same order as the hashes.
|
|
300
|
+
*/ async getTxsByHash(txHashes) {
|
|
301
|
+
const txs = await Promise.all(txHashes.map((txHash)=>this.txPool.getTxByHash(txHash)));
|
|
302
|
+
const missingTxHashes = txs.map((tx, index)=>[
|
|
303
|
+
tx,
|
|
304
|
+
index
|
|
305
|
+
]).filter(([tx, _index])=>!tx).map(([_tx, index])=>txHashes[index]);
|
|
306
|
+
if (missingTxHashes.length === 0) {
|
|
307
|
+
return txs;
|
|
308
|
+
}
|
|
309
|
+
const missingTxs = await this.requestTxsByHash(missingTxHashes);
|
|
310
|
+
return txs.filter((tx)=>!!tx).concat(missingTxs);
|
|
311
|
+
}
|
|
312
|
+
/**
|
|
313
|
+
* Returns an archived transaction in the transaction pool by its hash.
|
|
314
|
+
* @param txHash - Hash of the archived transaction to look for.
|
|
315
|
+
* @returns A single tx or undefined.
|
|
316
|
+
*/ getArchivedTxByHash(txHash) {
|
|
317
|
+
return this.txPool.getArchivedTxByHash(txHash);
|
|
318
|
+
}
|
|
319
|
+
/**
|
|
320
|
+
* Verifies the 'tx' and, if valid, adds it to local tx pool and forwards it to other peers.
|
|
321
|
+
* @param tx - The tx to verify.
|
|
322
|
+
* @returns Empty promise.
|
|
323
|
+
**/ async sendTx(tx) {
|
|
324
|
+
this.#assertIsReady();
|
|
325
|
+
await this.txPool.addTxs([
|
|
326
|
+
tx
|
|
327
|
+
]);
|
|
328
|
+
this.p2pService.propagate(tx);
|
|
329
|
+
}
|
|
330
|
+
/**
|
|
331
|
+
* Returns whether the given tx hash is flagged as pending or mined.
|
|
332
|
+
* @param txHash - Hash of the tx to query.
|
|
333
|
+
* @returns Pending or mined depending on its status, or undefined if not found.
|
|
334
|
+
*/ getTxStatus(txHash) {
|
|
335
|
+
return this.txPool.getTxStatus(txHash);
|
|
336
|
+
}
|
|
337
|
+
getEnr() {
|
|
338
|
+
return this.p2pService.getEnr();
|
|
339
|
+
}
|
|
340
|
+
getEncodedEnr() {
|
|
341
|
+
return Promise.resolve(this.p2pService.getEnr()?.encodeTxt());
|
|
342
|
+
}
|
|
343
|
+
/**
|
|
344
|
+
* Deletes the 'txs' from the pool.
|
|
345
|
+
* NOT used if we use sendTx as reconcileTxPool will handle this.
|
|
346
|
+
* @param txHashes - Hashes of the transactions to delete.
|
|
347
|
+
* @returns Empty promise.
|
|
348
|
+
**/ async deleteTxs(txHashes) {
|
|
349
|
+
this.#assertIsReady();
|
|
350
|
+
await this.txPool.deleteTxs(txHashes);
|
|
351
|
+
}
|
|
352
|
+
/**
|
|
353
|
+
* Public function to check if the p2p client is fully synced and ready to receive txs.
|
|
354
|
+
* @returns True if the P2P client is ready to receive txs.
|
|
355
|
+
*/ isReady() {
|
|
356
|
+
return this.currentState === 2;
|
|
357
|
+
}
|
|
358
|
+
/**
|
|
359
|
+
* Public function to check the latest block number that the P2P client is synced to.
|
|
360
|
+
* @returns Block number of latest L2 Block we've synced with.
|
|
361
|
+
*/ async getSyncedLatestBlockNum() {
|
|
362
|
+
return await this.synchedLatestBlockNumber.getAsync() ?? INITIAL_L2_BLOCK_NUM - 1;
|
|
363
|
+
}
|
|
364
|
+
/**
|
|
365
|
+
* Public function to check the latest proven block number that the P2P client is synced to.
|
|
366
|
+
* @returns Block number of latest proven L2 Block we've synced with.
|
|
367
|
+
*/ async getSyncedProvenBlockNum() {
|
|
368
|
+
return await this.synchedProvenBlockNumber.getAsync() ?? INITIAL_L2_BLOCK_NUM - 1;
|
|
369
|
+
}
|
|
370
|
+
/**
|
|
371
|
+
* Method to check the status the p2p client.
|
|
372
|
+
* @returns Information about p2p client status: state & syncedToBlockNum.
|
|
373
|
+
*/ async getStatus() {
|
|
374
|
+
const blockNumber = await this.getSyncedLatestBlockNum();
|
|
375
|
+
const blockHash = blockNumber === 0 ? '' : await this.l2BlockSource.getBlockHeader(blockNumber).then((header)=>header?.hash()).then((hash)=>hash?.toString());
|
|
376
|
+
return {
|
|
377
|
+
state: this.currentState,
|
|
378
|
+
syncedToL2Block: {
|
|
379
|
+
number: blockNumber,
|
|
380
|
+
hash: blockHash
|
|
381
|
+
}
|
|
382
|
+
};
|
|
383
|
+
}
|
|
384
|
+
/**
|
|
385
|
+
* Mark all txs from these blocks as mined.
|
|
386
|
+
* @param blocks - A list of existing blocks with txs that the P2P client needs to ensure the tx pool is reconciled with.
|
|
387
|
+
* @returns Empty promise.
|
|
388
|
+
*/ async markTxsAsMinedFromBlocks(blocks) {
|
|
389
|
+
for (const block of blocks){
|
|
390
|
+
const txHashes = block.body.txEffects.map((txEffect)=>txEffect.txHash);
|
|
391
|
+
await this.txPool.markAsMined(txHashes, block.number);
|
|
392
|
+
}
|
|
393
|
+
}
|
|
394
|
+
/**
|
|
395
|
+
* Deletes txs from these blocks.
|
|
396
|
+
* @param blocks - A list of existing blocks with txs that the P2P client needs to ensure the tx pool is reconciled with.
|
|
397
|
+
* @returns Empty promise.
|
|
398
|
+
*/ async deleteTxsFromBlocks(blocks) {
|
|
399
|
+
this.log.debug(`Deleting txs from blocks ${blocks[0].number} to ${blocks[blocks.length - 1].number}`);
|
|
400
|
+
for (const block of blocks){
|
|
401
|
+
const txHashes = block.body.txEffects.map((txEffect)=>txEffect.txHash);
|
|
402
|
+
await this.txPool.deleteTxs(txHashes);
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
/**
|
|
406
|
+
* Handles new mined blocks by marking the txs in them as mined.
|
|
407
|
+
* @param blocks - A list of existing blocks with txs that the P2P client needs to ensure the tx pool is reconciled with.
|
|
408
|
+
* @returns Empty promise.
|
|
409
|
+
*/ async handleLatestL2Blocks(blocks) {
|
|
410
|
+
if (!blocks.length) {
|
|
411
|
+
return Promise.resolve();
|
|
412
|
+
}
|
|
413
|
+
await this.markTxsAsMinedFromBlocks(blocks);
|
|
414
|
+
const lastBlockNum = blocks[blocks.length - 1].number;
|
|
415
|
+
await Promise.all(blocks.map(async (block)=>this.synchedBlockHashes.set(block.number, (await block.hash()).toString())));
|
|
416
|
+
await this.synchedLatestBlockNumber.set(lastBlockNum);
|
|
417
|
+
this.log.verbose(`Synched to latest block ${lastBlockNum}`);
|
|
418
|
+
await this.startServiceIfSynched();
|
|
419
|
+
}
|
|
420
|
+
/**
|
|
421
|
+
* Handles new proven blocks by deleting the txs in them, or by deleting the txs in blocks `keepProvenTxsFor` ago.
|
|
422
|
+
* @param blocks - A list of proven L2 blocks.
|
|
423
|
+
* @returns Empty promise.
|
|
424
|
+
*/ async handleProvenL2Blocks(blocks) {
|
|
425
|
+
if (!blocks.length) {
|
|
426
|
+
return Promise.resolve();
|
|
427
|
+
}
|
|
428
|
+
const firstBlockNum = blocks[0].number;
|
|
429
|
+
const lastBlockNum = blocks[blocks.length - 1].number;
|
|
430
|
+
const lastBlockSlot = blocks[blocks.length - 1].header.globalVariables.slotNumber.toBigInt();
|
|
431
|
+
// If keepProvenTxsFor is 0, we delete all txs from all proven blocks.
|
|
432
|
+
if (this.keepProvenTxsFor === 0) {
|
|
433
|
+
await this.deleteTxsFromBlocks(blocks);
|
|
434
|
+
} else if (lastBlockNum - this.keepProvenTxsFor >= INITIAL_L2_BLOCK_NUM) {
|
|
435
|
+
const fromBlock = Math.max(INITIAL_L2_BLOCK_NUM, firstBlockNum - this.keepProvenTxsFor);
|
|
436
|
+
const toBlock = lastBlockNum - this.keepProvenTxsFor;
|
|
437
|
+
const limit = toBlock - fromBlock + 1;
|
|
438
|
+
const blocksToDeleteTxsFrom = await this.l2BlockSource.getBlocks(fromBlock, limit, true);
|
|
439
|
+
await this.deleteTxsFromBlocks(blocksToDeleteTxsFrom);
|
|
440
|
+
}
|
|
441
|
+
// We delete attestations older than the last block slot minus the number of slots we want to keep in the pool.
|
|
442
|
+
const lastBlockSlotMinusKeepAttestationsInPoolFor = lastBlockSlot - BigInt(this.keepAttestationsInPoolFor);
|
|
443
|
+
if (lastBlockSlotMinusKeepAttestationsInPoolFor >= BigInt(INITIAL_L2_BLOCK_NUM)) {
|
|
444
|
+
await this.attestationPool?.deleteAttestationsOlderThan(lastBlockSlotMinusKeepAttestationsInPoolFor);
|
|
445
|
+
}
|
|
446
|
+
await this.synchedProvenBlockNumber.set(lastBlockNum);
|
|
447
|
+
this.log.debug(`Synched to proven block ${lastBlockNum}`);
|
|
448
|
+
await this.startServiceIfSynched();
|
|
449
|
+
}
|
|
450
|
+
/**
|
|
451
|
+
* Updates the tx pool after a chain prune.
|
|
452
|
+
* @param latestBlock - The block number the chain was pruned to.
|
|
453
|
+
*/ async handlePruneL2Blocks(latestBlock) {
|
|
454
|
+
const txsToDelete = [];
|
|
455
|
+
for (const tx of (await this.txPool.getAllTxs())){
|
|
456
|
+
// every tx that's been generated against a block that has now been pruned is no longer valid
|
|
457
|
+
if (tx.data.constants.historicalHeader.globalVariables.blockNumber.toNumber() > latestBlock) {
|
|
458
|
+
txsToDelete.push(await tx.getTxHash());
|
|
459
|
+
}
|
|
460
|
+
}
|
|
461
|
+
this.log.info(`Detected chain prune. Removing invalid txs count=${txsToDelete.length} newLatestBlock=${latestBlock} previousLatestBlock=${this.getSyncedLatestBlockNum()}`);
|
|
462
|
+
// delete invalid txs (both pending and mined)
|
|
463
|
+
await this.txPool.deleteTxs(txsToDelete);
|
|
464
|
+
// everything left in the mined set was built against a block on the proven chain so its still valid
|
|
465
|
+
// move back to pending the txs that were reorged out of the chain
|
|
466
|
+
// NOTE: we can't move _all_ txs back to pending because the tx pool could keep hold of mined txs for longer
|
|
467
|
+
// (see this.keepProvenTxsFor)
|
|
468
|
+
const txsToMoveToPending = [];
|
|
469
|
+
for (const [txHash, blockNumber] of (await this.txPool.getMinedTxHashes())){
|
|
470
|
+
if (blockNumber > latestBlock) {
|
|
471
|
+
txsToMoveToPending.push(txHash);
|
|
472
|
+
}
|
|
473
|
+
}
|
|
474
|
+
this.log.info(`Moving ${txsToMoveToPending.length} mined txs back to pending`);
|
|
475
|
+
await this.txPool.markMinedAsPending(txsToMoveToPending);
|
|
476
|
+
await this.synchedLatestBlockNumber.set(latestBlock);
|
|
477
|
+
// no need to update block hashes, as they will be updated as new blocks are added
|
|
478
|
+
}
|
|
479
|
+
async startServiceIfSynched() {
|
|
480
|
+
if (this.currentState === 1 && await this.getSyncedLatestBlockNum() >= this.latestBlockNumberAtStart && await this.getSyncedProvenBlockNum() >= this.provenBlockNumberAtStart) {
|
|
481
|
+
this.log.debug(`Synched to blocks at start`);
|
|
482
|
+
this.setCurrentState(2);
|
|
483
|
+
if (this.syncResolve !== undefined) {
|
|
484
|
+
this.syncResolve();
|
|
485
|
+
await this.p2pService.start();
|
|
486
|
+
}
|
|
487
|
+
}
|
|
488
|
+
}
|
|
489
|
+
/**
|
|
490
|
+
* Method to set the value of the current state.
|
|
491
|
+
* @param newState - New state value.
|
|
492
|
+
*/ setCurrentState(newState) {
|
|
493
|
+
const oldState = this.currentState;
|
|
494
|
+
this.currentState = newState;
|
|
495
|
+
this.log.debug(`Moved from state ${P2PClientState[oldState]} to ${P2PClientState[this.currentState]}`);
|
|
496
|
+
}
|
|
497
|
+
}
|
|
498
|
+
_ts_decorate([
|
|
499
|
+
trackSpan('p2pClient.broadcastProposal', async (proposal)=>({
|
|
500
|
+
[Attributes.BLOCK_NUMBER]: proposal.blockNumber.toNumber(),
|
|
501
|
+
[Attributes.SLOT_NUMBER]: proposal.slotNumber.toNumber(),
|
|
502
|
+
[Attributes.BLOCK_ARCHIVE]: proposal.archive.toString(),
|
|
503
|
+
[Attributes.P2P_ID]: (await proposal.p2pMessageIdentifier()).toString()
|
|
504
|
+
}))
|
|
505
|
+
], P2PClient.prototype, "broadcastProposal", null);
|