@aztec/prover-node 0.0.1-commit.9b94fc1 → 0.0.1-commit.9ee6fcc6

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 (51) hide show
  1. package/dest/actions/download-epoch-proving-job.d.ts +1 -1
  2. package/dest/actions/rerun-epoch-proving-job.d.ts +4 -3
  3. package/dest/actions/rerun-epoch-proving-job.d.ts.map +1 -1
  4. package/dest/actions/rerun-epoch-proving-job.js +2 -2
  5. package/dest/actions/upload-epoch-proof-failure.d.ts +2 -2
  6. package/dest/actions/upload-epoch-proof-failure.d.ts.map +1 -1
  7. package/dest/bin/run-failed-epoch.js +5 -2
  8. package/dest/config.d.ts +8 -10
  9. package/dest/config.d.ts.map +1 -1
  10. package/dest/config.js +19 -21
  11. package/dest/factory.d.ts +20 -16
  12. package/dest/factory.d.ts.map +1 -1
  13. package/dest/factory.js +47 -62
  14. package/dest/index.d.ts +2 -1
  15. package/dest/index.d.ts.map +1 -1
  16. package/dest/index.js +1 -0
  17. package/dest/job/epoch-proving-job-data.d.ts +7 -6
  18. package/dest/job/epoch-proving-job-data.d.ts.map +1 -1
  19. package/dest/job/epoch-proving-job-data.js +24 -18
  20. package/dest/job/epoch-proving-job.d.ts +5 -4
  21. package/dest/job/epoch-proving-job.d.ts.map +1 -1
  22. package/dest/job/epoch-proving-job.js +511 -108
  23. package/dest/metrics.d.ts +14 -3
  24. package/dest/metrics.d.ts.map +1 -1
  25. package/dest/metrics.js +54 -97
  26. package/dest/monitors/epoch-monitor.d.ts +1 -1
  27. package/dest/monitors/epoch-monitor.d.ts.map +1 -1
  28. package/dest/monitors/epoch-monitor.js +8 -18
  29. package/dest/prover-node-publisher.d.ts +12 -10
  30. package/dest/prover-node-publisher.d.ts.map +1 -1
  31. package/dest/prover-node-publisher.js +46 -40
  32. package/dest/prover-node.d.ts +22 -12
  33. package/dest/prover-node.d.ts.map +1 -1
  34. package/dest/prover-node.js +442 -55
  35. package/dest/prover-publisher-factory.d.ts +10 -6
  36. package/dest/prover-publisher-factory.d.ts.map +1 -1
  37. package/dest/prover-publisher-factory.js +7 -5
  38. package/package.json +26 -25
  39. package/src/actions/rerun-epoch-proving-job.ts +5 -3
  40. package/src/actions/upload-epoch-proof-failure.ts +1 -1
  41. package/src/bin/run-failed-epoch.ts +5 -2
  42. package/src/config.ts +27 -33
  43. package/src/factory.ts +80 -104
  44. package/src/index.ts +1 -0
  45. package/src/job/epoch-proving-job-data.ts +28 -23
  46. package/src/job/epoch-proving-job.ts +151 -111
  47. package/src/metrics.ts +64 -81
  48. package/src/monitors/epoch-monitor.ts +6 -14
  49. package/src/prover-node-publisher.ts +67 -55
  50. package/src/prover-node.ts +54 -42
  51. package/src/prover-publisher-factory.ts +19 -11
@@ -1,29 +1,30 @@
1
1
  import type { Archiver } from '@aztec/archiver';
2
- import type { RollupContract } from '@aztec/ethereum';
3
- import { EpochNumber } from '@aztec/foundation/branded-types';
2
+ import type { RollupContract } from '@aztec/ethereum/contracts';
3
+ import type { Delayer } from '@aztec/ethereum/l1-tx-utils';
4
+ import { BlockNumber, CheckpointNumber, EpochNumber } from '@aztec/foundation/branded-types';
4
5
  import { assertRequired, compact, pick, sum } from '@aztec/foundation/collection';
6
+ import type { Fr } from '@aztec/foundation/curves/bn254';
5
7
  import { memoize } from '@aztec/foundation/decorators';
6
- import type { Fr } from '@aztec/foundation/fields';
7
8
  import { createLogger } from '@aztec/foundation/log';
8
9
  import { DateProvider } from '@aztec/foundation/timer';
9
- import type { DataStoreConfig } from '@aztec/kv-store/config';
10
- import type { P2PClient } from '@aztec/p2p';
11
10
  import { PublicProcessorFactory } from '@aztec/simulator/server';
12
- import type { L2Block, L2BlockSource } from '@aztec/stdlib/block';
11
+ import type { L2BlockSource } from '@aztec/stdlib/block';
12
+ import type { Checkpoint } from '@aztec/stdlib/checkpoint';
13
13
  import type { ChainConfig } from '@aztec/stdlib/config';
14
14
  import type { ContractDataSource } from '@aztec/stdlib/contract';
15
15
  import { getProofSubmissionDeadlineTimestamp } from '@aztec/stdlib/epoch-helpers';
16
16
  import {
17
17
  type EpochProverManager,
18
18
  EpochProvingJobTerminalState,
19
+ type ITxProvider,
19
20
  type ProverNodeApi,
20
21
  type Service,
21
22
  type WorldStateSyncStatus,
22
23
  type WorldStateSynchronizer,
23
24
  tryStop,
24
25
  } from '@aztec/stdlib/interfaces/server';
26
+ import type { DataStoreConfig } from '@aztec/stdlib/kv-store';
25
27
  import type { L1ToL2MessageSource } from '@aztec/stdlib/messaging';
26
- import type { P2PClientType } from '@aztec/stdlib/p2p';
27
28
  import type { Tx } from '@aztec/stdlib/tx';
28
29
  import {
29
30
  Attributes,
@@ -54,7 +55,6 @@ type DataStoreOptions = Pick<DataStoreConfig, 'dataDirectory'> & Pick<ChainConfi
54
55
  */
55
56
  export class ProverNode implements EpochMonitorHandler, ProverNodeApi, Traceable {
56
57
  private log = createLogger('prover-node');
57
- private dateProvider = new DateProvider();
58
58
 
59
59
  private jobs: Map<string, EpochProvingJob> = new Map();
60
60
  private config: ProverNodeOptions;
@@ -72,17 +72,19 @@ export class ProverNode implements EpochMonitorHandler, ProverNodeApi, Traceable
72
72
  protected readonly l1ToL2MessageSource: L1ToL2MessageSource,
73
73
  protected readonly contractDataSource: ContractDataSource,
74
74
  protected readonly worldState: WorldStateSynchronizer,
75
- protected readonly p2pClient: Pick<P2PClient<P2PClientType.Prover>, 'getTxProvider'> & Partial<Service>,
75
+ protected readonly p2pClient: { getTxProvider(): ITxProvider } & Partial<Service>,
76
76
  protected readonly epochsMonitor: EpochMonitor,
77
77
  protected readonly rollupContract: RollupContract,
78
78
  protected readonly l1Metrics: L1Metrics,
79
79
  config: Partial<ProverNodeOptions> = {},
80
80
  protected readonly telemetryClient: TelemetryClient = getTelemetryClient(),
81
+ private delayer?: Delayer,
82
+ private readonly dateProvider: DateProvider = new DateProvider(),
81
83
  ) {
82
84
  this.config = {
83
85
  proverNodePollingIntervalMs: 1_000,
84
86
  proverNodeMaxPendingJobs: 100,
85
- proverNodeMaxParallelBlocksPerEpoch: 32,
87
+ proverNodeMaxParallelBlocksPerEpoch: 0,
86
88
  txGatheringIntervalMs: 1_000,
87
89
  txGatheringBatchSize: 10,
88
90
  txGatheringMaxParallelRequestsPerNode: 100,
@@ -110,6 +112,11 @@ export class ProverNode implements EpochMonitorHandler, ProverNodeApi, Traceable
110
112
  return this.p2pClient;
111
113
  }
112
114
 
115
+ /** Returns the shared tx delayer for prover L1 txs, if enabled. Test-only. */
116
+ public getDelayer(): Delayer | undefined {
117
+ return this.delayer;
118
+ }
119
+
113
120
  /**
114
121
  * Handles an epoch being completed by starting a proof for it if there are no active jobs for it.
115
122
  * @param epochNumber - The epoch number that was just completed.
@@ -154,17 +161,15 @@ export class ProverNode implements EpochMonitorHandler, ProverNodeApi, Traceable
154
161
 
155
162
  /**
156
163
  * Stops the prover node and all its dependencies.
164
+ * Resources not owned by this node (shared with the parent aztec-node) are skipped.
157
165
  */
158
166
  async stop() {
159
167
  this.log.info('Stopping ProverNode');
160
168
  await this.epochsMonitor.stop();
161
169
  await this.prover.stop();
162
- await tryStop(this.p2pClient);
163
- await tryStop(this.l2BlockSource);
164
170
  await tryStop(this.publisherFactory);
165
171
  this.publisher?.interrupt();
166
172
  await Promise.all(Array.from(this.jobs.values()).map(job => job.stop()));
167
- await this.worldState.stop();
168
173
  this.rewardsMetrics.stop();
169
174
  this.l1Metrics.stop();
170
175
  await this.telemetryClient.stop();
@@ -271,25 +276,30 @@ export class ProverNode implements EpochMonitorHandler, ProverNodeApi, Traceable
271
276
 
272
277
  // Gather all data for this epoch
273
278
  const epochData = await this.gatherEpochData(epochNumber);
274
-
275
- const fromBlock = epochData.blocks[0].number;
276
- const toBlock = epochData.blocks.at(-1)!.number;
277
- this.log.verbose(`Creating proving job for epoch ${epochNumber} for block range ${fromBlock} to ${toBlock}`);
279
+ const fromCheckpoint = epochData.checkpoints[0].number;
280
+ const toCheckpoint = epochData.checkpoints.at(-1)!.number;
281
+ const fromBlock = epochData.checkpoints[0].blocks[0].number;
282
+ const lastBlock = epochData.checkpoints.at(-1)!.blocks.at(-1)!;
283
+ const toBlock = lastBlock.number;
284
+ this.log.verbose(
285
+ `Creating proving job for epoch ${epochNumber} for checkpoint range ${fromCheckpoint} to ${toCheckpoint} and block range ${fromBlock} to ${toBlock}`,
286
+ );
278
287
 
279
288
  // Fast forward world state to right before the target block and get a fork
280
- await this.worldState.syncImmediate(toBlock);
289
+ const lastBlockHash = await lastBlock.header.hash();
290
+ await this.worldState.syncImmediate(toBlock, lastBlockHash);
281
291
 
282
292
  // Create a processor factory
283
293
  const publicProcessorFactory = new PublicProcessorFactory(
284
294
  this.contractDataSource,
285
295
  this.dateProvider,
286
296
  this.telemetryClient,
297
+ this.log.getBindings(),
287
298
  );
288
299
 
289
300
  // Set deadline for this job to run. It will abort if it takes too long.
290
301
  const deadlineTs = getProofSubmissionDeadlineTimestamp(epochNumber, await this.getL1Constants());
291
302
  const deadline = new Date(Number(deadlineTs) * 1000);
292
-
293
303
  const job = this.doCreateEpochProvingJob(epochData, deadline, publicProcessorFactory, this.publisher, opts);
294
304
  this.jobs.set(job.getId(), job);
295
305
  return job;
@@ -302,28 +312,30 @@ export class ProverNode implements EpochMonitorHandler, ProverNodeApi, Traceable
302
312
 
303
313
  @trackSpan('ProverNode.gatherEpochData', epochNumber => ({ [Attributes.EPOCH_NUMBER]: epochNumber }))
304
314
  private async gatherEpochData(epochNumber: EpochNumber): Promise<EpochProvingJobData> {
305
- const blocks = await this.gatherBlocks(epochNumber);
306
- const txArray = await this.gatherTxs(epochNumber, blocks);
315
+ const checkpoints = await this.gatherCheckpoints(epochNumber);
316
+ const txArray = await this.gatherTxs(epochNumber, checkpoints);
307
317
  const txs = new Map<string, Tx>(txArray.map(tx => [tx.getTxHash().toString(), tx]));
308
- const l1ToL2Messages = await this.gatherMessages(epochNumber, blocks);
309
- const previousBlockHeader = await this.gatherPreviousBlockHeader(epochNumber, blocks[0]);
310
- const [lastBlock] = await this.l2BlockSource.getPublishedBlocks(blocks.at(-1)!.number, 1);
311
- const attestations = lastBlock?.attestations ?? [];
318
+ const l1ToL2Messages = await this.gatherMessages(epochNumber, checkpoints);
319
+ const [firstBlock] = checkpoints[0].blocks;
320
+ const previousBlockHeader = await this.gatherPreviousBlockHeader(epochNumber, firstBlock.number - 1);
321
+ const [lastPublishedCheckpoint] = await this.l2BlockSource.getCheckpoints(checkpoints.at(-1)!.number, 1);
322
+ const attestations = lastPublishedCheckpoint?.attestations ?? [];
312
323
 
313
- return { blocks, txs, l1ToL2Messages, epochNumber, previousBlockHeader, attestations };
324
+ return { checkpoints, txs, l1ToL2Messages, epochNumber, previousBlockHeader, attestations };
314
325
  }
315
326
 
316
- private async gatherBlocks(epochNumber: EpochNumber) {
317
- const blocks = await this.l2BlockSource.getBlocksForEpoch(epochNumber);
318
- if (blocks.length === 0) {
327
+ private async gatherCheckpoints(epochNumber: EpochNumber) {
328
+ const checkpoints = await this.l2BlockSource.getCheckpointsForEpoch(epochNumber);
329
+ if (checkpoints.length === 0) {
319
330
  throw new EmptyEpochError(epochNumber);
320
331
  }
321
- return blocks;
332
+ return checkpoints;
322
333
  }
323
334
 
324
- private async gatherTxs(epochNumber: EpochNumber, blocks: L2Block[]) {
335
+ private async gatherTxs(epochNumber: EpochNumber, checkpoints: Checkpoint[]) {
325
336
  const deadline = new Date(this.dateProvider.now() + this.config.txGatheringTimeoutMs);
326
337
  const txProvider = this.p2pClient.getTxProvider();
338
+ const blocks = checkpoints.flatMap(checkpoint => checkpoint.blocks);
327
339
  const txsByBlock = await Promise.all(blocks.map(block => txProvider.getTxsForBlock(block, { deadline })));
328
340
  const txs = txsByBlock.map(({ txs }) => txs).flat();
329
341
  const missingTxs = txsByBlock.map(({ missingTxs }) => missingTxs).flat();
@@ -336,25 +348,24 @@ export class ProverNode implements EpochMonitorHandler, ProverNodeApi, Traceable
336
348
  throw new Error(`Txs not found for epoch ${epochNumber}: ${missingTxs.map(hash => hash.toString()).join(', ')}`);
337
349
  }
338
350
 
339
- private async gatherMessages(epochNumber: EpochNumber, blocks: L2Block[]) {
340
- const messages = await Promise.all(blocks.map(b => this.l1ToL2MessageSource.getL1ToL2Messages(b.number)));
351
+ private async gatherMessages(epochNumber: EpochNumber, checkpoints: Checkpoint[]) {
352
+ const messages = await Promise.all(checkpoints.map(c => this.l1ToL2MessageSource.getL1ToL2Messages(c.number)));
341
353
  const messageCount = sum(messages.map(m => m.length));
342
354
  this.log.verbose(`Gathered all ${messageCount} messages for epoch ${epochNumber}`, { epochNumber });
343
- const messagesByBlock: Record<number, Fr[]> = {};
344
- for (let i = 0; i < blocks.length; i++) {
345
- messagesByBlock[blocks[i].number] = messages[i];
355
+ const messagesByCheckpoint: Record<CheckpointNumber, Fr[]> = {};
356
+ for (let i = 0; i < checkpoints.length; i++) {
357
+ messagesByCheckpoint[checkpoints[i].number] = messages[i];
346
358
  }
347
- return messagesByBlock;
359
+ return messagesByCheckpoint;
348
360
  }
349
361
 
350
- private async gatherPreviousBlockHeader(epochNumber: EpochNumber, initialBlock: L2Block) {
351
- const previousBlockNumber = initialBlock.number - 1;
362
+ private async gatherPreviousBlockHeader(epochNumber: EpochNumber, previousBlockNumber: number) {
352
363
  const header = await (previousBlockNumber === 0
353
364
  ? this.worldState.getCommitted().getInitialHeader()
354
- : this.l2BlockSource.getBlockHeader(previousBlockNumber));
365
+ : this.l2BlockSource.getBlockHeader(BlockNumber(previousBlockNumber)));
355
366
 
356
367
  if (!header) {
357
- throw new Error(`Previous block header ${initialBlock.number} not found for proving epoch ${epochNumber}`);
368
+ throw new Error(`Previous block header ${previousBlockNumber} not found for proving epoch ${epochNumber}`);
358
369
  }
359
370
 
360
371
  this.log.verbose(`Gathered previous block header ${header.getBlockNumber()} for epoch ${epochNumber}`);
@@ -380,6 +391,7 @@ export class ProverNode implements EpochMonitorHandler, ProverNodeApi, Traceable
380
391
  this.jobMetrics,
381
392
  deadline,
382
393
  { parallelBlockLimit, skipSubmitProof: proverNodeDisableProofPublish, ...opts },
394
+ this.log.getBindings(),
383
395
  );
384
396
  }
385
397
 
@@ -391,7 +403,7 @@ export class ProverNode implements EpochMonitorHandler, ProverNodeApi, Traceable
391
403
  private validateConfig() {
392
404
  if (
393
405
  this.config.proverNodeFailedEpochStore &&
394
- (!this.config.dataDirectory || !this.config.l1ChainId || !this.config.rollupVersion)
406
+ (!this.config.dataDirectory || !this.config.l1ChainId || this.config.rollupVersion === undefined)
395
407
  ) {
396
408
  this.log.warn(
397
409
  `Invalid prover-node config (missing dataDirectory, l1ChainId, or rollupVersion)`,
@@ -1,25 +1,29 @@
1
- import type { L1TxUtils, PublisherManager, RollupContract } from '@aztec/ethereum';
2
- import type { PublisherConfig, TxSenderConfig } from '@aztec/sequencer-client';
1
+ import type { RollupContract } from '@aztec/ethereum/contracts';
2
+ import type { L1TxUtils } from '@aztec/ethereum/l1-tx-utils';
3
+ import type { PublisherManager } from '@aztec/ethereum/publisher-manager';
4
+ import type { LoggerBindings } from '@aztec/foundation/log';
5
+ import type { ProverPublisherConfig, ProverTxSenderConfig } from '@aztec/sequencer-client';
3
6
  import type { TelemetryClient } from '@aztec/telemetry-client';
4
7
 
5
8
  import { ProverNodePublisher } from './prover-node-publisher.js';
6
9
 
7
10
  export class ProverPublisherFactory {
8
11
  constructor(
9
- private config: TxSenderConfig & PublisherConfig,
12
+ private config: ProverTxSenderConfig & ProverPublisherConfig,
10
13
  private deps: {
11
14
  rollupContract: RollupContract;
12
15
  publisherManager: PublisherManager<L1TxUtils>;
13
16
  telemetry?: TelemetryClient;
14
17
  },
18
+ private bindings?: LoggerBindings,
15
19
  ) {}
16
20
 
17
21
  public async start() {
18
- await this.deps.publisherManager.loadState();
22
+ await this.deps.publisherManager.start();
19
23
  }
20
24
 
21
- public stop() {
22
- this.deps.publisherManager.interrupt();
25
+ public async stop() {
26
+ await this.deps.publisherManager.stop();
23
27
  }
24
28
 
25
29
  /**
@@ -28,10 +32,14 @@ export class ProverPublisherFactory {
28
32
  */
29
33
  public async create(): Promise<ProverNodePublisher> {
30
34
  const l1Publisher = await this.deps.publisherManager.getAvailablePublisher();
31
- return new ProverNodePublisher(this.config, {
32
- rollupContract: this.deps.rollupContract,
33
- l1TxUtils: l1Publisher,
34
- telemetry: this.deps.telemetry,
35
- });
35
+ return new ProverNodePublisher(
36
+ this.config,
37
+ {
38
+ rollupContract: this.deps.rollupContract,
39
+ l1TxUtils: l1Publisher,
40
+ telemetry: this.deps.telemetry,
41
+ },
42
+ this.bindings,
43
+ );
36
44
  }
37
45
  }