@aztec/sequencer-client 0.0.1-commit.f504929 → 0.0.1-commit.f5d02921e
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dest/client/sequencer-client.d.ts +4 -1
- package/dest/client/sequencer-client.d.ts.map +1 -1
- package/dest/client/sequencer-client.js +46 -23
- package/dest/config.d.ts +25 -5
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +21 -12
- package/dest/global_variable_builder/global_builder.d.ts +15 -9
- package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
- package/dest/global_variable_builder/global_builder.js +29 -25
- package/dest/global_variable_builder/index.d.ts +2 -2
- package/dest/global_variable_builder/index.d.ts.map +1 -1
- package/dest/publisher/config.d.ts +13 -1
- package/dest/publisher/config.d.ts.map +1 -1
- package/dest/publisher/config.js +17 -2
- package/dest/publisher/sequencer-publisher-factory.d.ts +3 -3
- package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher-factory.js +2 -2
- package/dest/publisher/sequencer-publisher.d.ts +52 -25
- package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher.js +98 -42
- package/dest/sequencer/checkpoint_proposal_job.d.ts +33 -8
- package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -1
- package/dest/sequencer/checkpoint_proposal_job.js +284 -158
- package/dest/sequencer/checkpoint_voter.d.ts +1 -2
- package/dest/sequencer/checkpoint_voter.d.ts.map +1 -1
- package/dest/sequencer/checkpoint_voter.js +2 -5
- package/dest/sequencer/events.d.ts +2 -1
- package/dest/sequencer/events.d.ts.map +1 -1
- package/dest/sequencer/metrics.d.ts +5 -1
- package/dest/sequencer/metrics.d.ts.map +1 -1
- package/dest/sequencer/metrics.js +11 -0
- package/dest/sequencer/sequencer.d.ts +23 -10
- package/dest/sequencer/sequencer.d.ts.map +1 -1
- package/dest/sequencer/sequencer.js +123 -68
- package/dest/sequencer/timetable.d.ts +4 -3
- package/dest/sequencer/timetable.d.ts.map +1 -1
- package/dest/sequencer/timetable.js +6 -7
- package/dest/sequencer/types.d.ts +2 -2
- package/dest/sequencer/types.d.ts.map +1 -1
- package/dest/test/mock_checkpoint_builder.d.ts +7 -9
- package/dest/test/mock_checkpoint_builder.d.ts.map +1 -1
- package/dest/test/mock_checkpoint_builder.js +39 -30
- package/package.json +27 -28
- package/src/client/sequencer-client.ts +56 -21
- package/src/config.ts +28 -14
- package/src/global_variable_builder/global_builder.ts +37 -26
- package/src/global_variable_builder/index.ts +1 -1
- package/src/publisher/config.ts +32 -0
- package/src/publisher/sequencer-publisher-factory.ts +3 -3
- package/src/publisher/sequencer-publisher.ts +144 -54
- package/src/sequencer/checkpoint_proposal_job.ts +367 -175
- package/src/sequencer/checkpoint_voter.ts +1 -12
- package/src/sequencer/events.ts +1 -1
- package/src/sequencer/metrics.ts +14 -0
- package/src/sequencer/sequencer.ts +178 -79
- package/src/sequencer/timetable.ts +7 -7
- package/src/sequencer/types.ts +1 -1
- package/src/test/mock_checkpoint_builder.ts +51 -48
|
@@ -436,20 +436,21 @@ function _apply_decs_2203_r(targetClass, memberDecs, classDecs, parentClass) {
|
|
|
436
436
|
return (_apply_decs_2203_r = applyDecs2203RFactory())(targetClass, memberDecs, classDecs, parentClass);
|
|
437
437
|
}
|
|
438
438
|
var _dec, _dec1, _dec2, _dec3, _dec4, _dec5, _dec6, _dec7, _initProto;
|
|
439
|
-
import {
|
|
440
|
-
import {
|
|
441
|
-
import { BlockNumber, IndexWithinCheckpoint } from '@aztec/foundation/branded-types';
|
|
439
|
+
import { RollupContract } from '@aztec/ethereum/contracts';
|
|
440
|
+
import { BlockNumber, CheckpointNumber, IndexWithinCheckpoint } from '@aztec/foundation/branded-types';
|
|
442
441
|
import { randomInt } from '@aztec/foundation/crypto/random';
|
|
443
442
|
import { flipSignature, generateRecoverableSignature, generateUnrecoverableSignature } from '@aztec/foundation/crypto/secp256k1-signer';
|
|
443
|
+
import { Signature } from '@aztec/foundation/eth-signature';
|
|
444
444
|
import { filter } from '@aztec/foundation/iterator';
|
|
445
445
|
import { createLogger } from '@aztec/foundation/log';
|
|
446
446
|
import { sleep, sleepUntil } from '@aztec/foundation/sleep';
|
|
447
447
|
import { Timer } from '@aztec/foundation/timer';
|
|
448
448
|
import { isErrorClass, unfreeze } from '@aztec/foundation/types';
|
|
449
449
|
import { CommitteeAttestationsAndSigners, MaliciousCommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
|
|
450
|
-
import {
|
|
450
|
+
import { validateCheckpoint } from '@aztec/stdlib/checkpoint';
|
|
451
|
+
import { computeQuorum, getSlotStartBuildTimestamp, getTimestampForSlot } from '@aztec/stdlib/epoch-helpers';
|
|
451
452
|
import { Gas } from '@aztec/stdlib/gas';
|
|
452
|
-
import {
|
|
453
|
+
import { InsufficientValidTxsError } from '@aztec/stdlib/interfaces/server';
|
|
453
454
|
import { computeInHashFromL1ToL2Messages } from '@aztec/stdlib/messaging';
|
|
454
455
|
import { orderAttestations, trimAttestations } from '@aztec/stdlib/p2p';
|
|
455
456
|
import { AttestationTimeoutError } from '@aztec/stdlib/validators';
|
|
@@ -463,7 +464,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
463
464
|
return {
|
|
464
465
|
// nullish operator needed for tests
|
|
465
466
|
[Attributes.COINBASE]: this.validatorClient.getCoinbaseForAttestor(this.attestorAddress)?.toString(),
|
|
466
|
-
[Attributes.SLOT_NUMBER]: this.
|
|
467
|
+
[Attributes.SLOT_NUMBER]: this.targetSlot
|
|
467
468
|
};
|
|
468
469
|
}), _dec2 = trackSpan('CheckpointProposalJob.buildBlocksForCheckpoint'), _dec3 = trackSpan('CheckpointProposalJob.waitUntilNextSubslot'), _dec4 = trackSpan('CheckpointProposalJob.buildSingleBlock'), _dec5 = trackSpan('CheckpointProposalJob.waitForMinTxs'), _dec6 = trackSpan('CheckpointProposalJob.waitForAttestations'), _dec7 = trackSpan('CheckpointProposalJob.waitUntilTimeInSlot');
|
|
469
470
|
/**
|
|
@@ -472,8 +473,9 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
472
473
|
* as well as enqueueing votes for slashing and governance proposals. This class is created from
|
|
473
474
|
* the Sequencer once the check for being the proposer for the slot has succeeded.
|
|
474
475
|
*/ export class CheckpointProposalJob {
|
|
475
|
-
|
|
476
|
-
|
|
476
|
+
slotNow;
|
|
477
|
+
targetSlot;
|
|
478
|
+
targetEpoch;
|
|
477
479
|
checkpointNumber;
|
|
478
480
|
syncedToBlockNumber;
|
|
479
481
|
proposer;
|
|
@@ -498,6 +500,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
498
500
|
eventEmitter;
|
|
499
501
|
setStateFn;
|
|
500
502
|
tracer;
|
|
503
|
+
proposedCheckpointData;
|
|
501
504
|
static{
|
|
502
505
|
({ e: [_initProto] } = _apply_decs_2203_r(this, [
|
|
503
506
|
[
|
|
@@ -543,10 +546,13 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
543
546
|
], []));
|
|
544
547
|
}
|
|
545
548
|
log;
|
|
546
|
-
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
|
|
549
|
+
/** Tracks the fire-and-forget L1 submission promise so it can be awaited during shutdown. */ pendingL1Submission;
|
|
550
|
+
/** Fee header override computed during proposeCheckpoint, reused in enqueueCheckpointForSubmission. */ computedForceProposedFeeHeader;
|
|
551
|
+
constructor(slotNow, targetSlot, targetEpoch, checkpointNumber, syncedToBlockNumber, // TODO(palla/mbps): Can we remove the proposer in favor of attestorAddress? Need to check fisherman-node flows.
|
|
552
|
+
proposer, publisher, attestorAddress, invalidateCheckpoint, validatorClient, globalsBuilder, p2pClient, worldState, l1ToL2MessageSource, l2BlockSource, checkpointsBuilder, blockSink, l1Constants, config, timetable, slasherClient, epochCache, dateProvider, metrics, eventEmitter, setStateFn, tracer, bindings, proposedCheckpointData){
|
|
553
|
+
this.slotNow = slotNow;
|
|
554
|
+
this.targetSlot = targetSlot;
|
|
555
|
+
this.targetEpoch = targetEpoch;
|
|
550
556
|
this.checkpointNumber = checkpointNumber;
|
|
551
557
|
this.syncedToBlockNumber = syncedToBlockNumber;
|
|
552
558
|
this.proposer = proposer;
|
|
@@ -571,22 +577,31 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
571
577
|
this.eventEmitter = eventEmitter;
|
|
572
578
|
this.setStateFn = setStateFn;
|
|
573
579
|
this.tracer = tracer;
|
|
580
|
+
this.proposedCheckpointData = proposedCheckpointData;
|
|
574
581
|
_initProto(this);
|
|
575
582
|
this.log = createLogger('sequencer:checkpoint-proposal', {
|
|
576
583
|
...bindings,
|
|
577
|
-
instanceId: `slot-${
|
|
584
|
+
instanceId: `slot-${this.slotNow}`
|
|
578
585
|
});
|
|
579
586
|
}
|
|
587
|
+
/** Awaits the pending L1 submission if one is in progress. Call during shutdown. */ async awaitPendingSubmission() {
|
|
588
|
+
this.log.info('Awaiting pending L1 payload submission');
|
|
589
|
+
await this.pendingL1Submission;
|
|
590
|
+
}
|
|
580
591
|
/**
|
|
581
592
|
* Executes the checkpoint proposal job.
|
|
582
|
-
*
|
|
593
|
+
* Builds blocks, collects attestations, enqueues requests, and schedules L1 submission as a
|
|
594
|
+
* background task so the work loop can return to IDLE immediately.
|
|
595
|
+
* Returns the built checkpoint if successful, undefined otherwise.
|
|
583
596
|
*/ async execute() {
|
|
584
597
|
// Enqueue governance and slashing votes (returns promises that will be awaited later)
|
|
585
598
|
// In fisherman mode, we simulate slashing but don't actually publish to L1
|
|
586
599
|
// These are constant for the whole slot, so we only enqueue them once
|
|
587
|
-
const votesPromises = new CheckpointVoter(this.
|
|
588
|
-
// Build and propose the checkpoint.
|
|
589
|
-
|
|
600
|
+
const votesPromises = new CheckpointVoter(this.targetSlot, this.publisher, this.attestorAddress, this.validatorClient, this.slasherClient, this.l1Constants, this.config, this.metrics, this.log).enqueueVotes();
|
|
601
|
+
// Build and propose the checkpoint. Builds blocks, broadcasts, collects attestations, and signs.
|
|
602
|
+
// Does NOT enqueue to L1 yet — that happens after the pipeline sleep.
|
|
603
|
+
const proposalResult = await this.proposeCheckpoint();
|
|
604
|
+
const checkpoint = proposalResult?.checkpoint;
|
|
590
605
|
// Wait until the voting promises have resolved, so all requests are enqueued (not sent)
|
|
591
606
|
await Promise.all(votesPromises);
|
|
592
607
|
if (checkpoint) {
|
|
@@ -597,24 +612,72 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
597
612
|
await this.handleCheckpointEndAsFisherman(checkpoint);
|
|
598
613
|
return;
|
|
599
614
|
}
|
|
600
|
-
//
|
|
601
|
-
|
|
602
|
-
|
|
603
|
-
|
|
604
|
-
|
|
605
|
-
checkpoint
|
|
606
|
-
|
|
607
|
-
}
|
|
608
|
-
const coinbase = checkpoint?.header.coinbase;
|
|
609
|
-
await this.metrics.incFilledSlot(this.publisher.getSenderAddress().toString(), coinbase);
|
|
610
|
-
return checkpoint;
|
|
611
|
-
} else if (checkpoint) {
|
|
612
|
-
this.eventEmitter.emit('checkpoint-publish-failed', {
|
|
613
|
-
...l1Response,
|
|
614
|
-
slot: this.slot
|
|
615
|
-
});
|
|
616
|
-
return undefined;
|
|
615
|
+
// Enqueue the checkpoint for L1 submission
|
|
616
|
+
if (proposalResult) {
|
|
617
|
+
try {
|
|
618
|
+
await this.enqueueCheckpointForSubmission(proposalResult);
|
|
619
|
+
} catch (err) {
|
|
620
|
+
this.log.error(`Failed to enqueue checkpoint for L1 submission at slot ${this.targetSlot}`, err);
|
|
621
|
+
// Continue to sendRequestsAt so votes are still sent
|
|
622
|
+
}
|
|
617
623
|
}
|
|
624
|
+
// Compute the earliest time to submit: pipeline slot start when pipelining, now otherwise.
|
|
625
|
+
const submitAfter = this.epochCache.isProposerPipeliningEnabled() ? new Date(Number(getTimestampForSlot(this.targetSlot, this.l1Constants)) * 1000) : new Date(this.dateProvider.now());
|
|
626
|
+
// TODO(https://github.com/AztecProtocol/aztec-packages/pull/21250): should discard the pending submission if a reorg occurs underneath
|
|
627
|
+
// Schedule L1 submission in the background so the work loop returns immediately.
|
|
628
|
+
// The publisher will sleep until submitAfter, then send the bundled requests.
|
|
629
|
+
// The promise is stored so it can be awaited during shutdown.
|
|
630
|
+
this.pendingL1Submission = this.publisher.sendRequestsAt(submitAfter).then(async (l1Response)=>{
|
|
631
|
+
const proposedAction = l1Response?.successfulActions.find((a)=>a === 'propose');
|
|
632
|
+
if (proposedAction) {
|
|
633
|
+
this.eventEmitter.emit('checkpoint-published', {
|
|
634
|
+
checkpoint: this.checkpointNumber,
|
|
635
|
+
slot: this.targetSlot
|
|
636
|
+
});
|
|
637
|
+
const coinbase = checkpoint?.header.coinbase;
|
|
638
|
+
await this.metrics.incFilledSlot(this.publisher.getSenderAddress().toString(), coinbase);
|
|
639
|
+
} else if (checkpoint) {
|
|
640
|
+
this.eventEmitter.emit('checkpoint-publish-failed', {
|
|
641
|
+
...l1Response,
|
|
642
|
+
slot: this.targetSlot
|
|
643
|
+
});
|
|
644
|
+
if (this.epochCache.isProposerPipeliningEnabled()) {
|
|
645
|
+
this.metrics.recordPipelineDiscard();
|
|
646
|
+
}
|
|
647
|
+
}
|
|
648
|
+
}).catch((err)=>{
|
|
649
|
+
this.log.error(`Background L1 submission failed for slot ${this.targetSlot}`, err);
|
|
650
|
+
if (checkpoint) {
|
|
651
|
+
this.eventEmitter.emit('checkpoint-publish-failed', {
|
|
652
|
+
slot: this.targetSlot
|
|
653
|
+
});
|
|
654
|
+
if (this.epochCache.isProposerPipeliningEnabled()) {
|
|
655
|
+
this.metrics.recordPipelineDiscard();
|
|
656
|
+
}
|
|
657
|
+
}
|
|
658
|
+
});
|
|
659
|
+
// Return the built checkpoint immediately — the work loop is now unblocked
|
|
660
|
+
return checkpoint;
|
|
661
|
+
}
|
|
662
|
+
/** Enqueues the checkpoint for L1 submission. Called after pipeline sleep in execute(). */ async enqueueCheckpointForSubmission(result) {
|
|
663
|
+
const { checkpoint, attestations, attestationsSignature } = result;
|
|
664
|
+
this.setStateFn(SequencerState.PUBLISHING_CHECKPOINT, this.targetSlot);
|
|
665
|
+
const aztecSlotDuration = this.l1Constants.slotDuration;
|
|
666
|
+
const submissionSlotStart = Number(getTimestampForSlot(this.targetSlot, this.l1Constants));
|
|
667
|
+
const txTimeoutAt = new Date((submissionSlotStart + aztecSlotDuration) * 1000);
|
|
668
|
+
// If we have been configured to potentially skip publishing checkpoint then roll the dice here
|
|
669
|
+
if (this.config.skipPublishingCheckpointsPercent !== undefined && this.config.skipPublishingCheckpointsPercent > 0) {
|
|
670
|
+
const roll = Math.max(0, randomInt(100));
|
|
671
|
+
if (roll < this.config.skipPublishingCheckpointsPercent) {
|
|
672
|
+
this.log.warn(`Skipping publishing proposal for checkpoint ${checkpoint.number}. Configured percentage: ${this.config.skipPublishingCheckpointsPercent}, generated value: ${roll}`);
|
|
673
|
+
return;
|
|
674
|
+
}
|
|
675
|
+
}
|
|
676
|
+
await this.publisher.enqueueProposeCheckpoint(checkpoint, attestations, attestationsSignature, {
|
|
677
|
+
txTimeoutAt,
|
|
678
|
+
forcePendingCheckpointNumber: this.invalidateCheckpoint?.forcePendingCheckpointNumber,
|
|
679
|
+
forceProposedFeeHeader: this.computedForceProposedFeeHeader
|
|
680
|
+
});
|
|
618
681
|
}
|
|
619
682
|
async proposeCheckpoint() {
|
|
620
683
|
try {
|
|
@@ -628,19 +691,37 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
628
691
|
const coinbase = this.validatorClient.getCoinbaseForAttestor(this.attestorAddress);
|
|
629
692
|
const feeRecipient = this.validatorClient.getFeeRecipientForAttestor(this.attestorAddress);
|
|
630
693
|
// Start the checkpoint
|
|
631
|
-
this.setStateFn(SequencerState.INITIALIZING_CHECKPOINT, this.
|
|
632
|
-
this.
|
|
694
|
+
this.setStateFn(SequencerState.INITIALIZING_CHECKPOINT, this.targetSlot);
|
|
695
|
+
this.log.info(`Starting checkpoint proposal`, {
|
|
696
|
+
buildSlot: this.slotNow,
|
|
697
|
+
submissionSlot: this.targetSlot,
|
|
698
|
+
pipelining: this.epochCache.isProposerPipeliningEnabled(),
|
|
699
|
+
proposer: this.proposer?.toString(),
|
|
700
|
+
coinbase: coinbase.toString()
|
|
701
|
+
});
|
|
702
|
+
this.metrics.incOpenSlot(this.targetSlot, this.proposer?.toString() ?? 'unknown');
|
|
633
703
|
// Enqueues checkpoint invalidation (constant for the whole slot)
|
|
634
704
|
if (this.invalidateCheckpoint && !this.config.skipInvalidateBlockAsProposer) {
|
|
635
705
|
this.publisher.enqueueInvalidateCheckpoint(this.invalidateCheckpoint);
|
|
636
706
|
}
|
|
637
|
-
// Create checkpoint builder for the slot
|
|
638
|
-
|
|
707
|
+
// Create checkpoint builder for the slot.
|
|
708
|
+
// When pipelining, force the proposed checkpoint number and fee header to our parent so the
|
|
709
|
+
// fee computation sees the same chain tip that L1 will see once the previous pipelined checkpoint lands.
|
|
710
|
+
const isPipelining = this.epochCache.isProposerPipeliningEnabled();
|
|
711
|
+
const parentCheckpointNumber = isPipelining ? CheckpointNumber(this.checkpointNumber - 1) : undefined;
|
|
712
|
+
// Compute the parent's fee header override when pipelining
|
|
713
|
+
if (isPipelining && this.proposedCheckpointData) {
|
|
714
|
+
this.computedForceProposedFeeHeader = await this.computeForceProposedFeeHeader(parentCheckpointNumber);
|
|
715
|
+
}
|
|
716
|
+
const checkpointGlobalVariables = await this.globalsBuilder.buildCheckpointGlobalVariables(coinbase, feeRecipient, this.targetSlot, {
|
|
717
|
+
forcePendingCheckpointNumber: parentCheckpointNumber,
|
|
718
|
+
forceProposedFeeHeader: this.computedForceProposedFeeHeader
|
|
719
|
+
});
|
|
639
720
|
// Collect L1 to L2 messages for the checkpoint and compute their hash
|
|
640
721
|
const l1ToL2Messages = await this.l1ToL2MessageSource.getL1ToL2Messages(this.checkpointNumber);
|
|
641
722
|
const inHash = computeInHashFromL1ToL2Messages(l1ToL2Messages);
|
|
642
723
|
// Collect the out hashes of all the checkpoints before this one in the same epoch
|
|
643
|
-
const previousCheckpointOutHashes = (await this.l2BlockSource.getCheckpointsDataForEpoch(this.
|
|
724
|
+
const previousCheckpointOutHashes = (await this.l2BlockSource.getCheckpointsDataForEpoch(this.targetEpoch)).filter((c)=>c.checkpointNumber < this.checkpointNumber).map((c)=>c.checkpointOutHash);
|
|
644
725
|
// Get the fee asset price modifier from the oracle
|
|
645
726
|
const feeAssetPriceModifier = await this.publisher.getFeeAssetPriceModifier();
|
|
646
727
|
const fork = _ts_add_disposable_resource(env, await this.worldState.fork(this.syncedToBlockNumber, {
|
|
@@ -675,18 +756,18 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
675
756
|
throw err;
|
|
676
757
|
}
|
|
677
758
|
if (blocksInCheckpoint.length === 0) {
|
|
678
|
-
this.log.warn(`No blocks were built for slot ${this.
|
|
679
|
-
slot: this.
|
|
759
|
+
this.log.warn(`No blocks were built for slot ${this.targetSlot}`, {
|
|
760
|
+
slot: this.targetSlot
|
|
680
761
|
});
|
|
681
762
|
this.eventEmitter.emit('checkpoint-empty', {
|
|
682
|
-
slot: this.
|
|
763
|
+
slot: this.targetSlot
|
|
683
764
|
});
|
|
684
765
|
return undefined;
|
|
685
766
|
}
|
|
686
767
|
const minBlocksForCheckpoint = this.config.minBlocksForCheckpoint;
|
|
687
768
|
if (minBlocksForCheckpoint !== undefined && blocksInCheckpoint.length < minBlocksForCheckpoint) {
|
|
688
769
|
this.log.warn(`Checkpoint has fewer blocks than minimum (${blocksInCheckpoint.length} < ${minBlocksForCheckpoint}), skipping proposal`, {
|
|
689
|
-
slot: this.
|
|
770
|
+
slot: this.targetSlot,
|
|
690
771
|
blocksBuilt: blocksInCheckpoint.length,
|
|
691
772
|
minBlocksForCheckpoint
|
|
692
773
|
});
|
|
@@ -694,19 +775,39 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
694
775
|
}
|
|
695
776
|
// Assemble and broadcast the checkpoint proposal, including the last block that was not
|
|
696
777
|
// broadcasted yet, and wait to collect the committee attestations.
|
|
697
|
-
this.setStateFn(SequencerState.ASSEMBLING_CHECKPOINT, this.
|
|
778
|
+
this.setStateFn(SequencerState.ASSEMBLING_CHECKPOINT, this.targetSlot);
|
|
698
779
|
const checkpoint = await checkpointBuilder.completeCheckpoint();
|
|
780
|
+
// Final validation: per-block limits are only checked if the operator set them explicitly.
|
|
781
|
+
// Otherwise, checkpoint-level budgets were already enforced by the redistribution logic.
|
|
782
|
+
try {
|
|
783
|
+
validateCheckpoint(checkpoint, {
|
|
784
|
+
rollupManaLimit: this.l1Constants.rollupManaLimit,
|
|
785
|
+
maxL2BlockGas: this.config.maxL2BlockGas,
|
|
786
|
+
maxDABlockGas: this.config.maxDABlockGas,
|
|
787
|
+
maxTxsPerBlock: this.config.maxTxsPerBlock,
|
|
788
|
+
maxTxsPerCheckpoint: this.config.maxTxsPerCheckpoint
|
|
789
|
+
});
|
|
790
|
+
} catch (err) {
|
|
791
|
+
this.log.error(`Built an invalid checkpoint at slot ${this.slotNow} (skipping proposal)`, err, {
|
|
792
|
+
checkpoint: checkpoint.header.toInspect()
|
|
793
|
+
});
|
|
794
|
+
return undefined;
|
|
795
|
+
}
|
|
699
796
|
// Record checkpoint-level build metrics
|
|
700
797
|
this.metrics.recordCheckpointBuild(checkpointBuildTimer.ms(), blocksInCheckpoint.length, checkpoint.getStats().txCount, Number(checkpoint.header.totalManaUsed.toBigInt()));
|
|
701
798
|
// Do not collect attestations nor publish to L1 in fisherman mode
|
|
702
799
|
if (this.config.fishermanMode) {
|
|
703
|
-
this.log.info(`Built checkpoint for slot ${this.
|
|
704
|
-
slot: this.
|
|
800
|
+
this.log.info(`Built checkpoint for slot ${this.targetSlot} with ${blocksInCheckpoint.length} blocks. ` + `Skipping proposal in fisherman mode.`, {
|
|
801
|
+
slot: this.targetSlot,
|
|
705
802
|
checkpoint: checkpoint.header.toInspect(),
|
|
706
803
|
blocksBuilt: blocksInCheckpoint.length
|
|
707
804
|
});
|
|
708
805
|
this.metrics.recordCheckpointSuccess();
|
|
709
|
-
return
|
|
806
|
+
return {
|
|
807
|
+
checkpoint,
|
|
808
|
+
attestations: CommitteeAttestationsAndSigners.empty(),
|
|
809
|
+
attestationsSignature: Signature.empty()
|
|
810
|
+
};
|
|
710
811
|
}
|
|
711
812
|
// Include the block pending broadcast in the checkpoint proposal if any
|
|
712
813
|
const lastBlock = blockPendingBroadcast && {
|
|
@@ -718,7 +819,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
718
819
|
const proposal = await this.validatorClient.createCheckpointProposal(checkpoint.header, checkpoint.archive.root, feeAssetPriceModifier, lastBlock, this.proposer, checkpointProposalOptions);
|
|
719
820
|
const blockProposedAt = this.dateProvider.now();
|
|
720
821
|
await this.p2pClient.broadcastCheckpointProposal(proposal);
|
|
721
|
-
this.setStateFn(SequencerState.COLLECTING_ATTESTATIONS, this.
|
|
822
|
+
this.setStateFn(SequencerState.COLLECTING_ATTESTATIONS, this.targetSlot);
|
|
722
823
|
const attestations = await this.waitForAttestations(proposal);
|
|
723
824
|
const blockAttestedAt = this.dateProvider.now();
|
|
724
825
|
this.metrics.recordCheckpointAttestationDelay(blockAttestedAt - blockProposedAt);
|
|
@@ -726,7 +827,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
726
827
|
const signer = this.proposer ?? this.publisher.getSenderAddress();
|
|
727
828
|
let attestationsSignature;
|
|
728
829
|
try {
|
|
729
|
-
attestationsSignature = await this.validatorClient.signAttestationsAndSigners(attestations, signer, this.
|
|
830
|
+
attestationsSignature = await this.validatorClient.signAttestationsAndSigners(attestations, signer, this.targetSlot, this.checkpointNumber);
|
|
730
831
|
} catch (err) {
|
|
731
832
|
// We shouldn't really get here since we yield to another HA node
|
|
732
833
|
// as soon as we see these errors when creating block or checkpoint proposals.
|
|
@@ -735,24 +836,12 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
735
836
|
}
|
|
736
837
|
throw err;
|
|
737
838
|
}
|
|
738
|
-
//
|
|
739
|
-
|
|
740
|
-
|
|
741
|
-
|
|
742
|
-
|
|
743
|
-
|
|
744
|
-
if (this.config.skipPublishingCheckpointsPercent !== undefined && this.config.skipPublishingCheckpointsPercent > 0) {
|
|
745
|
-
const result = Math.max(0, randomInt(100));
|
|
746
|
-
if (result < this.config.skipPublishingCheckpointsPercent) {
|
|
747
|
-
this.log.warn(`Skipping publishing proposal for checkpoint ${checkpoint.number}. Configured percentage: ${this.config.skipPublishingCheckpointsPercent}, generated value: ${result}`);
|
|
748
|
-
return checkpoint;
|
|
749
|
-
}
|
|
750
|
-
}
|
|
751
|
-
await this.publisher.enqueueProposeCheckpoint(checkpoint, attestations, attestationsSignature, {
|
|
752
|
-
txTimeoutAt,
|
|
753
|
-
forcePendingCheckpointNumber: this.invalidateCheckpoint?.forcePendingCheckpointNumber
|
|
754
|
-
});
|
|
755
|
-
return checkpoint;
|
|
839
|
+
// Return the result for the caller to enqueue after the pipeline sleep
|
|
840
|
+
return {
|
|
841
|
+
checkpoint,
|
|
842
|
+
attestations,
|
|
843
|
+
attestationsSignature
|
|
844
|
+
};
|
|
756
845
|
} catch (e) {
|
|
757
846
|
env.error = e;
|
|
758
847
|
env.hasError = true;
|
|
@@ -765,7 +854,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
765
854
|
// swallow this error. It's already been logged by a function deeper in the stack
|
|
766
855
|
return undefined;
|
|
767
856
|
}
|
|
768
|
-
this.log.error(`Error building checkpoint at slot ${this.
|
|
857
|
+
this.log.error(`Error building checkpoint at slot ${this.targetSlot}`, err);
|
|
769
858
|
return undefined;
|
|
770
859
|
}
|
|
771
860
|
}
|
|
@@ -775,8 +864,6 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
775
864
|
const blocksInCheckpoint = [];
|
|
776
865
|
const txHashesAlreadyIncluded = new Set();
|
|
777
866
|
const initialBlockNumber = BlockNumber(this.syncedToBlockNumber + 1);
|
|
778
|
-
// Remaining blob fields available for blocks (checkpoint end marker already subtracted)
|
|
779
|
-
let remainingBlobFields = BLOBS_PER_CHECKPOINT * FIELDS_PER_BLOB - NUM_CHECKPOINT_END_MARKER_FIELDS;
|
|
780
867
|
// Last block in the checkpoint will usually be flagged as pending broadcast, so we send it along with the checkpoint proposal
|
|
781
868
|
let blockPendingBroadcast = undefined;
|
|
782
869
|
while(true){
|
|
@@ -787,7 +874,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
787
874
|
const timingInfo = this.timetable.canStartNextBlock(secondsIntoSlot);
|
|
788
875
|
if (!timingInfo.canStart) {
|
|
789
876
|
this.log.debug(`Not enough time left in slot to start another block`, {
|
|
790
|
-
slot: this.
|
|
877
|
+
slot: this.targetSlot,
|
|
791
878
|
blocksBuilt,
|
|
792
879
|
secondsIntoSlot
|
|
793
880
|
});
|
|
@@ -802,8 +889,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
802
889
|
buildDeadline: timingInfo.deadline ? new Date((this.getSlotStartBuildTimestamp() + timingInfo.deadline) * 1000) : undefined,
|
|
803
890
|
blockNumber,
|
|
804
891
|
indexWithinCheckpoint,
|
|
805
|
-
txHashesAlreadyIncluded
|
|
806
|
-
remainingBlobFields
|
|
892
|
+
txHashesAlreadyIncluded
|
|
807
893
|
});
|
|
808
894
|
// TODO(palla/mbps): Review these conditions. We may want to keep trying in some scenarios.
|
|
809
895
|
if (!buildResult && timingInfo.isLastBlock) {
|
|
@@ -817,33 +903,23 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
817
903
|
} else if ('error' in buildResult) {
|
|
818
904
|
// If there was an error building the block, just exit the loop and give up the rest of the slot
|
|
819
905
|
if (!(buildResult.error instanceof SequencerInterruptedError)) {
|
|
820
|
-
this.log.warn(`Halting block building for slot ${this.
|
|
821
|
-
slot: this.
|
|
906
|
+
this.log.warn(`Halting block building for slot ${this.targetSlot}`, {
|
|
907
|
+
slot: this.targetSlot,
|
|
822
908
|
blocksBuilt,
|
|
823
909
|
error: buildResult.error
|
|
824
910
|
});
|
|
825
911
|
}
|
|
826
912
|
break;
|
|
827
913
|
}
|
|
828
|
-
const { block, usedTxs
|
|
914
|
+
const { block, usedTxs } = buildResult;
|
|
829
915
|
blocksInCheckpoint.push(block);
|
|
830
|
-
// Update remaining blob fields for the next block
|
|
831
|
-
remainingBlobFields = newRemainingBlobFields;
|
|
832
|
-
// Sync the proposed block to the archiver to make it available
|
|
833
|
-
// Note that the checkpoint builder uses its own fork so it should not need to wait for this syncing
|
|
834
|
-
// Eventually we should refactor the checkpoint builder to not need a separate long-lived fork
|
|
835
|
-
// Fire and forget - don't block the critical path, but log errors
|
|
836
|
-
this.syncProposedBlockToArchiver(block).catch((err)=>{
|
|
837
|
-
this.log.error(`Failed to sync proposed block ${block.number} to archiver`, {
|
|
838
|
-
blockNumber: block.number,
|
|
839
|
-
err
|
|
840
|
-
});
|
|
841
|
-
});
|
|
842
916
|
usedTxs.forEach((tx)=>txHashesAlreadyIncluded.add(tx.txHash.toString()));
|
|
843
|
-
// If this is the last block,
|
|
917
|
+
// If this is the last block, sync it to the archiver and exit the loop
|
|
918
|
+
// so we can build the checkpoint and start collecting attestations.
|
|
844
919
|
if (timingInfo.isLastBlock) {
|
|
845
|
-
this.
|
|
846
|
-
|
|
920
|
+
await this.syncProposedBlockToArchiver(block);
|
|
921
|
+
this.log.verbose(`Completed final block ${blockNumber} for slot ${this.targetSlot}`, {
|
|
922
|
+
slot: this.targetSlot,
|
|
847
923
|
blockNumber,
|
|
848
924
|
blocksBuilt
|
|
849
925
|
});
|
|
@@ -853,17 +929,22 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
853
929
|
};
|
|
854
930
|
break;
|
|
855
931
|
}
|
|
856
|
-
//
|
|
857
|
-
//
|
|
858
|
-
if
|
|
859
|
-
|
|
860
|
-
|
|
861
|
-
|
|
932
|
+
// Broadcast the block proposal (unless we're in fisherman mode) unless the block is the last one,
|
|
933
|
+
// in which case we'll broadcast it along with the checkpoint at the end of the loop.
|
|
934
|
+
// Note that we only send the block to the archiver if we manage to create the proposal, so if there's
|
|
935
|
+
// a HA error we don't pollute our archiver with a block that won't make it to the chain.
|
|
936
|
+
const proposal = await this.createBlockProposal(block, inHash, usedTxs, blockProposalOptions);
|
|
937
|
+
// Sync the proposed block to the archiver to make it available, only after we've managed to sign the proposal.
|
|
938
|
+
// We wait for the sync to succeed, as this helps catch consistency errors, even if it means we lose some time for block-building.
|
|
939
|
+
// If this throws, we abort the entire checkpoint.
|
|
940
|
+
await this.syncProposedBlockToArchiver(block);
|
|
941
|
+
// Once we have a signed proposal and the archiver agreed with our proposed block, then we broadcast it.
|
|
942
|
+
proposal && await this.p2pClient.broadcastProposal(proposal);
|
|
862
943
|
// Wait until the next block's start time
|
|
863
944
|
await this.waitUntilNextSubslot(timingInfo.deadline);
|
|
864
945
|
}
|
|
865
|
-
this.log.verbose(`Block building loop completed for slot ${this.
|
|
866
|
-
slot: this.
|
|
946
|
+
this.log.verbose(`Block building loop completed for slot ${this.targetSlot}`, {
|
|
947
|
+
slot: this.targetSlot,
|
|
867
948
|
blocksBuilt: blocksInCheckpoint.length
|
|
868
949
|
});
|
|
869
950
|
return {
|
|
@@ -871,33 +952,39 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
871
952
|
blockPendingBroadcast
|
|
872
953
|
};
|
|
873
954
|
}
|
|
955
|
+
/** Creates a block proposal for a given block via the validator client (unless in fisherman mode) */ createBlockProposal(block, inHash, usedTxs, blockProposalOptions) {
|
|
956
|
+
if (this.config.fishermanMode) {
|
|
957
|
+
this.log.info(`Skipping block proposal for block ${block.number} in fisherman mode`);
|
|
958
|
+
return Promise.resolve(undefined);
|
|
959
|
+
}
|
|
960
|
+
return this.validatorClient.createBlockProposal(block.header, block.indexWithinCheckpoint, inHash, block.archive.root, usedTxs, this.proposer, blockProposalOptions);
|
|
961
|
+
}
|
|
874
962
|
/** Sleeps until it is time to produce the next block in the slot */ async waitUntilNextSubslot(nextSubslotStart) {
|
|
875
|
-
this.setStateFn(SequencerState.WAITING_UNTIL_NEXT_BLOCK, this.
|
|
963
|
+
this.setStateFn(SequencerState.WAITING_UNTIL_NEXT_BLOCK, this.targetSlot);
|
|
876
964
|
this.log.verbose(`Waiting until time for the next block at ${nextSubslotStart}s into slot`, {
|
|
877
|
-
slot: this.
|
|
965
|
+
slot: this.targetSlot
|
|
878
966
|
});
|
|
879
967
|
await this.waitUntilTimeInSlot(nextSubslotStart);
|
|
880
968
|
}
|
|
881
969
|
/** Builds a single block. Called from the main block building loop. */ async buildSingleBlock(checkpointBuilder, opts) {
|
|
882
|
-
const { blockTimestamp, forceCreate, blockNumber, indexWithinCheckpoint, buildDeadline, txHashesAlreadyIncluded
|
|
883
|
-
this.log.verbose(`Preparing block ${blockNumber} index ${indexWithinCheckpoint} at checkpoint ${this.checkpointNumber} for slot ${this.
|
|
970
|
+
const { blockTimestamp, forceCreate, blockNumber, indexWithinCheckpoint, buildDeadline, txHashesAlreadyIncluded } = opts;
|
|
971
|
+
this.log.verbose(`Preparing block ${blockNumber} index ${indexWithinCheckpoint} at checkpoint ${this.checkpointNumber} for slot ${this.targetSlot}`, {
|
|
884
972
|
...checkpointBuilder.getConstantData(),
|
|
885
973
|
...opts
|
|
886
974
|
});
|
|
887
975
|
try {
|
|
888
976
|
// Wait until we have enough txs to build the block
|
|
889
|
-
const minTxs = this.
|
|
890
|
-
const { availableTxs, canStartBuilding } = await this.waitForMinTxs(opts);
|
|
977
|
+
const { availableTxs, canStartBuilding, minTxs } = await this.waitForMinTxs(opts);
|
|
891
978
|
if (!canStartBuilding) {
|
|
892
|
-
this.log.warn(`Not enough txs to build block ${blockNumber} at index ${indexWithinCheckpoint} in slot ${this.
|
|
979
|
+
this.log.warn(`Not enough txs to build block ${blockNumber} at index ${indexWithinCheckpoint} in slot ${this.targetSlot} (got ${availableTxs} txs but needs ${minTxs})`, {
|
|
893
980
|
blockNumber,
|
|
894
|
-
slot: this.
|
|
981
|
+
slot: this.targetSlot,
|
|
895
982
|
indexWithinCheckpoint
|
|
896
983
|
});
|
|
897
984
|
this.eventEmitter.emit('block-tx-count-check-failed', {
|
|
898
985
|
minTxs,
|
|
899
986
|
availableTxs,
|
|
900
|
-
slot: this.
|
|
987
|
+
slot: this.targetSlot
|
|
901
988
|
});
|
|
902
989
|
this.metrics.recordBlockProposalFailed('insufficient_txs');
|
|
903
990
|
return undefined;
|
|
@@ -905,48 +992,48 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
905
992
|
// Create iterator to pending txs. We filter out txs already included in previous blocks in the checkpoint
|
|
906
993
|
// just in case p2p failed to sync the provisional block and didn't get to remove those txs from the mempool yet.
|
|
907
994
|
const pendingTxs = filter(this.p2pClient.iterateEligiblePendingTxs(), (tx)=>!txHashesAlreadyIncluded.has(tx.txHash.toString()));
|
|
908
|
-
this.log.debug(`Building block ${blockNumber} at index ${indexWithinCheckpoint} for slot ${this.
|
|
909
|
-
slot: this.
|
|
995
|
+
this.log.debug(`Building block ${blockNumber} at index ${indexWithinCheckpoint} for slot ${this.targetSlot} with ${availableTxs} available txs`, {
|
|
996
|
+
slot: this.targetSlot,
|
|
910
997
|
blockNumber,
|
|
911
998
|
indexWithinCheckpoint
|
|
912
999
|
});
|
|
913
|
-
this.setStateFn(SequencerState.CREATING_BLOCK, this.
|
|
914
|
-
//
|
|
915
|
-
|
|
916
|
-
|
|
1000
|
+
this.setStateFn(SequencerState.CREATING_BLOCK, this.targetSlot);
|
|
1001
|
+
// Per-block limits are operator overrides (from SEQ_MAX_L2_BLOCK_GAS etc.) further capped
|
|
1002
|
+
// by remaining checkpoint-level budgets inside CheckpointBuilder before each block is built.
|
|
1003
|
+
// minValidTxs is passed into the builder so it can reject the block *before* updating state.
|
|
1004
|
+
const minValidTxs = forceCreate ? 0 : this.config.minValidTxsPerBlock ?? minTxs;
|
|
917
1005
|
const blockBuilderOptions = {
|
|
918
1006
|
maxTransactions: this.config.maxTxsPerBlock,
|
|
919
|
-
|
|
920
|
-
|
|
921
|
-
|
|
922
|
-
|
|
1007
|
+
maxBlockGas: this.config.maxL2BlockGas !== undefined || this.config.maxDABlockGas !== undefined ? new Gas(this.config.maxDABlockGas ?? Infinity, this.config.maxL2BlockGas ?? Infinity) : undefined,
|
|
1008
|
+
deadline: buildDeadline,
|
|
1009
|
+
isBuildingProposal: true,
|
|
1010
|
+
minValidTxs,
|
|
1011
|
+
maxBlocksPerCheckpoint: this.timetable.maxNumberOfBlocks,
|
|
1012
|
+
perBlockAllocationMultiplier: this.config.perBlockAllocationMultiplier
|
|
923
1013
|
};
|
|
924
|
-
// Actually build the block by executing txs
|
|
1014
|
+
// Actually build the block by executing txs. The builder throws InsufficientValidTxsError
|
|
1015
|
+
// if the number of successfully processed txs is below minValidTxs, ensuring state is not
|
|
1016
|
+
// updated for blocks that will be discarded.
|
|
925
1017
|
const buildResult = await this.buildSingleBlockWithCheckpointBuilder(checkpointBuilder, pendingTxs, blockNumber, blockTimestamp, blockBuilderOptions);
|
|
926
1018
|
// If any txs failed during execution, drop them from the mempool so we don't pick them up again
|
|
927
1019
|
await this.dropFailedTxsFromP2P(buildResult.failedTxs);
|
|
928
|
-
|
|
929
|
-
|
|
930
|
-
|
|
931
|
-
const numTxs = buildResult.status === 'no-valid-txs' ? 0 : buildResult.numTxs;
|
|
932
|
-
if (buildResult.status === 'no-valid-txs' || !forceCreate && numTxs < minValidTxs) {
|
|
933
|
-
this.log.warn(`Block ${blockNumber} at index ${indexWithinCheckpoint} on slot ${this.slot} has too few valid txs to be proposed`, {
|
|
934
|
-
slot: this.slot,
|
|
1020
|
+
if (buildResult.status === 'insufficient-valid-txs') {
|
|
1021
|
+
this.log.warn(`Block ${blockNumber} at index ${indexWithinCheckpoint} on slot ${this.targetSlot} has too few valid txs to be proposed`, {
|
|
1022
|
+
slot: this.targetSlot,
|
|
935
1023
|
blockNumber,
|
|
936
|
-
numTxs,
|
|
1024
|
+
numTxs: buildResult.processedCount,
|
|
937
1025
|
indexWithinCheckpoint,
|
|
938
|
-
minValidTxs
|
|
939
|
-
buildResult: buildResult.status
|
|
1026
|
+
minValidTxs
|
|
940
1027
|
});
|
|
941
1028
|
this.eventEmitter.emit('block-build-failed', {
|
|
942
1029
|
reason: `Insufficient valid txs`,
|
|
943
|
-
slot: this.
|
|
1030
|
+
slot: this.targetSlot
|
|
944
1031
|
});
|
|
945
1032
|
this.metrics.recordBlockProposalFailed('insufficient_valid_txs');
|
|
946
1033
|
return undefined;
|
|
947
1034
|
}
|
|
948
1035
|
// Block creation succeeded, emit stats and metrics
|
|
949
|
-
const {
|
|
1036
|
+
const { block, publicProcessorDuration, usedTxs, blockBuildDuration, numTxs } = buildResult;
|
|
950
1037
|
const blockStats = {
|
|
951
1038
|
eventName: 'l2-block-built',
|
|
952
1039
|
duration: blockBuildDuration,
|
|
@@ -955,31 +1042,33 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
955
1042
|
};
|
|
956
1043
|
const blockHash = await block.hash();
|
|
957
1044
|
const txHashes = block.body.txEffects.map((tx)=>tx.txHash);
|
|
958
|
-
const manaPerSec =
|
|
959
|
-
this.log.info(`Built block ${block.number} at checkpoint ${this.checkpointNumber} for slot ${this.
|
|
1045
|
+
const manaPerSec = block.header.totalManaUsed.toNumberUnsafe() / (blockBuildDuration / 1000);
|
|
1046
|
+
this.log.info(`Built block ${block.number} at checkpoint ${this.checkpointNumber} for slot ${this.targetSlot} with ${numTxs} txs`, {
|
|
960
1047
|
blockHash,
|
|
961
1048
|
txHashes,
|
|
962
1049
|
manaPerSec,
|
|
963
1050
|
...blockStats
|
|
964
1051
|
});
|
|
1052
|
+
// `slot` is the target/submission slot (may be one ahead when pipelining),
|
|
1053
|
+
// `buildSlot` is the wall-clock slot during which the block was actually built.
|
|
965
1054
|
this.eventEmitter.emit('block-proposed', {
|
|
966
1055
|
blockNumber: block.number,
|
|
967
|
-
slot: this.
|
|
1056
|
+
slot: this.targetSlot,
|
|
1057
|
+
buildSlot: this.slotNow
|
|
968
1058
|
});
|
|
969
|
-
this.metrics.recordBuiltBlock(blockBuildDuration,
|
|
1059
|
+
this.metrics.recordBuiltBlock(blockBuildDuration, block.header.totalManaUsed.toNumberUnsafe());
|
|
970
1060
|
return {
|
|
971
1061
|
block,
|
|
972
|
-
usedTxs
|
|
973
|
-
remainingBlobFields: maxBlobFieldsForTxs - usedTxBlobFields
|
|
1062
|
+
usedTxs
|
|
974
1063
|
};
|
|
975
1064
|
} catch (err) {
|
|
976
1065
|
this.eventEmitter.emit('block-build-failed', {
|
|
977
1066
|
reason: err.message,
|
|
978
|
-
slot: this.
|
|
1067
|
+
slot: this.targetSlot
|
|
979
1068
|
});
|
|
980
1069
|
this.log.error(`Error building block`, err, {
|
|
981
1070
|
blockNumber,
|
|
982
|
-
slot: this.
|
|
1071
|
+
slot: this.targetSlot
|
|
983
1072
|
});
|
|
984
1073
|
this.metrics.recordBlockProposalFailed(err.name || 'unknown_error');
|
|
985
1074
|
this.metrics.recordFailedBlock();
|
|
@@ -988,7 +1077,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
988
1077
|
};
|
|
989
1078
|
}
|
|
990
1079
|
}
|
|
991
|
-
/** Uses the checkpoint builder to build a block, catching
|
|
1080
|
+
/** Uses the checkpoint builder to build a block, catching InsufficientValidTxsError. */ async buildSingleBlockWithCheckpointBuilder(checkpointBuilder, pendingTxs, blockNumber, blockTimestamp, blockBuilderOptions) {
|
|
992
1081
|
try {
|
|
993
1082
|
const workTimer = new Timer();
|
|
994
1083
|
const result = await checkpointBuilder.buildBlock(pendingTxs, blockNumber, blockTimestamp, blockBuilderOptions);
|
|
@@ -999,10 +1088,11 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
999
1088
|
status: 'success'
|
|
1000
1089
|
};
|
|
1001
1090
|
} catch (err) {
|
|
1002
|
-
if (isErrorClass(err,
|
|
1091
|
+
if (isErrorClass(err, InsufficientValidTxsError)) {
|
|
1003
1092
|
return {
|
|
1004
1093
|
failedTxs: err.failedTxs,
|
|
1005
|
-
|
|
1094
|
+
processedCount: err.processedCount,
|
|
1095
|
+
status: 'insufficient-valid-txs'
|
|
1006
1096
|
};
|
|
1007
1097
|
}
|
|
1008
1098
|
throw err;
|
|
@@ -1021,14 +1111,15 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
1021
1111
|
if (startBuildingDeadline === undefined || now >= startBuildingDeadline) {
|
|
1022
1112
|
return {
|
|
1023
1113
|
canStartBuilding: false,
|
|
1024
|
-
availableTxs
|
|
1114
|
+
availableTxs,
|
|
1115
|
+
minTxs
|
|
1025
1116
|
};
|
|
1026
1117
|
}
|
|
1027
1118
|
// Wait a bit before checking again
|
|
1028
|
-
this.setStateFn(SequencerState.WAITING_FOR_TXS, this.
|
|
1029
|
-
this.log.verbose(`Waiting for enough txs to build block ${blockNumber} at index ${indexWithinCheckpoint} in slot ${this.
|
|
1119
|
+
this.setStateFn(SequencerState.WAITING_FOR_TXS, this.targetSlot);
|
|
1120
|
+
this.log.verbose(`Waiting for enough txs to build block ${blockNumber} at index ${indexWithinCheckpoint} in slot ${this.targetSlot} (have ${availableTxs} but need ${minTxs})`, {
|
|
1030
1121
|
blockNumber,
|
|
1031
|
-
slot: this.
|
|
1122
|
+
slot: this.targetSlot,
|
|
1032
1123
|
indexWithinCheckpoint
|
|
1033
1124
|
});
|
|
1034
1125
|
await this.waitForTxsPollingInterval();
|
|
@@ -1036,7 +1127,8 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
1036
1127
|
}
|
|
1037
1128
|
return {
|
|
1038
1129
|
canStartBuilding: true,
|
|
1039
|
-
availableTxs
|
|
1130
|
+
availableTxs,
|
|
1131
|
+
minTxs
|
|
1040
1132
|
};
|
|
1041
1133
|
}
|
|
1042
1134
|
/**
|
|
@@ -1059,7 +1151,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
1059
1151
|
committee
|
|
1060
1152
|
});
|
|
1061
1153
|
}
|
|
1062
|
-
const numberOfRequiredAttestations =
|
|
1154
|
+
const numberOfRequiredAttestations = computeQuorum(committee.length);
|
|
1063
1155
|
if (this.config.skipCollectingAttestations) {
|
|
1064
1156
|
this.log.warn('Skipping attestation collection as per config (attesting with own keys only)');
|
|
1065
1157
|
const attestations = await this.validatorClient?.collectOwnAttestations(proposal);
|
|
@@ -1165,7 +1257,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
1165
1257
|
* Gossip doesn't echo messages back to the sender, so the proposer's archiver/world-state
|
|
1166
1258
|
* would never receive its own block without this explicit sync.
|
|
1167
1259
|
*/ async syncProposedBlockToArchiver(block) {
|
|
1168
|
-
if (this.config.skipPushProposedBlocksToArchiver
|
|
1260
|
+
if (this.config.skipPushProposedBlocksToArchiver) {
|
|
1169
1261
|
this.log.warn(`Skipping push of proposed block ${block.number} to archiver`, {
|
|
1170
1262
|
blockNumber: block.number,
|
|
1171
1263
|
slot: block.header.globalVariables.slotNumber
|
|
@@ -1181,16 +1273,16 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
1181
1273
|
/** Runs fee analysis and logs checkpoint outcome as fisherman */ async handleCheckpointEndAsFisherman(checkpoint) {
|
|
1182
1274
|
// Perform L1 fee analysis before clearing requests
|
|
1183
1275
|
// The callback is invoked asynchronously after the next block is mined
|
|
1184
|
-
const feeAnalysis = await this.publisher.analyzeL1Fees(this.
|
|
1276
|
+
const feeAnalysis = await this.publisher.analyzeL1Fees(this.targetSlot, (analysis)=>this.metrics.recordFishermanFeeAnalysis(analysis));
|
|
1185
1277
|
if (checkpoint) {
|
|
1186
|
-
this.log.info(`Validation checkpoint building SUCCEEDED for slot ${this.
|
|
1278
|
+
this.log.info(`Validation checkpoint building SUCCEEDED for slot ${this.targetSlot}`, {
|
|
1187
1279
|
...checkpoint.toCheckpointInfo(),
|
|
1188
1280
|
...checkpoint.getStats(),
|
|
1189
1281
|
feeAnalysisId: feeAnalysis?.id
|
|
1190
1282
|
});
|
|
1191
1283
|
} else {
|
|
1192
|
-
this.log.warn(`Validation block building FAILED for slot ${this.
|
|
1193
|
-
slot: this.
|
|
1284
|
+
this.log.warn(`Validation block building FAILED for slot ${this.targetSlot}`, {
|
|
1285
|
+
slot: this.targetSlot,
|
|
1194
1286
|
feeAnalysisId: feeAnalysis?.id
|
|
1195
1287
|
});
|
|
1196
1288
|
this.metrics.recordCheckpointProposalFailed('block_build_failed');
|
|
@@ -1201,15 +1293,15 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
1201
1293
|
* Helper to handle HA double-signing errors. Returns true if the error was handled (caller should yield).
|
|
1202
1294
|
*/ handleHASigningError(err, errorContext) {
|
|
1203
1295
|
if (err instanceof DutyAlreadySignedError) {
|
|
1204
|
-
this.log.info(`${errorContext} for slot ${this.
|
|
1205
|
-
slot: this.
|
|
1296
|
+
this.log.info(`${errorContext} for slot ${this.targetSlot} already signed by another HA node, yielding`, {
|
|
1297
|
+
slot: this.targetSlot,
|
|
1206
1298
|
signedByNode: err.signedByNode
|
|
1207
1299
|
});
|
|
1208
1300
|
return true;
|
|
1209
1301
|
}
|
|
1210
1302
|
if (err instanceof SlashingProtectionError) {
|
|
1211
|
-
this.log.info(`${errorContext} for slot ${this.
|
|
1212
|
-
slot: this.
|
|
1303
|
+
this.log.info(`${errorContext} for slot ${this.targetSlot} blocked by slashing protection, yielding`, {
|
|
1304
|
+
slot: this.targetSlot,
|
|
1213
1305
|
existingMessageHash: err.existingMessageHash,
|
|
1214
1306
|
attemptedMessageHash: err.attemptedMessageHash
|
|
1215
1307
|
});
|
|
@@ -1217,6 +1309,40 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
1217
1309
|
}
|
|
1218
1310
|
return false;
|
|
1219
1311
|
}
|
|
1312
|
+
/**
|
|
1313
|
+
* In times of congestion we need to simulate using the correct fee header override for the previous block
|
|
1314
|
+
* We calculate the correct fee header values.
|
|
1315
|
+
*
|
|
1316
|
+
* If we are in block 1, or the checkpoint we are querying does not exist, we return undefined. However
|
|
1317
|
+
* If we are pipelining - where this function is called, the grandparentCheckpointNumber should always exist
|
|
1318
|
+
* @param parentCheckpointNumber
|
|
1319
|
+
* @returns
|
|
1320
|
+
*/ async computeForceProposedFeeHeader(parentCheckpointNumber) {
|
|
1321
|
+
if (!this.proposedCheckpointData) {
|
|
1322
|
+
return undefined;
|
|
1323
|
+
}
|
|
1324
|
+
const rollup = this.publisher.rollupContract;
|
|
1325
|
+
const grandparentCheckpointNumber = CheckpointNumber(this.checkpointNumber - 2);
|
|
1326
|
+
try {
|
|
1327
|
+
const [grandparentCheckpoint, manaTarget] = await Promise.all([
|
|
1328
|
+
rollup.getCheckpoint(grandparentCheckpointNumber),
|
|
1329
|
+
rollup.getManaTarget()
|
|
1330
|
+
]);
|
|
1331
|
+
if (!grandparentCheckpoint || !grandparentCheckpoint.feeHeader) {
|
|
1332
|
+
this.log.error(`Grandparent checkpoint or its feeHeader is undefined for checkpointNumber=${grandparentCheckpointNumber.toString()}`);
|
|
1333
|
+
return undefined;
|
|
1334
|
+
} else {
|
|
1335
|
+
const parentFeeHeader = RollupContract.computeChildFeeHeader(grandparentCheckpoint.feeHeader, this.proposedCheckpointData.totalManaUsed, this.proposedCheckpointData.feeAssetPriceModifier, manaTarget);
|
|
1336
|
+
return {
|
|
1337
|
+
checkpointNumber: parentCheckpointNumber,
|
|
1338
|
+
feeHeader: parentFeeHeader
|
|
1339
|
+
};
|
|
1340
|
+
}
|
|
1341
|
+
} catch (err) {
|
|
1342
|
+
this.log.error(`Failed to fetch grandparent checkpoint or mana target for checkpointNumber=${grandparentCheckpointNumber.toString()}: ${err}`);
|
|
1343
|
+
return undefined;
|
|
1344
|
+
}
|
|
1345
|
+
}
|
|
1220
1346
|
/** Waits until a specific time within the current slot */ async waitUntilTimeInSlot(targetSecondsIntoSlot) {
|
|
1221
1347
|
const slotStartTimestamp = this.getSlotStartBuildTimestamp();
|
|
1222
1348
|
const targetTimestamp = slotStartTimestamp + targetSecondsIntoSlot;
|
|
@@ -1226,7 +1352,7 @@ _dec = trackSpan('CheckpointProposalJob.execute'), _dec1 = trackSpan('Checkpoint
|
|
|
1226
1352
|
await sleep(TXS_POLLING_MS);
|
|
1227
1353
|
}
|
|
1228
1354
|
getSlotStartBuildTimestamp() {
|
|
1229
|
-
return getSlotStartBuildTimestamp(this.
|
|
1355
|
+
return getSlotStartBuildTimestamp(this.slotNow, this.l1Constants);
|
|
1230
1356
|
}
|
|
1231
1357
|
getSecondsIntoSlot() {
|
|
1232
1358
|
const slotStartTimestamp = this.getSlotStartBuildTimestamp();
|