@aztec/p2p 0.57.0 → 0.59.0

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.
Files changed (107) hide show
  1. package/dest/client/index.d.ts +4 -4
  2. package/dest/client/index.d.ts.map +1 -1
  3. package/dest/client/index.js +11 -9
  4. package/dest/client/p2p_client.d.ts +9 -10
  5. package/dest/client/p2p_client.d.ts.map +1 -1
  6. package/dest/client/p2p_client.js +375 -350
  7. package/dest/config.js +3 -3
  8. package/dest/index.d.ts +3 -3
  9. package/dest/index.d.ts.map +1 -1
  10. package/dest/index.js +4 -4
  11. package/dest/mem_pools/attestation_pool/attestation_pool.d.ts.map +1 -0
  12. package/dest/{attestation_pool → mem_pools/attestation_pool}/attestation_pool.js +1 -1
  13. package/dest/mem_pools/attestation_pool/index.d.ts.map +1 -0
  14. package/dest/{attestation_pool → mem_pools/attestation_pool}/index.js +1 -1
  15. package/dest/{attestation_pool → mem_pools/attestation_pool}/memory_attestation_pool.d.ts +4 -1
  16. package/dest/mem_pools/attestation_pool/memory_attestation_pool.d.ts.map +1 -0
  17. package/dest/mem_pools/attestation_pool/memory_attestation_pool.js +114 -0
  18. package/dest/mem_pools/attestation_pool/mocks.d.ts.map +1 -0
  19. package/dest/mem_pools/attestation_pool/mocks.js +31 -0
  20. package/dest/mem_pools/epoch_proof_quote_pool/epoch_proof_quote_pool.d.ts.map +1 -0
  21. package/dest/mem_pools/epoch_proof_quote_pool/epoch_proof_quote_pool.js +2 -0
  22. package/dest/mem_pools/epoch_proof_quote_pool/index.d.ts.map +1 -0
  23. package/dest/{epoch_proof_quote_pool → mem_pools/epoch_proof_quote_pool}/index.js +1 -1
  24. package/dest/{epoch_proof_quote_pool → mem_pools/epoch_proof_quote_pool}/memory_epoch_proof_quote_pool.d.ts +3 -1
  25. package/dest/mem_pools/epoch_proof_quote_pool/memory_epoch_proof_quote_pool.d.ts.map +1 -0
  26. package/dest/mem_pools/epoch_proof_quote_pool/memory_epoch_proof_quote_pool.js +30 -0
  27. package/dest/mem_pools/epoch_proof_quote_pool/test_utils.d.ts.map +1 -0
  28. package/dest/mem_pools/epoch_proof_quote_pool/test_utils.js +21 -0
  29. package/dest/mem_pools/index.d.ts +5 -0
  30. package/dest/mem_pools/index.d.ts.map +1 -0
  31. package/dest/mem_pools/index.js +2 -0
  32. package/dest/mem_pools/instrumentation.d.ts +25 -0
  33. package/dest/mem_pools/instrumentation.d.ts.map +1 -0
  34. package/dest/mem_pools/instrumentation.js +70 -0
  35. package/dest/mem_pools/interface.d.ts +12 -0
  36. package/dest/mem_pools/interface.d.ts.map +1 -0
  37. package/dest/mem_pools/interface.js +2 -0
  38. package/dest/mem_pools/tx_pool/aztec_kv_tx_pool.d.ts.map +1 -0
  39. package/dest/{tx_pool → mem_pools/tx_pool}/aztec_kv_tx_pool.js +9 -9
  40. package/dest/mem_pools/tx_pool/index.d.ts.map +1 -0
  41. package/dest/{tx_pool → mem_pools/tx_pool}/index.js +1 -1
  42. package/dest/mem_pools/tx_pool/memory_tx_pool.d.ts.map +1 -0
  43. package/dest/mem_pools/tx_pool/memory_tx_pool.js +111 -0
  44. package/dest/mem_pools/tx_pool/tx_pool.d.ts.map +1 -0
  45. package/dest/{tx_pool → mem_pools/tx_pool}/tx_pool.js +1 -1
  46. package/dest/mem_pools/tx_pool/tx_pool_test_suite.d.ts.map +1 -0
  47. package/dest/mem_pools/tx_pool/tx_pool_test_suite.js +55 -0
  48. package/dest/service/libp2p_service.d.ts +12 -7
  49. package/dest/service/libp2p_service.d.ts.map +1 -1
  50. package/dest/service/libp2p_service.js +453 -381
  51. package/package.json +6 -6
  52. package/src/client/index.ts +16 -20
  53. package/src/client/p2p_client.ts +33 -10
  54. package/src/config.ts +2 -2
  55. package/src/index.ts +3 -3
  56. package/src/{attestation_pool → mem_pools/attestation_pool}/memory_attestation_pool.ts +40 -8
  57. package/src/{epoch_proof_quote_pool → mem_pools/epoch_proof_quote_pool}/memory_epoch_proof_quote_pool.ts +17 -1
  58. package/src/mem_pools/index.ts +4 -0
  59. package/src/mem_pools/instrumentation.ts +85 -0
  60. package/src/mem_pools/interface.ts +12 -0
  61. package/src/{tx_pool → mem_pools/tx_pool}/aztec_kv_tx_pool.ts +9 -9
  62. package/src/{tx_pool → mem_pools/tx_pool}/memory_tx_pool.ts +9 -9
  63. package/src/service/libp2p_service.ts +76 -18
  64. package/dest/attestation_pool/attestation_pool.d.ts.map +0 -1
  65. package/dest/attestation_pool/index.d.ts.map +0 -1
  66. package/dest/attestation_pool/memory_attestation_pool.d.ts.map +0 -1
  67. package/dest/attestation_pool/memory_attestation_pool.js +0 -89
  68. package/dest/attestation_pool/mocks.d.ts.map +0 -1
  69. package/dest/attestation_pool/mocks.js +0 -31
  70. package/dest/epoch_proof_quote_pool/epoch_proof_quote_pool.d.ts.map +0 -1
  71. package/dest/epoch_proof_quote_pool/epoch_proof_quote_pool.js +0 -2
  72. package/dest/epoch_proof_quote_pool/index.d.ts.map +0 -1
  73. package/dest/epoch_proof_quote_pool/memory_epoch_proof_quote_pool.d.ts.map +0 -1
  74. package/dest/epoch_proof_quote_pool/memory_epoch_proof_quote_pool.js +0 -22
  75. package/dest/epoch_proof_quote_pool/test_utils.d.ts.map +0 -1
  76. package/dest/epoch_proof_quote_pool/test_utils.js +0 -21
  77. package/dest/tx_pool/aztec_kv_tx_pool.d.ts.map +0 -1
  78. package/dest/tx_pool/index.d.ts.map +0 -1
  79. package/dest/tx_pool/instrumentation.d.ts +0 -25
  80. package/dest/tx_pool/instrumentation.d.ts.map +0 -1
  81. package/dest/tx_pool/instrumentation.js +0 -61
  82. package/dest/tx_pool/memory_tx_pool.d.ts.map +0 -1
  83. package/dest/tx_pool/memory_tx_pool.js +0 -111
  84. package/dest/tx_pool/tx_pool.d.ts.map +0 -1
  85. package/dest/tx_pool/tx_pool_test_suite.d.ts.map +0 -1
  86. package/dest/tx_pool/tx_pool_test_suite.js +0 -55
  87. package/src/tx_pool/instrumentation.ts +0 -73
  88. /package/dest/{attestation_pool → mem_pools/attestation_pool}/attestation_pool.d.ts +0 -0
  89. /package/dest/{attestation_pool → mem_pools/attestation_pool}/index.d.ts +0 -0
  90. /package/dest/{attestation_pool → mem_pools/attestation_pool}/mocks.d.ts +0 -0
  91. /package/dest/{epoch_proof_quote_pool → mem_pools/epoch_proof_quote_pool}/epoch_proof_quote_pool.d.ts +0 -0
  92. /package/dest/{epoch_proof_quote_pool → mem_pools/epoch_proof_quote_pool}/index.d.ts +0 -0
  93. /package/dest/{epoch_proof_quote_pool → mem_pools/epoch_proof_quote_pool}/test_utils.d.ts +0 -0
  94. /package/dest/{tx_pool → mem_pools/tx_pool}/aztec_kv_tx_pool.d.ts +0 -0
  95. /package/dest/{tx_pool → mem_pools/tx_pool}/index.d.ts +0 -0
  96. /package/dest/{tx_pool → mem_pools/tx_pool}/memory_tx_pool.d.ts +0 -0
  97. /package/dest/{tx_pool → mem_pools/tx_pool}/tx_pool.d.ts +0 -0
  98. /package/dest/{tx_pool → mem_pools/tx_pool}/tx_pool_test_suite.d.ts +0 -0
  99. /package/src/{attestation_pool → mem_pools/attestation_pool}/attestation_pool.ts +0 -0
  100. /package/src/{attestation_pool → mem_pools/attestation_pool}/index.ts +0 -0
  101. /package/src/{attestation_pool → mem_pools/attestation_pool}/mocks.ts +0 -0
  102. /package/src/{epoch_proof_quote_pool → mem_pools/epoch_proof_quote_pool}/epoch_proof_quote_pool.ts +0 -0
  103. /package/src/{epoch_proof_quote_pool → mem_pools/epoch_proof_quote_pool}/index.ts +0 -0
  104. /package/src/{epoch_proof_quote_pool → mem_pools/epoch_proof_quote_pool}/test_utils.ts +0 -0
  105. /package/src/{tx_pool → mem_pools/tx_pool}/index.ts +0 -0
  106. /package/src/{tx_pool → mem_pools/tx_pool}/tx_pool.ts +0 -0
  107. /package/src/{tx_pool → mem_pools/tx_pool}/tx_pool_test_suite.ts +0 -0
@@ -1,8 +1,8 @@
1
- var _P2PClient_instances, _P2PClient_assertIsReady;
2
- import { __classPrivateFieldGet } from "tslib";
1
+ import { __classPrivateFieldGet, __esDecorate, __runInitializers } from "tslib";
3
2
  import { L2BlockDownloader, } from '@aztec/circuit-types';
4
3
  import { INITIAL_L2_BLOCK_NUM } from '@aztec/circuits.js/constants';
5
4
  import { createDebugLogger } from '@aztec/foundation/log';
5
+ import { Attributes, WithTracer, trackSpan } from '@aztec/telemetry-client';
6
6
  import { getP2PConfigEnvVars } from '../config.js';
7
7
  import { TX_REQ_PROTOCOL } from '../service/reqresp/interface.js';
8
8
  /**
@@ -18,354 +18,379 @@ export var P2PClientState;
18
18
  /**
19
19
  * The P2P client implementation.
20
20
  */
21
- export class P2PClient {
22
- /**
23
- * In-memory P2P client constructor.
24
- * @param store - The client's instance of the KV store.
25
- * @param l2BlockSource - P2P client's source for fetching existing blocks.
26
- * @param txPool - The client's instance of a transaction pool. Defaults to in-memory implementation.
27
- * @param p2pService - The concrete instance of p2p networking to use.
28
- * @param keepProvenTxsFor - How many blocks have to pass after a block is proven before its txs are deleted (zero to delete immediately once proven).
29
- * @param log - A logger.
30
- */
31
- constructor(store, l2BlockSource, txPool, attestationPool, epochProofQuotePool, p2pService, keepProvenTxsFor, log = createDebugLogger('aztec:p2p')) {
32
- _P2PClient_instances.add(this);
33
- this.l2BlockSource = l2BlockSource;
34
- this.txPool = txPool;
35
- this.attestationPool = attestationPool;
36
- this.epochProofQuotePool = epochProofQuotePool;
37
- this.p2pService = p2pService;
38
- this.keepProvenTxsFor = keepProvenTxsFor;
39
- this.log = log;
40
- /** Property that indicates whether the client is running. */
41
- this.stopping = false;
42
- this.currentState = P2PClientState.IDLE;
43
- this.syncPromise = Promise.resolve();
44
- this.syncResolve = undefined;
45
- this.latestBlockNumberAtStart = -1;
46
- this.provenBlockNumberAtStart = -1;
47
- const { blockCheckIntervalMS: checkInterval, l2QueueSize: p2pL2QueueSize } = getP2PConfigEnvVars();
48
- const l2DownloaderOpts = { maxQueueSize: p2pL2QueueSize, pollIntervalMS: checkInterval };
49
- // TODO(palla/prover-node): This effectively downloads blocks twice from the archiver, which is an issue
50
- // if the archiver is remote. We should refactor this so the downloader keeps a single queue and handles
51
- // latest/proven metadata, as well as block reorgs.
52
- this.latestBlockDownloader = new L2BlockDownloader(l2BlockSource, l2DownloaderOpts);
53
- this.provenBlockDownloader = new L2BlockDownloader(l2BlockSource, { ...l2DownloaderOpts, proven: true });
54
- this.synchedLatestBlockNumber = store.openSingleton('p2p_pool_last_l2_block');
55
- this.synchedProvenBlockNumber = store.openSingleton('p2p_pool_last_proven_l2_block');
56
- }
57
- getEpochProofQuotes(epoch) {
58
- return Promise.resolve(this.epochProofQuotePool.getQuotes(epoch));
59
- }
60
- broadcastEpochProofQuote(quote) {
61
- __classPrivateFieldGet(this, _P2PClient_instances, "m", _P2PClient_assertIsReady).call(this);
62
- this.epochProofQuotePool.addQuote(quote);
63
- return this.p2pService.propagate(quote);
64
- }
65
- /**
66
- * Starts the P2P client.
67
- * @returns An empty promise signalling the synching process.
68
- */
69
- async start() {
70
- if (this.currentState === P2PClientState.STOPPED) {
71
- throw new Error('P2P client already stopped');
72
- }
73
- if (this.currentState !== P2PClientState.IDLE) {
74
- return this.syncPromise;
75
- }
76
- // get the current latest block numbers
77
- this.latestBlockNumberAtStart = await this.l2BlockSource.getBlockNumber();
78
- this.provenBlockNumberAtStart = await this.l2BlockSource.getProvenBlockNumber();
79
- const syncedLatestBlock = this.getSyncedLatestBlockNum() + 1;
80
- const syncedProvenBlock = this.getSyncedProvenBlockNum() + 1;
81
- // if there are blocks to be retrieved, go to a synching state
82
- if (syncedLatestBlock <= this.latestBlockNumberAtStart || syncedProvenBlock <= this.provenBlockNumberAtStart) {
83
- this.setCurrentState(P2PClientState.SYNCHING);
84
- this.syncPromise = new Promise(resolve => {
85
- this.syncResolve = resolve;
86
- });
87
- this.log.verbose(`Starting sync from ${syncedLatestBlock} (last proven ${syncedProvenBlock})`);
88
- }
89
- else {
90
- // if no blocks to be retrieved, go straight to running
91
- this.setCurrentState(P2PClientState.RUNNING);
92
- this.syncPromise = Promise.resolve();
93
- await this.p2pService.start();
94
- this.log.verbose(`Block ${syncedLatestBlock} (proven ${syncedProvenBlock}) already beyond current block`);
95
- }
96
- // publish any txs in TxPool after its doing initial sync
97
- this.syncPromise = this.syncPromise.then(() => this.publishStoredTxs());
98
- // start looking for further blocks
99
- const processLatest = async () => {
100
- while (!this.stopping) {
101
- await this.latestBlockDownloader.getBlocks(1).then(this.handleLatestL2Blocks.bind(this));
21
+ let P2PClient = (() => {
22
+ var _P2PClient_instances, _a, _P2PClient_assertIsReady;
23
+ let _classSuper = WithTracer;
24
+ let _instanceExtraInitializers = [];
25
+ let _broadcastProposal_decorators;
26
+ return _a = class P2PClient extends _classSuper {
27
+ /**
28
+ * In-memory P2P client constructor.
29
+ * @param store - The client's instance of the KV store.
30
+ * @param l2BlockSource - P2P client's source for fetching existing blocks.
31
+ * @param txPool - The client's instance of a transaction pool. Defaults to in-memory implementation.
32
+ * @param p2pService - The concrete instance of p2p networking to use.
33
+ * @param keepProvenTxsFor - How many blocks have to pass after a block is proven before its txs are deleted (zero to delete immediately once proven).
34
+ * @param log - A logger.
35
+ */
36
+ constructor(store, l2BlockSource, mempools, p2pService, keepProvenTxsFor, telemetryClient, log = createDebugLogger('aztec:p2p')) {
37
+ super(telemetryClient, 'P2PClient');
38
+ _P2PClient_instances.add(this);
39
+ this.l2BlockSource = (__runInitializers(this, _instanceExtraInitializers), l2BlockSource);
40
+ this.p2pService = p2pService;
41
+ this.keepProvenTxsFor = keepProvenTxsFor;
42
+ this.log = log;
43
+ /** Property that indicates whether the client is running. */
44
+ this.stopping = false;
45
+ this.currentState = P2PClientState.IDLE;
46
+ this.syncPromise = Promise.resolve();
47
+ this.syncResolve = undefined;
48
+ this.latestBlockNumberAtStart = -1;
49
+ this.provenBlockNumberAtStart = -1;
50
+ const { blockCheckIntervalMS: checkInterval, l2QueueSize: p2pL2QueueSize } = getP2PConfigEnvVars();
51
+ const l2DownloaderOpts = { maxQueueSize: p2pL2QueueSize, pollIntervalMS: checkInterval };
52
+ // TODO(palla/prover-node): This effectively downloads blocks twice from the archiver, which is an issue
53
+ // if the archiver is remote. We should refactor this so the downloader keeps a single queue and handles
54
+ // latest/proven metadata, as well as block reorgs.
55
+ this.latestBlockDownloader = new L2BlockDownloader(l2BlockSource, l2DownloaderOpts);
56
+ this.provenBlockDownloader = new L2BlockDownloader(l2BlockSource, { ...l2DownloaderOpts, proven: true });
57
+ this.synchedLatestBlockNumber = store.openSingleton('p2p_pool_last_l2_block');
58
+ this.synchedProvenBlockNumber = store.openSingleton('p2p_pool_last_proven_l2_block');
59
+ this.txPool = mempools.txPool;
60
+ this.attestationPool = mempools.attestationPool;
61
+ this.epochProofQuotePool = mempools.epochProofQuotePool;
102
62
  }
103
- };
104
- const processProven = async () => {
105
- while (!this.stopping) {
106
- await this.provenBlockDownloader.getBlocks(1).then(this.handleProvenL2Blocks.bind(this));
63
+ getEpochProofQuotes(epoch) {
64
+ return Promise.resolve(this.epochProofQuotePool.getQuotes(epoch));
107
65
  }
108
- };
109
- this.runningPromise = Promise.all([processLatest(), processProven()]).then(() => { });
110
- this.latestBlockDownloader.start(syncedLatestBlock);
111
- this.provenBlockDownloader.start(syncedLatestBlock);
112
- this.log.verbose(`Started block downloader from block ${syncedLatestBlock}`);
113
- return this.syncPromise;
114
- }
115
- /**
116
- * Allows consumers to stop the instance of the P2P client.
117
- * 'ready' will now return 'false' and the running promise that keeps the client synced is interrupted.
118
- */
119
- async stop() {
120
- this.log.debug('Stopping p2p client...');
121
- this.stopping = true;
122
- await this.p2pService.stop();
123
- this.log.debug('Stopped p2p service');
124
- await this.latestBlockDownloader.stop();
125
- await this.provenBlockDownloader.stop();
126
- this.log.debug('Stopped block downloader');
127
- await this.runningPromise;
128
- this.setCurrentState(P2PClientState.STOPPED);
129
- this.log.info('P2P client stopped.');
130
- }
131
- broadcastProposal(proposal) {
132
- this.log.verbose(`Broadcasting proposal ${proposal.p2pMessageIdentifier()} to peers`);
133
- return this.p2pService.propagate(proposal);
134
- }
135
- getAttestationsForSlot(slot, proposalId) {
136
- return Promise.resolve(this.attestationPool.getAttestationsForSlot(slot, proposalId));
137
- }
138
- // REVIEW: https://github.com/AztecProtocol/aztec-packages/issues/7963
139
- // ^ This pattern is not my favorite (md)
140
- registerBlockProposalHandler(handler) {
141
- this.p2pService.registerBlockReceivedCallback(handler);
142
- }
143
- /**
144
- * Requests the transactions with the given hashes from the network.
145
- *
146
- * If a transaction can be retrieved, it will be returned, if not an undefined
147
- * will be returned. In place.
148
- *
149
- * @param txHashes - The hashes of the transactions to request.
150
- * @returns A promise that resolves to an array of transactions or undefined.
151
- */
152
- requestTxs(txHashes) {
153
- const requestPromises = txHashes.map(txHash => this.requestTxByHash(txHash));
154
- return Promise.all(requestPromises);
155
- }
156
- /**
157
- * Uses the Request Response protocol to request a transaction from the network.
158
- *
159
- * If the underlying request response protocol fails, then we return undefined.
160
- * If it succeeds then we add the transaction to our transaction pool and return.
161
- *
162
- * @param txHash - The hash of the transaction to request.
163
- * @returns A promise that resolves to a transaction or undefined.
164
- */
165
- async requestTxByHash(txHash) {
166
- const tx = await this.p2pService.sendRequest(TX_REQ_PROTOCOL, txHash);
167
- this.log.debug(`Requested ${txHash.toString()} from peer | success = ${!!tx}`);
168
- if (tx) {
169
- await this.txPool.addTxs([tx]);
170
- }
171
- return tx;
172
- }
173
- /**
174
- * Returns all transactions in the transaction pool.
175
- * @returns An array of Txs.
176
- */
177
- getTxs(filter) {
178
- if (filter === 'all') {
179
- return this.txPool.getAllTxs();
180
- }
181
- else if (filter === 'mined') {
182
- return this.txPool
183
- .getMinedTxHashes()
184
- .map(txHash => this.txPool.getTxByHash(txHash))
185
- .filter((tx) => !!tx);
186
- }
187
- else if (filter === 'pending') {
188
- return this.txPool
189
- .getPendingTxHashes()
190
- .map(txHash => this.txPool.getTxByHash(txHash))
191
- .filter((tx) => !!tx);
192
- }
193
- else {
194
- const _ = filter;
195
- throw new Error(`Unknown filter ${filter}`);
196
- }
197
- }
198
- /**
199
- * Returns a transaction in the transaction pool by its hash.
200
- * @param txHash - Hash of the transaction to look for in the pool.
201
- * @returns A single tx or undefined.
202
- */
203
- getTxByHash(txHash) {
204
- return this.txPool.getTxByHash(txHash);
205
- }
206
- /**
207
- * Verifies the 'tx' and, if valid, adds it to local tx pool and forwards it to other peers.
208
- * @param tx - The tx to verify.
209
- * @returns Empty promise.
210
- **/
211
- async sendTx(tx) {
212
- __classPrivateFieldGet(this, _P2PClient_instances, "m", _P2PClient_assertIsReady).call(this);
213
- await this.txPool.addTxs([tx]);
214
- this.p2pService.propagate(tx);
215
- }
216
- /**
217
- * Returns whether the given tx hash is flagged as pending or mined.
218
- * @param txHash - Hash of the tx to query.
219
- * @returns Pending or mined depending on its status, or undefined if not found.
220
- */
221
- getTxStatus(txHash) {
222
- return this.txPool.getTxStatus(txHash);
223
- }
224
- getEnr() {
225
- return this.p2pService.getEnr();
226
- }
227
- /**
228
- * Deletes the 'txs' from the pool.
229
- * NOT used if we use sendTx as reconcileTxPool will handle this.
230
- * @param txHashes - Hashes of the transactions to delete.
231
- * @returns Empty promise.
232
- **/
233
- async deleteTxs(txHashes) {
234
- __classPrivateFieldGet(this, _P2PClient_instances, "m", _P2PClient_assertIsReady).call(this);
235
- await this.txPool.deleteTxs(txHashes);
236
- }
237
- /**
238
- * Public function to check if the p2p client is fully synced and ready to receive txs.
239
- * @returns True if the P2P client is ready to receive txs.
240
- */
241
- isReady() {
242
- return this.currentState === P2PClientState.RUNNING;
243
- }
244
- /**
245
- * Public function to check the latest block number that the P2P client is synced to.
246
- * @returns Block number of latest L2 Block we've synced with.
247
- */
248
- getSyncedLatestBlockNum() {
249
- return this.synchedLatestBlockNumber.get() ?? INITIAL_L2_BLOCK_NUM - 1;
250
- }
251
- /**
252
- * Public function to check the latest proven block number that the P2P client is synced to.
253
- * @returns Block number of latest proven L2 Block we've synced with.
254
- */
255
- getSyncedProvenBlockNum() {
256
- return this.synchedProvenBlockNumber.get() ?? INITIAL_L2_BLOCK_NUM - 1;
257
- }
258
- /**
259
- * Method to check the status the p2p client.
260
- * @returns Information about p2p client status: state & syncedToBlockNum.
261
- */
262
- getStatus() {
263
- return Promise.resolve({
264
- state: this.currentState,
265
- syncedToL2Block: this.getSyncedLatestBlockNum(),
266
- });
267
- }
268
- /**
269
- * Mark all txs from these blocks as mined.
270
- * @param blocks - A list of existing blocks with txs that the P2P client needs to ensure the tx pool is reconciled with.
271
- * @returns Empty promise.
272
- */
273
- async markTxsAsMinedFromBlocks(blocks) {
274
- for (const block of blocks) {
275
- const txHashes = block.body.txEffects.map(txEffect => txEffect.txHash);
276
- await this.txPool.markAsMined(txHashes);
277
- }
278
- }
279
- /**
280
- * Deletes txs from these blocks.
281
- * @param blocks - A list of existing blocks with txs that the P2P client needs to ensure the tx pool is reconciled with.
282
- * @returns Empty promise.
283
- */
284
- async deleteTxsFromBlocks(blocks) {
285
- this.log.debug(`Deleting txs from blocks ${blocks[0].number} to ${blocks[blocks.length - 1].number}`);
286
- for (const block of blocks) {
287
- const txHashes = block.body.txEffects.map(txEffect => txEffect.txHash);
288
- await this.txPool.deleteTxs(txHashes);
289
- }
290
- }
291
- /**
292
- * Handles new mined blocks by marking the txs in them as mined.
293
- * @param blocks - A list of existing blocks with txs that the P2P client needs to ensure the tx pool is reconciled with.
294
- * @returns Empty promise.
295
- */
296
- async handleLatestL2Blocks(blocks) {
297
- if (!blocks.length) {
298
- return Promise.resolve();
299
- }
300
- await this.markTxsAsMinedFromBlocks(blocks);
301
- const lastBlockNum = blocks[blocks.length - 1].number;
302
- await this.synchedLatestBlockNumber.set(lastBlockNum);
303
- this.log.debug(`Synched to latest block ${lastBlockNum}`);
304
- await this.startServiceIfSynched();
305
- }
306
- /**
307
- * Handles new proven blocks by deleting the txs in them, or by deleting the txs in blocks `keepProvenTxsFor` ago.
308
- * @param blocks - A list of proven L2 blocks.
309
- * @returns Empty promise.
310
- */
311
- async handleProvenL2Blocks(blocks) {
312
- if (!blocks.length) {
313
- return Promise.resolve();
314
- }
315
- const firstBlockNum = blocks[0].number;
316
- const lastBlockNum = blocks[blocks.length - 1].number;
317
- if (this.keepProvenTxsFor === 0) {
318
- await this.deleteTxsFromBlocks(blocks);
319
- }
320
- else if (lastBlockNum - this.keepProvenTxsFor >= INITIAL_L2_BLOCK_NUM) {
321
- const fromBlock = Math.max(INITIAL_L2_BLOCK_NUM, firstBlockNum - this.keepProvenTxsFor);
322
- const toBlock = lastBlockNum - this.keepProvenTxsFor;
323
- const limit = toBlock - fromBlock + 1;
324
- const blocksToDeleteTxsFrom = await this.l2BlockSource.getBlocks(fromBlock, limit, true);
325
- await this.deleteTxsFromBlocks(blocksToDeleteTxsFrom);
326
- }
327
- await this.synchedProvenBlockNumber.set(lastBlockNum);
328
- this.log.debug(`Synched to proven block ${lastBlockNum}`);
329
- const provenEpochNumber = await this.l2BlockSource.getProvenL2EpochNumber();
330
- if (provenEpochNumber !== undefined) {
331
- this.epochProofQuotePool.deleteQuotesToEpoch(BigInt(provenEpochNumber));
332
- }
333
- await this.startServiceIfSynched();
334
- }
335
- async startServiceIfSynched() {
336
- if (this.currentState === P2PClientState.SYNCHING &&
337
- this.getSyncedLatestBlockNum() >= this.latestBlockNumberAtStart &&
338
- this.getSyncedProvenBlockNum() >= this.provenBlockNumberAtStart) {
339
- this.log.debug(`Synched to blocks at start`);
340
- this.setCurrentState(P2PClientState.RUNNING);
341
- if (this.syncResolve !== undefined) {
342
- this.syncResolve();
343
- await this.p2pService.start();
66
+ broadcastEpochProofQuote(quote) {
67
+ __classPrivateFieldGet(this, _P2PClient_instances, "m", _P2PClient_assertIsReady).call(this);
68
+ this.epochProofQuotePool.addQuote(quote);
69
+ return this.p2pService.propagate(quote);
344
70
  }
345
- }
346
- }
347
- /**
348
- * Method to set the value of the current state.
349
- * @param newState - New state value.
350
- */
351
- setCurrentState(newState) {
352
- this.currentState = newState;
353
- this.log.debug(`Moved to state ${P2PClientState[this.currentState]}`);
354
- }
355
- async publishStoredTxs() {
356
- if (!this.isReady()) {
357
- return;
358
- }
359
- const txs = this.txPool.getAllTxs();
360
- if (txs.length > 0) {
361
- this.log.debug(`Publishing ${txs.length} previously stored txs`);
362
- await Promise.all(txs.map(tx => this.p2pService.propagate(tx)));
363
- }
364
- }
365
- }
366
- _P2PClient_instances = new WeakSet(), _P2PClient_assertIsReady = function _P2PClient_assertIsReady() {
367
- if (!this.isReady()) {
368
- throw new Error('P2P client not ready');
369
- }
370
- };
371
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicDJwX2NsaWVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jbGllbnQvcDJwX2NsaWVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUFBLE9BQU8sRUFLTCxpQkFBaUIsR0FJbEIsTUFBTSxzQkFBc0IsQ0FBQztBQUM5QixPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSw4QkFBOEIsQ0FBQztBQUNwRSxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQU0xRCxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFFbkQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBSWxFOztHQUVHO0FBQ0gsTUFBTSxDQUFOLElBQVksY0FLWDtBQUxELFdBQVksY0FBYztJQUN4QixtREFBSSxDQUFBO0lBQ0osMkRBQVEsQ0FBQTtJQUNSLHlEQUFPLENBQUE7SUFDUCx5REFBTyxDQUFBO0FBQ1QsQ0FBQyxFQUxXLGNBQWMsS0FBZCxjQUFjLFFBS3pCO0FBd0lEOztHQUVHO0FBQ0gsTUFBTSxPQUFPLFNBQVM7SUFzQnBCOzs7Ozs7OztPQVFHO0lBQ0gsWUFDRSxLQUFtQixFQUNYLGFBQTRCLEVBQzVCLE1BQWMsRUFDZCxlQUFnQyxFQUNoQyxtQkFBd0MsRUFDeEMsVUFBc0IsRUFDdEIsZ0JBQXdCLEVBQ3hCLE1BQU0saUJBQWlCLENBQUMsV0FBVyxDQUFDOztRQU5wQyxrQkFBYSxHQUFiLGFBQWEsQ0FBZTtRQUM1QixXQUFNLEdBQU4sTUFBTSxDQUFRO1FBQ2Qsb0JBQWUsR0FBZixlQUFlLENBQWlCO1FBQ2hDLHdCQUFtQixHQUFuQixtQkFBbUIsQ0FBcUI7UUFDeEMsZUFBVSxHQUFWLFVBQVUsQ0FBWTtRQUN0QixxQkFBZ0IsR0FBaEIsZ0JBQWdCLENBQVE7UUFDeEIsUUFBRyxHQUFILEdBQUcsQ0FBaUM7UUFoQzlDLDZEQUE2RDtRQUNyRCxhQUFRLEdBQUcsS0FBSyxD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
71
+ /**
72
+ * Starts the P2P client.
73
+ * @returns An empty promise signalling the synching process.
74
+ */
75
+ async start() {
76
+ if (this.currentState === P2PClientState.STOPPED) {
77
+ throw new Error('P2P client already stopped');
78
+ }
79
+ if (this.currentState !== P2PClientState.IDLE) {
80
+ return this.syncPromise;
81
+ }
82
+ // get the current latest block numbers
83
+ this.latestBlockNumberAtStart = await this.l2BlockSource.getBlockNumber();
84
+ this.provenBlockNumberAtStart = await this.l2BlockSource.getProvenBlockNumber();
85
+ const syncedLatestBlock = this.getSyncedLatestBlockNum() + 1;
86
+ const syncedProvenBlock = this.getSyncedProvenBlockNum() + 1;
87
+ // if there are blocks to be retrieved, go to a synching state
88
+ if (syncedLatestBlock <= this.latestBlockNumberAtStart || syncedProvenBlock <= this.provenBlockNumberAtStart) {
89
+ this.setCurrentState(P2PClientState.SYNCHING);
90
+ this.syncPromise = new Promise(resolve => {
91
+ this.syncResolve = resolve;
92
+ });
93
+ this.log.verbose(`Starting sync from ${syncedLatestBlock} (last proven ${syncedProvenBlock})`);
94
+ }
95
+ else {
96
+ // if no blocks to be retrieved, go straight to running
97
+ this.setCurrentState(P2PClientState.RUNNING);
98
+ this.syncPromise = Promise.resolve();
99
+ await this.p2pService.start();
100
+ this.log.verbose(`Block ${syncedLatestBlock} (proven ${syncedProvenBlock}) already beyond current block`);
101
+ }
102
+ // publish any txs in TxPool after its doing initial sync
103
+ this.syncPromise = this.syncPromise.then(() => this.publishStoredTxs());
104
+ // start looking for further blocks
105
+ const processLatest = async () => {
106
+ while (!this.stopping) {
107
+ await this.latestBlockDownloader.getBlocks(1).then(this.handleLatestL2Blocks.bind(this));
108
+ }
109
+ };
110
+ const processProven = async () => {
111
+ while (!this.stopping) {
112
+ await this.provenBlockDownloader.getBlocks(1).then(this.handleProvenL2Blocks.bind(this));
113
+ }
114
+ };
115
+ this.runningPromise = Promise.all([processLatest(), processProven()]).then(() => { });
116
+ this.latestBlockDownloader.start(syncedLatestBlock);
117
+ this.provenBlockDownloader.start(syncedLatestBlock);
118
+ this.log.verbose(`Started block downloader from block ${syncedLatestBlock}`);
119
+ return this.syncPromise;
120
+ }
121
+ /**
122
+ * Allows consumers to stop the instance of the P2P client.
123
+ * 'ready' will now return 'false' and the running promise that keeps the client synced is interrupted.
124
+ */
125
+ async stop() {
126
+ this.log.debug('Stopping p2p client...');
127
+ this.stopping = true;
128
+ await this.p2pService.stop();
129
+ this.log.debug('Stopped p2p service');
130
+ await this.latestBlockDownloader.stop();
131
+ await this.provenBlockDownloader.stop();
132
+ this.log.debug('Stopped block downloader');
133
+ await this.runningPromise;
134
+ this.setCurrentState(P2PClientState.STOPPED);
135
+ this.log.info('P2P client stopped.');
136
+ }
137
+ broadcastProposal(proposal) {
138
+ this.log.verbose(`Broadcasting proposal ${proposal.p2pMessageIdentifier()} to peers`);
139
+ return this.p2pService.propagate(proposal);
140
+ }
141
+ getAttestationsForSlot(slot, proposalId) {
142
+ return Promise.resolve(this.attestationPool.getAttestationsForSlot(slot, proposalId));
143
+ }
144
+ // REVIEW: https://github.com/AztecProtocol/aztec-packages/issues/7963
145
+ // ^ This pattern is not my favorite (md)
146
+ registerBlockProposalHandler(handler) {
147
+ this.p2pService.registerBlockReceivedCallback(handler);
148
+ }
149
+ /**
150
+ * Requests the transactions with the given hashes from the network.
151
+ *
152
+ * If a transaction can be retrieved, it will be returned, if not an undefined
153
+ * will be returned. In place.
154
+ *
155
+ * @param txHashes - The hashes of the transactions to request.
156
+ * @returns A promise that resolves to an array of transactions or undefined.
157
+ */
158
+ requestTxs(txHashes) {
159
+ const requestPromises = txHashes.map(txHash => this.requestTxByHash(txHash));
160
+ return Promise.all(requestPromises);
161
+ }
162
+ /**
163
+ * Uses the Request Response protocol to request a transaction from the network.
164
+ *
165
+ * If the underlying request response protocol fails, then we return undefined.
166
+ * If it succeeds then we add the transaction to our transaction pool and return.
167
+ *
168
+ * @param txHash - The hash of the transaction to request.
169
+ * @returns A promise that resolves to a transaction or undefined.
170
+ */
171
+ async requestTxByHash(txHash) {
172
+ const tx = await this.p2pService.sendRequest(TX_REQ_PROTOCOL, txHash);
173
+ this.log.debug(`Requested ${txHash.toString()} from peer | success = ${!!tx}`);
174
+ if (tx) {
175
+ await this.txPool.addTxs([tx]);
176
+ }
177
+ return tx;
178
+ }
179
+ /**
180
+ * Returns all transactions in the transaction pool.
181
+ * @returns An array of Txs.
182
+ */
183
+ getTxs(filter) {
184
+ if (filter === 'all') {
185
+ return this.txPool.getAllTxs();
186
+ }
187
+ else if (filter === 'mined') {
188
+ return this.txPool
189
+ .getMinedTxHashes()
190
+ .map(txHash => this.txPool.getTxByHash(txHash))
191
+ .filter((tx) => !!tx);
192
+ }
193
+ else if (filter === 'pending') {
194
+ return this.txPool
195
+ .getPendingTxHashes()
196
+ .map(txHash => this.txPool.getTxByHash(txHash))
197
+ .filter((tx) => !!tx);
198
+ }
199
+ else {
200
+ const _ = filter;
201
+ throw new Error(`Unknown filter ${filter}`);
202
+ }
203
+ }
204
+ /**
205
+ * Returns a transaction in the transaction pool by its hash.
206
+ * @param txHash - Hash of the transaction to look for in the pool.
207
+ * @returns A single tx or undefined.
208
+ */
209
+ getTxByHash(txHash) {
210
+ return this.txPool.getTxByHash(txHash);
211
+ }
212
+ /**
213
+ * Verifies the 'tx' and, if valid, adds it to local tx pool and forwards it to other peers.
214
+ * @param tx - The tx to verify.
215
+ * @returns Empty promise.
216
+ **/
217
+ async sendTx(tx) {
218
+ __classPrivateFieldGet(this, _P2PClient_instances, "m", _P2PClient_assertIsReady).call(this);
219
+ await this.txPool.addTxs([tx]);
220
+ this.p2pService.propagate(tx);
221
+ }
222
+ /**
223
+ * Returns whether the given tx hash is flagged as pending or mined.
224
+ * @param txHash - Hash of the tx to query.
225
+ * @returns Pending or mined depending on its status, or undefined if not found.
226
+ */
227
+ getTxStatus(txHash) {
228
+ return this.txPool.getTxStatus(txHash);
229
+ }
230
+ getEnr() {
231
+ return this.p2pService.getEnr();
232
+ }
233
+ /**
234
+ * Deletes the 'txs' from the pool.
235
+ * NOT used if we use sendTx as reconcileTxPool will handle this.
236
+ * @param txHashes - Hashes of the transactions to delete.
237
+ * @returns Empty promise.
238
+ **/
239
+ async deleteTxs(txHashes) {
240
+ __classPrivateFieldGet(this, _P2PClient_instances, "m", _P2PClient_assertIsReady).call(this);
241
+ await this.txPool.deleteTxs(txHashes);
242
+ }
243
+ /**
244
+ * Public function to check if the p2p client is fully synced and ready to receive txs.
245
+ * @returns True if the P2P client is ready to receive txs.
246
+ */
247
+ isReady() {
248
+ return this.currentState === P2PClientState.RUNNING;
249
+ }
250
+ /**
251
+ * Public function to check the latest block number that the P2P client is synced to.
252
+ * @returns Block number of latest L2 Block we've synced with.
253
+ */
254
+ getSyncedLatestBlockNum() {
255
+ return this.synchedLatestBlockNumber.get() ?? INITIAL_L2_BLOCK_NUM - 1;
256
+ }
257
+ /**
258
+ * Public function to check the latest proven block number that the P2P client is synced to.
259
+ * @returns Block number of latest proven L2 Block we've synced with.
260
+ */
261
+ getSyncedProvenBlockNum() {
262
+ return this.synchedProvenBlockNumber.get() ?? INITIAL_L2_BLOCK_NUM - 1;
263
+ }
264
+ /**
265
+ * Method to check the status the p2p client.
266
+ * @returns Information about p2p client status: state & syncedToBlockNum.
267
+ */
268
+ async getStatus() {
269
+ const blockNumber = this.getSyncedLatestBlockNum();
270
+ const blockHash = blockNumber == 0
271
+ ? ''
272
+ : await this.l2BlockSource.getBlockHeader(blockNumber).then(header => header?.hash().toString());
273
+ return Promise.resolve({
274
+ state: this.currentState,
275
+ syncedToL2Block: { number: blockNumber, hash: blockHash },
276
+ });
277
+ }
278
+ /**
279
+ * Mark all txs from these blocks as mined.
280
+ * @param blocks - A list of existing blocks with txs that the P2P client needs to ensure the tx pool is reconciled with.
281
+ * @returns Empty promise.
282
+ */
283
+ async markTxsAsMinedFromBlocks(blocks) {
284
+ for (const block of blocks) {
285
+ const txHashes = block.body.txEffects.map(txEffect => txEffect.txHash);
286
+ await this.txPool.markAsMined(txHashes);
287
+ }
288
+ }
289
+ /**
290
+ * Deletes txs from these blocks.
291
+ * @param blocks - A list of existing blocks with txs that the P2P client needs to ensure the tx pool is reconciled with.
292
+ * @returns Empty promise.
293
+ */
294
+ async deleteTxsFromBlocks(blocks) {
295
+ this.log.debug(`Deleting txs from blocks ${blocks[0].number} to ${blocks[blocks.length - 1].number}`);
296
+ for (const block of blocks) {
297
+ const txHashes = block.body.txEffects.map(txEffect => txEffect.txHash);
298
+ await this.txPool.deleteTxs(txHashes);
299
+ }
300
+ }
301
+ /**
302
+ * Handles new mined blocks by marking the txs in them as mined.
303
+ * @param blocks - A list of existing blocks with txs that the P2P client needs to ensure the tx pool is reconciled with.
304
+ * @returns Empty promise.
305
+ */
306
+ async handleLatestL2Blocks(blocks) {
307
+ if (!blocks.length) {
308
+ return Promise.resolve();
309
+ }
310
+ await this.markTxsAsMinedFromBlocks(blocks);
311
+ const lastBlockNum = blocks[blocks.length - 1].number;
312
+ await this.synchedLatestBlockNumber.set(lastBlockNum);
313
+ this.log.debug(`Synched to latest block ${lastBlockNum}`);
314
+ await this.startServiceIfSynched();
315
+ }
316
+ /**
317
+ * Handles new proven blocks by deleting the txs in them, or by deleting the txs in blocks `keepProvenTxsFor` ago.
318
+ * @param blocks - A list of proven L2 blocks.
319
+ * @returns Empty promise.
320
+ */
321
+ async handleProvenL2Blocks(blocks) {
322
+ if (!blocks.length) {
323
+ return Promise.resolve();
324
+ }
325
+ const firstBlockNum = blocks[0].number;
326
+ const lastBlockNum = blocks[blocks.length - 1].number;
327
+ if (this.keepProvenTxsFor === 0) {
328
+ await this.deleteTxsFromBlocks(blocks);
329
+ }
330
+ else if (lastBlockNum - this.keepProvenTxsFor >= INITIAL_L2_BLOCK_NUM) {
331
+ const fromBlock = Math.max(INITIAL_L2_BLOCK_NUM, firstBlockNum - this.keepProvenTxsFor);
332
+ const toBlock = lastBlockNum - this.keepProvenTxsFor;
333
+ const limit = toBlock - fromBlock + 1;
334
+ const blocksToDeleteTxsFrom = await this.l2BlockSource.getBlocks(fromBlock, limit, true);
335
+ await this.deleteTxsFromBlocks(blocksToDeleteTxsFrom);
336
+ }
337
+ await this.synchedProvenBlockNumber.set(lastBlockNum);
338
+ this.log.debug(`Synched to proven block ${lastBlockNum}`);
339
+ const provenEpochNumber = await this.l2BlockSource.getProvenL2EpochNumber();
340
+ if (provenEpochNumber !== undefined) {
341
+ this.epochProofQuotePool.deleteQuotesToEpoch(BigInt(provenEpochNumber));
342
+ }
343
+ await this.startServiceIfSynched();
344
+ }
345
+ async startServiceIfSynched() {
346
+ if (this.currentState === P2PClientState.SYNCHING &&
347
+ this.getSyncedLatestBlockNum() >= this.latestBlockNumberAtStart &&
348
+ this.getSyncedProvenBlockNum() >= this.provenBlockNumberAtStart) {
349
+ this.log.debug(`Synched to blocks at start`);
350
+ this.setCurrentState(P2PClientState.RUNNING);
351
+ if (this.syncResolve !== undefined) {
352
+ this.syncResolve();
353
+ await this.p2pService.start();
354
+ }
355
+ }
356
+ }
357
+ /**
358
+ * Method to set the value of the current state.
359
+ * @param newState - New state value.
360
+ */
361
+ setCurrentState(newState) {
362
+ this.currentState = newState;
363
+ this.log.debug(`Moved to state ${P2PClientState[this.currentState]}`);
364
+ }
365
+ async publishStoredTxs() {
366
+ if (!this.isReady()) {
367
+ return;
368
+ }
369
+ const txs = this.txPool.getAllTxs();
370
+ if (txs.length > 0) {
371
+ this.log.debug(`Publishing ${txs.length} previously stored txs`);
372
+ await Promise.all(txs.map(tx => this.p2pService.propagate(tx)));
373
+ }
374
+ }
375
+ },
376
+ _P2PClient_instances = new WeakSet(),
377
+ _P2PClient_assertIsReady = function _P2PClient_assertIsReady() {
378
+ if (!this.isReady()) {
379
+ throw new Error('P2P client not ready');
380
+ }
381
+ },
382
+ (() => {
383
+ const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
384
+ _broadcastProposal_decorators = [trackSpan('p2pClient.broadcastProposal', proposal => ({
385
+ [Attributes.BLOCK_NUMBER]: proposal.payload.header.globalVariables.blockNumber.toNumber(),
386
+ [Attributes.SLOT_NUMBER]: proposal.payload.header.globalVariables.slotNumber.toNumber(),
387
+ [Attributes.BLOCK_ARCHIVE]: proposal.archive.toString(),
388
+ [Attributes.P2P_ID]: proposal.p2pMessageIdentifier().toString(),
389
+ }))];
390
+ __esDecorate(_a, null, _broadcastProposal_decorators, { kind: "method", name: "broadcastProposal", static: false, private: false, access: { has: obj => "broadcastProposal" in obj, get: obj => obj.broadcastProposal }, metadata: _metadata }, null, _instanceExtraInitializers);
391
+ if (_metadata) Object.defineProperty(_a, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
392
+ })(),
393
+ _a;
394
+ })();
395
+ export { P2PClient };
396
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicDJwX2NsaWVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9jbGllbnQvcDJwX2NsaWVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsT0FBTyxFQUtMLGlCQUFpQixHQUtsQixNQUFNLHNCQUFzQixDQUFDO0FBQzlCLE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLDhCQUE4QixDQUFDO0FBQ3BFLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBRTFELE9BQU8sRUFBRSxVQUFVLEVBQXdCLFVBQVUsRUFBRSxTQUFTLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUlsRyxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFLbkQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBR2xFOztHQUVHO0FBQ0gsTUFBTSxDQUFOLElBQVksY0FLWDtBQUxELFdBQVksY0FBYztJQUN4QixtREFBSSxDQUFBO0lBQ0osMkRBQVEsQ0FBQTtJQUNSLHlEQUFPLENBQUE7SUFDUCx5REFBTyxDQUFBO0FBQ1QsQ0FBQyxFQUxXLGNBQWMsS0FBZCxjQUFjLFFBS3pCO0FBd0lEOztHQUVHO0lBQ1UsU0FBUzs7c0JBQVMsVUFBVTs7O3NCQUE1QixTQUFVLFNBQVEsV0FBVTtZQTBCdkM7Ozs7Ozs7O2VBUUc7WUFDSCxZQUNFLEtBQW1CLEVBQ1gsYUFBNEIsRUFDcEMsUUFBa0IsRUFDVixVQUFzQixFQUN0QixnQkFBd0IsRUFDaEMsZUFBZ0MsRUFDeEIsTUFBTSxpQkFBaUIsQ0FBQyxXQUFXLENBQUM7Z0JBRTVDLEtBQUssQ0FBQyxlQUFlLEVBQUUsV0FBVyxDQUFDLENBQUM7O2dCQVA1QixrQkFBYSxJQXJDWixtREFBUyxFQXFDVixhQUFhLEVBQWU7Z0JBRTVCLGVBQVUsR0FBVixVQUFVLENBQVk7Z0JBQ3RCLHFCQUFnQixHQUFoQixnQkFBZ0IsQ0FBUTtnQkFFeEIsUUFBRyxH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