@aztec/sequencer-client 0.0.1-commit.c2595eba → 0.0.1-commit.c2eed6949
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 +15 -7
- package/dest/client/sequencer-client.d.ts.map +1 -1
- package/dest/client/sequencer-client.js +60 -26
- package/dest/config.d.ts +26 -7
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +47 -28
- package/dest/global_variable_builder/global_builder.d.ts +14 -10
- package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
- package/dest/global_variable_builder/global_builder.js +22 -21
- 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 +47 -17
- package/dest/publisher/config.d.ts.map +1 -1
- package/dest/publisher/config.js +121 -42
- package/dest/publisher/index.d.ts +2 -1
- package/dest/publisher/index.d.ts.map +1 -1
- package/dest/publisher/l1_tx_failed_store/factory.d.ts +11 -0
- package/dest/publisher/l1_tx_failed_store/factory.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/factory.js +22 -0
- package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts +59 -0
- package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/failed_tx_store.js +1 -0
- package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts +15 -0
- package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.js +34 -0
- package/dest/publisher/l1_tx_failed_store/index.d.ts +4 -0
- package/dest/publisher/l1_tx_failed_store/index.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/index.js +2 -0
- package/dest/publisher/sequencer-publisher-factory.d.ts +11 -3
- package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher-factory.js +27 -2
- package/dest/publisher/sequencer-publisher.d.ts +33 -10
- package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher.js +371 -57
- package/dest/sequencer/checkpoint_proposal_job.d.ts +39 -10
- package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -1
- package/dest/sequencer/checkpoint_proposal_job.js +287 -167
- package/dest/sequencer/events.d.ts +2 -1
- package/dest/sequencer/events.d.ts.map +1 -1
- package/dest/sequencer/metrics.d.ts +21 -5
- package/dest/sequencer/metrics.d.ts.map +1 -1
- package/dest/sequencer/metrics.js +97 -15
- package/dest/sequencer/sequencer.d.ts +30 -15
- package/dest/sequencer/sequencer.d.ts.map +1 -1
- package/dest/sequencer/sequencer.js +95 -82
- package/dest/sequencer/timetable.d.ts +4 -6
- package/dest/sequencer/timetable.d.ts.map +1 -1
- package/dest/sequencer/timetable.js +7 -11
- package/dest/sequencer/types.d.ts +2 -2
- package/dest/sequencer/types.d.ts.map +1 -1
- package/dest/test/index.d.ts +3 -5
- package/dest/test/index.d.ts.map +1 -1
- package/dest/test/mock_checkpoint_builder.d.ts +12 -12
- package/dest/test/mock_checkpoint_builder.d.ts.map +1 -1
- package/dest/test/mock_checkpoint_builder.js +45 -36
- package/dest/test/utils.d.ts +3 -3
- package/dest/test/utils.d.ts.map +1 -1
- package/dest/test/utils.js +5 -4
- package/package.json +27 -28
- package/src/client/sequencer-client.ts +76 -23
- package/src/config.ts +65 -38
- package/src/global_variable_builder/global_builder.ts +23 -24
- package/src/global_variable_builder/index.ts +1 -1
- package/src/publisher/config.ts +153 -43
- package/src/publisher/index.ts +3 -0
- package/src/publisher/l1_tx_failed_store/factory.ts +32 -0
- package/src/publisher/l1_tx_failed_store/failed_tx_store.ts +55 -0
- package/src/publisher/l1_tx_failed_store/file_store_failed_tx_store.ts +46 -0
- package/src/publisher/l1_tx_failed_store/index.ts +3 -0
- package/src/publisher/sequencer-publisher-factory.ts +38 -6
- package/src/publisher/sequencer-publisher.ts +371 -70
- package/src/sequencer/checkpoint_proposal_job.ts +392 -193
- package/src/sequencer/events.ts +1 -1
- package/src/sequencer/metrics.ts +106 -18
- package/src/sequencer/sequencer.ts +131 -94
- package/src/sequencer/timetable.ts +13 -12
- package/src/sequencer/types.ts +1 -1
- package/src/test/index.ts +2 -4
- package/src/test/mock_checkpoint_builder.ts +65 -53
- package/src/test/utils.ts +5 -2
|
@@ -1,8 +1,17 @@
|
|
|
1
|
-
import { NUM_CHECKPOINT_END_MARKER_FIELDS, getNumBlockEndBlobFields } from '@aztec/blob-lib/encoding';
|
|
2
|
-
import { BLOBS_PER_CHECKPOINT, FIELDS_PER_BLOB } from '@aztec/constants';
|
|
3
1
|
import type { EpochCache } from '@aztec/epoch-cache';
|
|
4
|
-
import {
|
|
2
|
+
import {
|
|
3
|
+
BlockNumber,
|
|
4
|
+
CheckpointNumber,
|
|
5
|
+
EpochNumber,
|
|
6
|
+
IndexWithinCheckpoint,
|
|
7
|
+
SlotNumber,
|
|
8
|
+
} from '@aztec/foundation/branded-types';
|
|
5
9
|
import { randomInt } from '@aztec/foundation/crypto/random';
|
|
10
|
+
import {
|
|
11
|
+
flipSignature,
|
|
12
|
+
generateRecoverableSignature,
|
|
13
|
+
generateUnrecoverableSignature,
|
|
14
|
+
} from '@aztec/foundation/crypto/secp256k1-signer';
|
|
6
15
|
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
7
16
|
import { EthAddress } from '@aztec/foundation/eth-address';
|
|
8
17
|
import { Signature } from '@aztec/foundation/eth-signature';
|
|
@@ -10,7 +19,7 @@ import { filter } from '@aztec/foundation/iterator';
|
|
|
10
19
|
import { type Logger, type LoggerBindings, createLogger } from '@aztec/foundation/log';
|
|
11
20
|
import { sleep, sleepUntil } from '@aztec/foundation/sleep';
|
|
12
21
|
import { type DateProvider, Timer } from '@aztec/foundation/timer';
|
|
13
|
-
import { type TypedEventEmitter, unfreeze } from '@aztec/foundation/types';
|
|
22
|
+
import { type TypedEventEmitter, isErrorClass, unfreeze } from '@aztec/foundation/types';
|
|
14
23
|
import type { P2P } from '@aztec/p2p';
|
|
15
24
|
import type { SlasherClientInterface } from '@aztec/slasher';
|
|
16
25
|
import {
|
|
@@ -21,17 +30,23 @@ import {
|
|
|
21
30
|
type L2BlockSource,
|
|
22
31
|
MaliciousCommitteeAttestationsAndSigners,
|
|
23
32
|
} from '@aztec/stdlib/block';
|
|
24
|
-
import type
|
|
25
|
-
import { getSlotStartBuildTimestamp } from '@aztec/stdlib/epoch-helpers';
|
|
33
|
+
import { type Checkpoint, validateCheckpoint } from '@aztec/stdlib/checkpoint';
|
|
34
|
+
import { getSlotStartBuildTimestamp, getTimestampForSlot } from '@aztec/stdlib/epoch-helpers';
|
|
26
35
|
import { Gas } from '@aztec/stdlib/gas';
|
|
27
|
-
import
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
36
|
+
import {
|
|
37
|
+
type BlockBuilderOptions,
|
|
38
|
+
InsufficientValidTxsError,
|
|
39
|
+
type ResolvedSequencerConfig,
|
|
40
|
+
type WorldStateSynchronizer,
|
|
31
41
|
} from '@aztec/stdlib/interfaces/server';
|
|
32
42
|
import { type L1ToL2MessageSource, computeInHashFromL1ToL2Messages } from '@aztec/stdlib/messaging';
|
|
33
|
-
import type {
|
|
34
|
-
|
|
43
|
+
import type {
|
|
44
|
+
BlockProposal,
|
|
45
|
+
BlockProposalOptions,
|
|
46
|
+
CheckpointProposal,
|
|
47
|
+
CheckpointProposalOptions,
|
|
48
|
+
} from '@aztec/stdlib/p2p';
|
|
49
|
+
import { orderAttestations, trimAttestations } from '@aztec/stdlib/p2p';
|
|
35
50
|
import type { L2BlockBuiltStats } from '@aztec/stdlib/stats';
|
|
36
51
|
import { type FailedTx, Tx } from '@aztec/stdlib/tx';
|
|
37
52
|
import { AttestationTimeoutError } from '@aztec/stdlib/validators';
|
|
@@ -62,8 +77,10 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
62
77
|
protected readonly log: Logger;
|
|
63
78
|
|
|
64
79
|
constructor(
|
|
65
|
-
private readonly
|
|
66
|
-
private readonly
|
|
80
|
+
private readonly slotNow: SlotNumber,
|
|
81
|
+
private readonly targetSlot: SlotNumber,
|
|
82
|
+
private readonly epochNow: EpochNumber,
|
|
83
|
+
private readonly targetEpoch: EpochNumber,
|
|
67
84
|
private readonly checkpointNumber: CheckpointNumber,
|
|
68
85
|
private readonly syncedToBlockNumber: BlockNumber,
|
|
69
86
|
// TODO(palla/mbps): Can we remove the proposer in favor of attestorAddress? Need to check fisherman-node flows.
|
|
@@ -91,7 +108,20 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
91
108
|
public readonly tracer: Tracer,
|
|
92
109
|
bindings?: LoggerBindings,
|
|
93
110
|
) {
|
|
94
|
-
this.log = createLogger('sequencer:checkpoint-proposal', {
|
|
111
|
+
this.log = createLogger('sequencer:checkpoint-proposal', {
|
|
112
|
+
...bindings,
|
|
113
|
+
instanceId: `slot-${this.slotNow}`,
|
|
114
|
+
});
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
/** The wall-clock slot during which the proposer builds. */
|
|
118
|
+
private get slot(): SlotNumber {
|
|
119
|
+
return this.slotNow;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
/** The wall-clock epoch. */
|
|
123
|
+
private get epoch(): EpochNumber {
|
|
124
|
+
return this.epochNow;
|
|
95
125
|
}
|
|
96
126
|
|
|
97
127
|
/**
|
|
@@ -104,7 +134,7 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
104
134
|
// In fisherman mode, we simulate slashing but don't actually publish to L1
|
|
105
135
|
// These are constant for the whole slot, so we only enqueue them once
|
|
106
136
|
const votesPromises = new CheckpointVoter(
|
|
107
|
-
this.
|
|
137
|
+
this.targetSlot,
|
|
108
138
|
this.publisher,
|
|
109
139
|
this.attestorAddress,
|
|
110
140
|
this.validatorClient,
|
|
@@ -122,7 +152,7 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
122
152
|
await Promise.all(votesPromises);
|
|
123
153
|
|
|
124
154
|
if (checkpoint) {
|
|
125
|
-
this.metrics.
|
|
155
|
+
this.metrics.recordCheckpointProposalSuccess();
|
|
126
156
|
}
|
|
127
157
|
|
|
128
158
|
// Do not post anything to L1 if we are fishermen, but do perform L1 fee analysis
|
|
@@ -131,6 +161,29 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
131
161
|
return;
|
|
132
162
|
}
|
|
133
163
|
|
|
164
|
+
// If pipelining, wait until the submission slot so L1 recognizes the pipelined proposer
|
|
165
|
+
if (this.epochCache.isProposerPipeliningEnabled()) {
|
|
166
|
+
const submissionSlotTimestamp =
|
|
167
|
+
getTimestampForSlot(this.targetSlot, this.l1Constants) - BigInt(this.l1Constants.ethereumSlotDuration);
|
|
168
|
+
this.log.info(`Waiting until submission slot ${this.targetSlot} for L1 submission`, {
|
|
169
|
+
slot: this.slot,
|
|
170
|
+
submissionSlot: this.targetSlot,
|
|
171
|
+
submissionSlotTimestamp,
|
|
172
|
+
});
|
|
173
|
+
await sleepUntil(new Date(Number(submissionSlotTimestamp) * 1000), this.dateProvider.nowAsDate());
|
|
174
|
+
|
|
175
|
+
// After waking, verify the parent checkpoint wasn't pruned during the sleep.
|
|
176
|
+
// We check L1's pending tip directly instead of canProposeAt, which also validates the proposer
|
|
177
|
+
// identity and would fail because the timestamp resolves to a different slot's proposer.
|
|
178
|
+
const l1Tips = await this.publisher.rollupContract.getTips();
|
|
179
|
+
if (l1Tips.pending < this.checkpointNumber - 1) {
|
|
180
|
+
this.log.warn(
|
|
181
|
+
`Parent checkpoint was pruned during pipelining sleep (L1 pending=${l1Tips.pending}, expected>=${this.checkpointNumber - 1}), skipping L1 submission for checkpoint ${this.checkpointNumber}`,
|
|
182
|
+
);
|
|
183
|
+
return undefined;
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
|
|
134
187
|
// Then send everything to L1
|
|
135
188
|
const l1Response = await this.publisher.sendRequests();
|
|
136
189
|
const proposedAction = l1Response?.successfulActions.find(a => a === 'propose');
|
|
@@ -149,7 +202,7 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
149
202
|
return {
|
|
150
203
|
// nullish operator needed for tests
|
|
151
204
|
[Attributes.COINBASE]: this.validatorClient.getCoinbaseForAttestor(this.attestorAddress)?.toString(),
|
|
152
|
-
[Attributes.SLOT_NUMBER]: this.
|
|
205
|
+
[Attributes.SLOT_NUMBER]: this.targetSlot,
|
|
153
206
|
};
|
|
154
207
|
})
|
|
155
208
|
private async proposeCheckpoint(): Promise<Checkpoint | undefined> {
|
|
@@ -159,8 +212,15 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
159
212
|
const feeRecipient = this.validatorClient.getFeeRecipientForAttestor(this.attestorAddress);
|
|
160
213
|
|
|
161
214
|
// Start the checkpoint
|
|
162
|
-
this.setStateFn(SequencerState.INITIALIZING_CHECKPOINT, this.
|
|
163
|
-
this.
|
|
215
|
+
this.setStateFn(SequencerState.INITIALIZING_CHECKPOINT, this.targetSlot);
|
|
216
|
+
this.log.info(`Starting checkpoint proposal`, {
|
|
217
|
+
buildSlot: this.slot,
|
|
218
|
+
submissionSlot: this.targetSlot,
|
|
219
|
+
pipelining: this.epochCache.isProposerPipeliningEnabled(),
|
|
220
|
+
proposer: this.proposer?.toString(),
|
|
221
|
+
coinbase: coinbase.toString(),
|
|
222
|
+
});
|
|
223
|
+
this.metrics.incOpenSlot(this.targetSlot, this.proposer?.toString() ?? 'unknown');
|
|
164
224
|
|
|
165
225
|
// Enqueues checkpoint invalidation (constant for the whole slot)
|
|
166
226
|
if (this.invalidateCheckpoint && !this.config.skipInvalidateBlockAsProposer) {
|
|
@@ -171,7 +231,7 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
171
231
|
const checkpointGlobalVariables = await this.globalsBuilder.buildCheckpointGlobalVariables(
|
|
172
232
|
coinbase,
|
|
173
233
|
feeRecipient,
|
|
174
|
-
this.
|
|
234
|
+
this.targetSlot,
|
|
175
235
|
);
|
|
176
236
|
|
|
177
237
|
// Collect L1 to L2 messages for the checkpoint and compute their hash
|
|
@@ -179,18 +239,21 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
179
239
|
const inHash = computeInHashFromL1ToL2Messages(l1ToL2Messages);
|
|
180
240
|
|
|
181
241
|
// Collect the out hashes of all the checkpoints before this one in the same epoch
|
|
182
|
-
const
|
|
183
|
-
c => c.
|
|
184
|
-
|
|
185
|
-
|
|
242
|
+
const previousCheckpointOutHashes = (await this.l2BlockSource.getCheckpointsDataForEpoch(this.targetEpoch))
|
|
243
|
+
.filter(c => c.checkpointNumber < this.checkpointNumber)
|
|
244
|
+
.map(c => c.checkpointOutHash);
|
|
245
|
+
|
|
246
|
+
// Get the fee asset price modifier from the oracle
|
|
247
|
+
const feeAssetPriceModifier = await this.publisher.getFeeAssetPriceModifier();
|
|
186
248
|
|
|
187
249
|
// Create a long-lived forked world state for the checkpoint builder
|
|
188
|
-
using fork = await this.worldState.fork(this.syncedToBlockNumber, { closeDelayMs: 12_000 });
|
|
250
|
+
await using fork = await this.worldState.fork(this.syncedToBlockNumber, { closeDelayMs: 12_000 });
|
|
189
251
|
|
|
190
252
|
// Create checkpoint builder for the entire slot
|
|
191
253
|
const checkpointBuilder = await this.checkpointsBuilder.startCheckpoint(
|
|
192
254
|
this.checkpointNumber,
|
|
193
255
|
checkpointGlobalVariables,
|
|
256
|
+
feeAssetPriceModifier,
|
|
194
257
|
l1ToL2Messages,
|
|
195
258
|
previousCheckpointOutHashes,
|
|
196
259
|
fork,
|
|
@@ -210,6 +273,7 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
210
273
|
|
|
211
274
|
let blocksInCheckpoint: L2Block[] = [];
|
|
212
275
|
let blockPendingBroadcast: { block: L2Block; txs: Tx[] } | undefined = undefined;
|
|
276
|
+
const checkpointBuildTimer = new Timer();
|
|
213
277
|
|
|
214
278
|
try {
|
|
215
279
|
// Main loop: build blocks for the checkpoint
|
|
@@ -225,42 +289,64 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
225
289
|
// These errors are expected in HA mode, so we yield and let another HA node handle the slot
|
|
226
290
|
// The only distinction between the 2 errors is SlashingProtectionError throws when the payload is different,
|
|
227
291
|
// which is normal for block building (may have picked different txs)
|
|
228
|
-
if (err
|
|
229
|
-
this.log.info(`Checkpoint proposal for slot ${this.slot} already signed by another HA node, yielding`, {
|
|
230
|
-
slot: this.slot,
|
|
231
|
-
signedByNode: err.signedByNode,
|
|
232
|
-
});
|
|
233
|
-
return undefined;
|
|
234
|
-
}
|
|
235
|
-
if (err instanceof SlashingProtectionError) {
|
|
236
|
-
this.log.info(`Checkpoint proposal for slot ${this.slot} blocked by slashing protection, yielding`, {
|
|
237
|
-
slot: this.slot,
|
|
238
|
-
existingMessageHash: err.existingMessageHash,
|
|
239
|
-
attemptedMessageHash: err.attemptedMessageHash,
|
|
240
|
-
});
|
|
292
|
+
if (this.handleHASigningError(err, 'Block proposal')) {
|
|
241
293
|
return undefined;
|
|
242
294
|
}
|
|
243
295
|
throw err;
|
|
244
296
|
}
|
|
245
297
|
|
|
246
298
|
if (blocksInCheckpoint.length === 0) {
|
|
247
|
-
this.log.warn(`No blocks were built for slot ${this.
|
|
248
|
-
this.eventEmitter.emit('checkpoint-empty', { slot: this.
|
|
299
|
+
this.log.warn(`No blocks were built for slot ${this.targetSlot}`, { slot: this.targetSlot });
|
|
300
|
+
this.eventEmitter.emit('checkpoint-empty', { slot: this.targetSlot });
|
|
301
|
+
return undefined;
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
const minBlocksForCheckpoint = this.config.minBlocksForCheckpoint;
|
|
305
|
+
if (minBlocksForCheckpoint !== undefined && blocksInCheckpoint.length < minBlocksForCheckpoint) {
|
|
306
|
+
this.log.warn(
|
|
307
|
+
`Checkpoint has fewer blocks than minimum (${blocksInCheckpoint.length} < ${minBlocksForCheckpoint}), skipping proposal`,
|
|
308
|
+
{ slot: this.targetSlot, blocksBuilt: blocksInCheckpoint.length, minBlocksForCheckpoint },
|
|
309
|
+
);
|
|
249
310
|
return undefined;
|
|
250
311
|
}
|
|
251
312
|
|
|
252
313
|
// Assemble and broadcast the checkpoint proposal, including the last block that was not
|
|
253
314
|
// broadcasted yet, and wait to collect the committee attestations.
|
|
254
|
-
this.setStateFn(SequencerState.ASSEMBLING_CHECKPOINT, this.
|
|
315
|
+
this.setStateFn(SequencerState.ASSEMBLING_CHECKPOINT, this.targetSlot);
|
|
255
316
|
const checkpoint = await checkpointBuilder.completeCheckpoint();
|
|
256
317
|
|
|
318
|
+
// Final validation: per-block limits are only checked if the operator set them explicitly.
|
|
319
|
+
// Otherwise, checkpoint-level budgets were already enforced by the redistribution logic.
|
|
320
|
+
try {
|
|
321
|
+
validateCheckpoint(checkpoint, {
|
|
322
|
+
rollupManaLimit: this.l1Constants.rollupManaLimit,
|
|
323
|
+
maxL2BlockGas: this.config.maxL2BlockGas,
|
|
324
|
+
maxDABlockGas: this.config.maxDABlockGas,
|
|
325
|
+
maxTxsPerBlock: this.config.maxTxsPerBlock,
|
|
326
|
+
maxTxsPerCheckpoint: this.config.maxTxsPerCheckpoint,
|
|
327
|
+
});
|
|
328
|
+
} catch (err) {
|
|
329
|
+
this.log.error(`Built an invalid checkpoint at slot ${this.slot} (skipping proposal)`, err, {
|
|
330
|
+
checkpoint: checkpoint.header.toInspect(),
|
|
331
|
+
});
|
|
332
|
+
return undefined;
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
// Record checkpoint-level build metrics
|
|
336
|
+
this.metrics.recordCheckpointBuild(
|
|
337
|
+
checkpointBuildTimer.ms(),
|
|
338
|
+
blocksInCheckpoint.length,
|
|
339
|
+
checkpoint.getStats().txCount,
|
|
340
|
+
Number(checkpoint.header.totalManaUsed.toBigInt()),
|
|
341
|
+
);
|
|
342
|
+
|
|
257
343
|
// Do not collect attestations nor publish to L1 in fisherman mode
|
|
258
344
|
if (this.config.fishermanMode) {
|
|
259
345
|
this.log.info(
|
|
260
|
-
`Built checkpoint for slot ${this.
|
|
346
|
+
`Built checkpoint for slot ${this.targetSlot} with ${blocksInCheckpoint.length} blocks. ` +
|
|
261
347
|
`Skipping proposal in fisherman mode.`,
|
|
262
348
|
{
|
|
263
|
-
slot: this.
|
|
349
|
+
slot: this.targetSlot,
|
|
264
350
|
checkpoint: checkpoint.header.toInspect(),
|
|
265
351
|
blocksBuilt: blocksInCheckpoint.length,
|
|
266
352
|
},
|
|
@@ -280,6 +366,7 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
280
366
|
const proposal = await this.validatorClient.createCheckpointProposal(
|
|
281
367
|
checkpoint.header,
|
|
282
368
|
checkpoint.archive.root,
|
|
369
|
+
feeAssetPriceModifier,
|
|
283
370
|
lastBlock,
|
|
284
371
|
this.proposer,
|
|
285
372
|
checkpointProposalOptions,
|
|
@@ -288,7 +375,7 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
288
375
|
const blockProposedAt = this.dateProvider.now();
|
|
289
376
|
await this.p2pClient.broadcastCheckpointProposal(proposal);
|
|
290
377
|
|
|
291
|
-
this.setStateFn(SequencerState.COLLECTING_ATTESTATIONS, this.
|
|
378
|
+
this.setStateFn(SequencerState.COLLECTING_ATTESTATIONS, this.targetSlot);
|
|
292
379
|
const attestations = await this.waitForAttestations(proposal);
|
|
293
380
|
const blockAttestedAt = this.dateProvider.now();
|
|
294
381
|
|
|
@@ -301,35 +388,38 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
301
388
|
attestationsSignature = await this.validatorClient.signAttestationsAndSigners(
|
|
302
389
|
attestations,
|
|
303
390
|
signer,
|
|
304
|
-
this.
|
|
391
|
+
this.targetSlot,
|
|
305
392
|
this.checkpointNumber,
|
|
306
393
|
);
|
|
307
394
|
} catch (err) {
|
|
308
395
|
// We shouldn't really get here since we yield to another HA node
|
|
309
|
-
// as soon as we see these errors when creating block proposals.
|
|
310
|
-
if (err
|
|
311
|
-
this.log.info(`Attestations signature for slot ${this.slot} already signed by another HA node, yielding`, {
|
|
312
|
-
slot: this.slot,
|
|
313
|
-
signedByNode: err.signedByNode,
|
|
314
|
-
});
|
|
315
|
-
return undefined;
|
|
316
|
-
}
|
|
317
|
-
if (err instanceof SlashingProtectionError) {
|
|
318
|
-
this.log.info(`Attestations signature for slot ${this.slot} blocked by slashing protection, yielding`, {
|
|
319
|
-
slot: this.slot,
|
|
320
|
-
existingMessageHash: err.existingMessageHash,
|
|
321
|
-
attemptedMessageHash: err.attemptedMessageHash,
|
|
322
|
-
});
|
|
396
|
+
// as soon as we see these errors when creating block or checkpoint proposals.
|
|
397
|
+
if (this.handleHASigningError(err, 'Attestations signature')) {
|
|
323
398
|
return undefined;
|
|
324
399
|
}
|
|
325
400
|
throw err;
|
|
326
401
|
}
|
|
327
402
|
|
|
328
403
|
// Enqueue publishing the checkpoint to L1
|
|
329
|
-
this.setStateFn(SequencerState.PUBLISHING_CHECKPOINT, this.
|
|
404
|
+
this.setStateFn(SequencerState.PUBLISHING_CHECKPOINT, this.targetSlot);
|
|
330
405
|
const aztecSlotDuration = this.l1Constants.slotDuration;
|
|
331
|
-
const
|
|
332
|
-
const txTimeoutAt = new Date((
|
|
406
|
+
const submissionSlotStart = Number(getTimestampForSlot(this.targetSlot, this.l1Constants));
|
|
407
|
+
const txTimeoutAt = new Date((submissionSlotStart + aztecSlotDuration) * 1000);
|
|
408
|
+
|
|
409
|
+
// If we have been configured to potentially skip publishing checkpoint then roll the dice here
|
|
410
|
+
if (
|
|
411
|
+
this.config.skipPublishingCheckpointsPercent !== undefined &&
|
|
412
|
+
this.config.skipPublishingCheckpointsPercent > 0
|
|
413
|
+
) {
|
|
414
|
+
const result = Math.max(0, randomInt(100));
|
|
415
|
+
if (result < this.config.skipPublishingCheckpointsPercent) {
|
|
416
|
+
this.log.warn(
|
|
417
|
+
`Skipping publishing proposal for checkpoint ${checkpoint.number}. Configured percentage: ${this.config.skipPublishingCheckpointsPercent}, generated value: ${result}`,
|
|
418
|
+
);
|
|
419
|
+
return checkpoint;
|
|
420
|
+
}
|
|
421
|
+
}
|
|
422
|
+
|
|
333
423
|
await this.publisher.enqueueProposeCheckpoint(checkpoint, attestations, attestationsSignature, {
|
|
334
424
|
txTimeoutAt,
|
|
335
425
|
forcePendingCheckpointNumber: this.invalidateCheckpoint?.forcePendingCheckpointNumber,
|
|
@@ -364,15 +454,12 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
364
454
|
const txHashesAlreadyIncluded = new Set<string>();
|
|
365
455
|
const initialBlockNumber = BlockNumber(this.syncedToBlockNumber + 1);
|
|
366
456
|
|
|
367
|
-
// Remaining blob fields available for blocks (checkpoint end marker already subtracted)
|
|
368
|
-
let remainingBlobFields = BLOBS_PER_CHECKPOINT * FIELDS_PER_BLOB - NUM_CHECKPOINT_END_MARKER_FIELDS;
|
|
369
|
-
|
|
370
457
|
// Last block in the checkpoint will usually be flagged as pending broadcast, so we send it along with the checkpoint proposal
|
|
371
458
|
let blockPendingBroadcast: { block: L2Block; txs: Tx[] } | undefined = undefined;
|
|
372
459
|
|
|
373
460
|
while (true) {
|
|
374
461
|
const blocksBuilt = blocksInCheckpoint.length;
|
|
375
|
-
const indexWithinCheckpoint = blocksBuilt;
|
|
462
|
+
const indexWithinCheckpoint = IndexWithinCheckpoint(blocksBuilt);
|
|
376
463
|
const blockNumber = BlockNumber(initialBlockNumber + blocksBuilt);
|
|
377
464
|
|
|
378
465
|
const secondsIntoSlot = this.getSecondsIntoSlot();
|
|
@@ -380,7 +467,7 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
380
467
|
|
|
381
468
|
if (!timingInfo.canStart) {
|
|
382
469
|
this.log.debug(`Not enough time left in slot to start another block`, {
|
|
383
|
-
slot: this.
|
|
470
|
+
slot: this.targetSlot,
|
|
384
471
|
blocksBuilt,
|
|
385
472
|
secondsIntoSlot,
|
|
386
473
|
});
|
|
@@ -399,9 +486,9 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
399
486
|
blockNumber,
|
|
400
487
|
indexWithinCheckpoint,
|
|
401
488
|
txHashesAlreadyIncluded,
|
|
402
|
-
remainingBlobFields,
|
|
403
489
|
});
|
|
404
490
|
|
|
491
|
+
// TODO(palla/mbps): Review these conditions. We may want to keep trying in some scenarios.
|
|
405
492
|
if (!buildResult && timingInfo.isLastBlock) {
|
|
406
493
|
// If no block was produced due to not enough txs and this was the last subslot, exit
|
|
407
494
|
break;
|
|
@@ -415,8 +502,8 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
415
502
|
} else if ('error' in buildResult) {
|
|
416
503
|
// If there was an error building the block, just exit the loop and give up the rest of the slot
|
|
417
504
|
if (!(buildResult.error instanceof SequencerInterruptedError)) {
|
|
418
|
-
this.log.warn(`Halting block building for slot ${this.
|
|
419
|
-
slot: this.
|
|
505
|
+
this.log.warn(`Halting block building for slot ${this.targetSlot}`, {
|
|
506
|
+
slot: this.targetSlot,
|
|
420
507
|
blocksBuilt,
|
|
421
508
|
error: buildResult.error,
|
|
422
509
|
});
|
|
@@ -424,26 +511,16 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
424
511
|
break;
|
|
425
512
|
}
|
|
426
513
|
|
|
427
|
-
const { block, usedTxs
|
|
514
|
+
const { block, usedTxs } = buildResult;
|
|
428
515
|
blocksInCheckpoint.push(block);
|
|
429
|
-
|
|
430
|
-
// Update remaining blob fields for the next block
|
|
431
|
-
remainingBlobFields = newRemainingBlobFields;
|
|
432
|
-
|
|
433
|
-
// Sync the proposed block to the archiver to make it available
|
|
434
|
-
// Note that the checkpoint builder uses its own fork so it should not need to wait for this syncing
|
|
435
|
-
// Eventually we should refactor the checkpoint builder to not need a separate long-lived fork
|
|
436
|
-
// Fire and forget - don't block the critical path, but log errors
|
|
437
|
-
this.syncProposedBlockToArchiver(block).catch(err => {
|
|
438
|
-
this.log.error(`Failed to sync proposed block ${block.number} to archiver`, { blockNumber: block.number, err });
|
|
439
|
-
});
|
|
440
|
-
|
|
441
516
|
usedTxs.forEach(tx => txHashesAlreadyIncluded.add(tx.txHash.toString()));
|
|
442
517
|
|
|
443
|
-
// If this is the last block,
|
|
518
|
+
// If this is the last block, sync it to the archiver and exit the loop
|
|
519
|
+
// so we can build the checkpoint and start collecting attestations.
|
|
444
520
|
if (timingInfo.isLastBlock) {
|
|
445
|
-
this.
|
|
446
|
-
|
|
521
|
+
await this.syncProposedBlockToArchiver(block);
|
|
522
|
+
this.log.verbose(`Completed final block ${blockNumber} for slot ${this.targetSlot}`, {
|
|
523
|
+
slot: this.targetSlot,
|
|
447
524
|
blockNumber,
|
|
448
525
|
blocksBuilt,
|
|
449
526
|
});
|
|
@@ -451,80 +528,94 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
451
528
|
break;
|
|
452
529
|
}
|
|
453
530
|
|
|
454
|
-
//
|
|
455
|
-
//
|
|
456
|
-
if
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
}
|
|
531
|
+
// Broadcast the block proposal (unless we're in fisherman mode) unless the block is the last one,
|
|
532
|
+
// in which case we'll broadcast it along with the checkpoint at the end of the loop.
|
|
533
|
+
// Note that we only send the block to the archiver if we manage to create the proposal, so if there's
|
|
534
|
+
// a HA error we don't pollute our archiver with a block that won't make it to the chain.
|
|
535
|
+
const proposal = await this.createBlockProposal(block, inHash, usedTxs, blockProposalOptions);
|
|
536
|
+
|
|
537
|
+
// Sync the proposed block to the archiver to make it available, only after we've managed to sign the proposal.
|
|
538
|
+
// We wait for the sync to succeed, as this helps catch consistency errors, even if it means we lose some time for block-building.
|
|
539
|
+
// If this throws, we abort the entire checkpoint.
|
|
540
|
+
await this.syncProposedBlockToArchiver(block);
|
|
541
|
+
|
|
542
|
+
// Once we have a signed proposal and the archiver agreed with our proposed block, then we broadcast it.
|
|
543
|
+
proposal && (await this.p2pClient.broadcastProposal(proposal));
|
|
468
544
|
|
|
469
545
|
// Wait until the next block's start time
|
|
470
546
|
await this.waitUntilNextSubslot(timingInfo.deadline);
|
|
471
547
|
}
|
|
472
548
|
|
|
473
|
-
this.log.verbose(`Block building loop completed for slot ${this.
|
|
474
|
-
slot: this.
|
|
549
|
+
this.log.verbose(`Block building loop completed for slot ${this.targetSlot}`, {
|
|
550
|
+
slot: this.targetSlot,
|
|
475
551
|
blocksBuilt: blocksInCheckpoint.length,
|
|
476
552
|
});
|
|
477
553
|
|
|
478
554
|
return { blocksInCheckpoint, blockPendingBroadcast };
|
|
479
555
|
}
|
|
480
556
|
|
|
557
|
+
/** Creates a block proposal for a given block via the validator client (unless in fisherman mode) */
|
|
558
|
+
private createBlockProposal(
|
|
559
|
+
block: L2Block,
|
|
560
|
+
inHash: Fr,
|
|
561
|
+
usedTxs: Tx[],
|
|
562
|
+
blockProposalOptions: BlockProposalOptions,
|
|
563
|
+
): Promise<BlockProposal | undefined> {
|
|
564
|
+
if (this.config.fishermanMode) {
|
|
565
|
+
this.log.info(`Skipping block proposal for block ${block.number} in fisherman mode`);
|
|
566
|
+
return Promise.resolve(undefined);
|
|
567
|
+
}
|
|
568
|
+
return this.validatorClient.createBlockProposal(
|
|
569
|
+
block.header,
|
|
570
|
+
block.indexWithinCheckpoint,
|
|
571
|
+
inHash,
|
|
572
|
+
block.archive.root,
|
|
573
|
+
usedTxs,
|
|
574
|
+
this.proposer,
|
|
575
|
+
blockProposalOptions,
|
|
576
|
+
);
|
|
577
|
+
}
|
|
578
|
+
|
|
481
579
|
/** Sleeps until it is time to produce the next block in the slot */
|
|
482
580
|
@trackSpan('CheckpointProposalJob.waitUntilNextSubslot')
|
|
483
581
|
private async waitUntilNextSubslot(nextSubslotStart: number) {
|
|
484
|
-
this.setStateFn(SequencerState.WAITING_UNTIL_NEXT_BLOCK, this.
|
|
485
|
-
this.log.verbose(`Waiting until time for the next block at ${nextSubslotStart}s into slot`, {
|
|
582
|
+
this.setStateFn(SequencerState.WAITING_UNTIL_NEXT_BLOCK, this.targetSlot);
|
|
583
|
+
this.log.verbose(`Waiting until time for the next block at ${nextSubslotStart}s into slot`, {
|
|
584
|
+
slot: this.targetSlot,
|
|
585
|
+
});
|
|
486
586
|
await this.waitUntilTimeInSlot(nextSubslotStart);
|
|
487
587
|
}
|
|
488
588
|
|
|
489
589
|
/** Builds a single block. Called from the main block building loop. */
|
|
490
590
|
@trackSpan('CheckpointProposalJob.buildSingleBlock')
|
|
491
|
-
|
|
591
|
+
protected async buildSingleBlock(
|
|
492
592
|
checkpointBuilder: CheckpointBuilder,
|
|
493
593
|
opts: {
|
|
494
594
|
forceCreate?: boolean;
|
|
495
595
|
blockTimestamp: bigint;
|
|
496
596
|
blockNumber: BlockNumber;
|
|
497
|
-
indexWithinCheckpoint:
|
|
597
|
+
indexWithinCheckpoint: IndexWithinCheckpoint;
|
|
498
598
|
buildDeadline: Date | undefined;
|
|
499
599
|
txHashesAlreadyIncluded: Set<string>;
|
|
500
|
-
remainingBlobFields: number;
|
|
501
600
|
},
|
|
502
|
-
): Promise<{ block: L2Block; usedTxs: Tx[]
|
|
503
|
-
const {
|
|
504
|
-
|
|
505
|
-
forceCreate,
|
|
506
|
-
blockNumber,
|
|
507
|
-
indexWithinCheckpoint,
|
|
508
|
-
buildDeadline,
|
|
509
|
-
txHashesAlreadyIncluded,
|
|
510
|
-
remainingBlobFields,
|
|
511
|
-
} = opts;
|
|
601
|
+
): Promise<{ block: L2Block; usedTxs: Tx[] } | { error: Error } | undefined> {
|
|
602
|
+
const { blockTimestamp, forceCreate, blockNumber, indexWithinCheckpoint, buildDeadline, txHashesAlreadyIncluded } =
|
|
603
|
+
opts;
|
|
512
604
|
|
|
513
605
|
this.log.verbose(
|
|
514
|
-
`Preparing block ${blockNumber} index ${indexWithinCheckpoint} at checkpoint ${this.checkpointNumber} for slot ${this.
|
|
606
|
+
`Preparing block ${blockNumber} index ${indexWithinCheckpoint} at checkpoint ${this.checkpointNumber} for slot ${this.targetSlot}`,
|
|
515
607
|
{ ...checkpointBuilder.getConstantData(), ...opts },
|
|
516
608
|
);
|
|
517
609
|
|
|
518
610
|
try {
|
|
519
611
|
// Wait until we have enough txs to build the block
|
|
520
|
-
const minTxs = this.
|
|
521
|
-
const { availableTxs, canStartBuilding } = await this.waitForMinTxs(opts);
|
|
612
|
+
const { availableTxs, canStartBuilding, minTxs } = await this.waitForMinTxs(opts);
|
|
522
613
|
if (!canStartBuilding) {
|
|
523
614
|
this.log.warn(
|
|
524
|
-
`Not enough txs to build block ${blockNumber} at index ${indexWithinCheckpoint} in slot ${this.
|
|
525
|
-
{ blockNumber, slot: this.
|
|
615
|
+
`Not enough txs to build block ${blockNumber} at index ${indexWithinCheckpoint} in slot ${this.targetSlot} (got ${availableTxs} txs but needs ${minTxs})`,
|
|
616
|
+
{ blockNumber, slot: this.targetSlot, indexWithinCheckpoint },
|
|
526
617
|
);
|
|
527
|
-
this.eventEmitter.emit('block-tx-count-check-failed', { minTxs, availableTxs, slot: this.
|
|
618
|
+
this.eventEmitter.emit('block-tx-count-check-failed', { minTxs, availableTxs, slot: this.targetSlot });
|
|
528
619
|
this.metrics.recordBlockProposalFailed('insufficient_txs');
|
|
529
620
|
return undefined;
|
|
530
621
|
}
|
|
@@ -532,104 +623,143 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
532
623
|
// Create iterator to pending txs. We filter out txs already included in previous blocks in the checkpoint
|
|
533
624
|
// just in case p2p failed to sync the provisional block and didn't get to remove those txs from the mempool yet.
|
|
534
625
|
const pendingTxs = filter(
|
|
535
|
-
this.p2pClient.
|
|
626
|
+
this.p2pClient.iterateEligiblePendingTxs(),
|
|
536
627
|
tx => !txHashesAlreadyIncluded.has(tx.txHash.toString()),
|
|
537
628
|
);
|
|
538
629
|
|
|
539
630
|
this.log.debug(
|
|
540
|
-
`Building block ${blockNumber} at index ${indexWithinCheckpoint} for slot ${this.
|
|
541
|
-
{ slot: this.
|
|
631
|
+
`Building block ${blockNumber} at index ${indexWithinCheckpoint} for slot ${this.targetSlot} with ${availableTxs} available txs`,
|
|
632
|
+
{ slot: this.targetSlot, blockNumber, indexWithinCheckpoint },
|
|
542
633
|
);
|
|
543
|
-
this.setStateFn(SequencerState.CREATING_BLOCK, this.
|
|
544
|
-
|
|
545
|
-
// Calculate blob fields limit for txs (remaining capacity - this block's end overhead)
|
|
546
|
-
const blockEndOverhead = getNumBlockEndBlobFields(indexWithinCheckpoint === 0);
|
|
547
|
-
const maxBlobFieldsForTxs = remainingBlobFields - blockEndOverhead;
|
|
634
|
+
this.setStateFn(SequencerState.CREATING_BLOCK, this.targetSlot);
|
|
548
635
|
|
|
549
|
-
|
|
636
|
+
// Per-block limits are operator overrides (from SEQ_MAX_L2_BLOCK_GAS etc.) further capped
|
|
637
|
+
// by remaining checkpoint-level budgets inside CheckpointBuilder before each block is built.
|
|
638
|
+
// minValidTxs is passed into the builder so it can reject the block *before* updating state.
|
|
639
|
+
const minValidTxs = forceCreate ? 0 : (this.config.minValidTxsPerBlock ?? minTxs);
|
|
640
|
+
const blockBuilderOptions: BlockBuilderOptions = {
|
|
550
641
|
maxTransactions: this.config.maxTxsPerBlock,
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
|
|
642
|
+
maxBlockGas:
|
|
643
|
+
this.config.maxL2BlockGas !== undefined || this.config.maxDABlockGas !== undefined
|
|
644
|
+
? new Gas(this.config.maxDABlockGas ?? Infinity, this.config.maxL2BlockGas ?? Infinity)
|
|
645
|
+
: undefined,
|
|
554
646
|
deadline: buildDeadline,
|
|
647
|
+
isBuildingProposal: true,
|
|
648
|
+
minValidTxs,
|
|
649
|
+
maxBlocksPerCheckpoint: this.timetable.maxNumberOfBlocks,
|
|
650
|
+
perBlockAllocationMultiplier: this.config.perBlockAllocationMultiplier,
|
|
555
651
|
};
|
|
556
652
|
|
|
557
|
-
// Actually build the block by executing txs
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
|
|
565
|
-
|
|
566
|
-
|
|
567
|
-
usedTxBlobFields,
|
|
568
|
-
} = await checkpointBuilder.buildBlock(pendingTxs, blockNumber, blockTimestamp, blockBuilderOptions);
|
|
569
|
-
const blockBuildDuration = workTimer.ms();
|
|
653
|
+
// Actually build the block by executing txs. The builder throws InsufficientValidTxsError
|
|
654
|
+
// if the number of successfully processed txs is below minValidTxs, ensuring state is not
|
|
655
|
+
// updated for blocks that will be discarded.
|
|
656
|
+
const buildResult = await this.buildSingleBlockWithCheckpointBuilder(
|
|
657
|
+
checkpointBuilder,
|
|
658
|
+
pendingTxs,
|
|
659
|
+
blockNumber,
|
|
660
|
+
blockTimestamp,
|
|
661
|
+
blockBuilderOptions,
|
|
662
|
+
);
|
|
570
663
|
|
|
571
664
|
// If any txs failed during execution, drop them from the mempool so we don't pick them up again
|
|
572
|
-
await this.dropFailedTxsFromP2P(failedTxs);
|
|
665
|
+
await this.dropFailedTxsFromP2P(buildResult.failedTxs);
|
|
573
666
|
|
|
574
|
-
|
|
575
|
-
// too long, then we may not get to minTxsPerBlock after executing public functions.
|
|
576
|
-
const minValidTxs = this.config.minValidTxsPerBlock ?? minTxs;
|
|
577
|
-
if (!forceCreate && numTxs < minValidTxs) {
|
|
667
|
+
if (buildResult.status === 'insufficient-valid-txs') {
|
|
578
668
|
this.log.warn(
|
|
579
|
-
`Block ${blockNumber} at index ${indexWithinCheckpoint} on slot ${this.
|
|
580
|
-
{
|
|
669
|
+
`Block ${blockNumber} at index ${indexWithinCheckpoint} on slot ${this.targetSlot} has too few valid txs to be proposed`,
|
|
670
|
+
{
|
|
671
|
+
slot: this.targetSlot,
|
|
672
|
+
blockNumber,
|
|
673
|
+
numTxs: buildResult.processedCount,
|
|
674
|
+
indexWithinCheckpoint,
|
|
675
|
+
minValidTxs,
|
|
676
|
+
},
|
|
581
677
|
);
|
|
582
|
-
this.eventEmitter.emit('block-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
slot: this.slot,
|
|
678
|
+
this.eventEmitter.emit('block-build-failed', {
|
|
679
|
+
reason: `Insufficient valid txs`,
|
|
680
|
+
slot: this.targetSlot,
|
|
586
681
|
});
|
|
587
682
|
this.metrics.recordBlockProposalFailed('insufficient_valid_txs');
|
|
588
683
|
return undefined;
|
|
589
684
|
}
|
|
590
685
|
|
|
591
686
|
// Block creation succeeded, emit stats and metrics
|
|
687
|
+
const { block, publicProcessorDuration, usedTxs, blockBuildDuration, numTxs } = buildResult;
|
|
688
|
+
|
|
592
689
|
const blockStats = {
|
|
593
690
|
eventName: 'l2-block-built',
|
|
594
691
|
duration: blockBuildDuration,
|
|
595
692
|
publicProcessDuration: publicProcessorDuration,
|
|
596
|
-
rollupCircuitsDuration: blockBuildingTimer.ms(),
|
|
597
693
|
...block.getStats(),
|
|
598
694
|
} satisfies L2BlockBuiltStats;
|
|
599
695
|
|
|
600
696
|
const blockHash = await block.hash();
|
|
601
697
|
const txHashes = block.body.txEffects.map(tx => tx.txHash);
|
|
602
|
-
const manaPerSec =
|
|
698
|
+
const manaPerSec = block.header.totalManaUsed.toNumberUnsafe() / (blockBuildDuration / 1000);
|
|
603
699
|
|
|
604
700
|
this.log.info(
|
|
605
|
-
`Built block ${block.number} at checkpoint ${this.checkpointNumber} for slot ${this.
|
|
701
|
+
`Built block ${block.number} at checkpoint ${this.checkpointNumber} for slot ${this.targetSlot} with ${numTxs} txs`,
|
|
606
702
|
{ blockHash, txHashes, manaPerSec, ...blockStats },
|
|
607
703
|
);
|
|
608
704
|
|
|
609
|
-
this.eventEmitter.emit('block-proposed', {
|
|
610
|
-
|
|
705
|
+
this.eventEmitter.emit('block-proposed', {
|
|
706
|
+
blockNumber: block.number,
|
|
707
|
+
slot: this.targetSlot,
|
|
708
|
+
buildSlot: this.slotNow,
|
|
709
|
+
});
|
|
710
|
+
this.metrics.recordBuiltBlock(blockBuildDuration, block.header.totalManaUsed.toNumberUnsafe());
|
|
611
711
|
|
|
612
|
-
return { block, usedTxs
|
|
712
|
+
return { block, usedTxs };
|
|
613
713
|
} catch (err: any) {
|
|
614
|
-
this.eventEmitter.emit('block-build-failed', {
|
|
615
|
-
|
|
714
|
+
this.eventEmitter.emit('block-build-failed', {
|
|
715
|
+
reason: err.message,
|
|
716
|
+
slot: this.targetSlot,
|
|
717
|
+
});
|
|
718
|
+
this.log.error(`Error building block`, err, { blockNumber, slot: this.targetSlot });
|
|
616
719
|
this.metrics.recordBlockProposalFailed(err.name || 'unknown_error');
|
|
617
720
|
this.metrics.recordFailedBlock();
|
|
618
721
|
return { error: err };
|
|
619
722
|
}
|
|
620
723
|
}
|
|
621
724
|
|
|
725
|
+
/** Uses the checkpoint builder to build a block, catching InsufficientValidTxsError. */
|
|
726
|
+
private async buildSingleBlockWithCheckpointBuilder(
|
|
727
|
+
checkpointBuilder: CheckpointBuilder,
|
|
728
|
+
pendingTxs: AsyncIterable<Tx>,
|
|
729
|
+
blockNumber: BlockNumber,
|
|
730
|
+
blockTimestamp: bigint,
|
|
731
|
+
blockBuilderOptions: BlockBuilderOptions,
|
|
732
|
+
) {
|
|
733
|
+
try {
|
|
734
|
+
const workTimer = new Timer();
|
|
735
|
+
const result = await checkpointBuilder.buildBlock(pendingTxs, blockNumber, blockTimestamp, blockBuilderOptions);
|
|
736
|
+
const blockBuildDuration = workTimer.ms();
|
|
737
|
+
return { ...result, blockBuildDuration, status: 'success' as const };
|
|
738
|
+
} catch (err: unknown) {
|
|
739
|
+
if (isErrorClass(err, InsufficientValidTxsError)) {
|
|
740
|
+
return {
|
|
741
|
+
failedTxs: err.failedTxs,
|
|
742
|
+
processedCount: err.processedCount,
|
|
743
|
+
status: 'insufficient-valid-txs' as const,
|
|
744
|
+
};
|
|
745
|
+
}
|
|
746
|
+
throw err;
|
|
747
|
+
}
|
|
748
|
+
}
|
|
749
|
+
|
|
622
750
|
/** Waits until minTxs are available on the pool for building a block. */
|
|
623
751
|
@trackSpan('CheckpointProposalJob.waitForMinTxs')
|
|
624
752
|
private async waitForMinTxs(opts: {
|
|
625
753
|
forceCreate?: boolean;
|
|
626
754
|
blockNumber: BlockNumber;
|
|
627
|
-
indexWithinCheckpoint:
|
|
755
|
+
indexWithinCheckpoint: IndexWithinCheckpoint;
|
|
628
756
|
buildDeadline: Date | undefined;
|
|
629
|
-
}): Promise<{ canStartBuilding: boolean; availableTxs: number }> {
|
|
630
|
-
const minTxs = this.config.minTxsPerBlock;
|
|
757
|
+
}): Promise<{ canStartBuilding: boolean; availableTxs: number; minTxs: number }> {
|
|
631
758
|
const { indexWithinCheckpoint, blockNumber, buildDeadline, forceCreate } = opts;
|
|
632
759
|
|
|
760
|
+
// We only allow a block with 0 txs in the first block of the checkpoint
|
|
761
|
+
const minTxs = indexWithinCheckpoint > 0 && this.config.minTxsPerBlock === 0 ? 1 : this.config.minTxsPerBlock;
|
|
762
|
+
|
|
633
763
|
// Deadline is undefined if we are not enforcing the timetable, meaning we'll exit immediately when out of time
|
|
634
764
|
const startBuildingDeadline = buildDeadline
|
|
635
765
|
? new Date(buildDeadline.getTime() - this.timetable.minExecutionTime * 1000)
|
|
@@ -641,20 +771,20 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
641
771
|
// If we're past deadline, or we have no deadline, give up
|
|
642
772
|
const now = this.dateProvider.nowAsDate();
|
|
643
773
|
if (startBuildingDeadline === undefined || now >= startBuildingDeadline) {
|
|
644
|
-
return { canStartBuilding: false, availableTxs
|
|
774
|
+
return { canStartBuilding: false, availableTxs, minTxs };
|
|
645
775
|
}
|
|
646
776
|
|
|
647
777
|
// Wait a bit before checking again
|
|
648
|
-
this.setStateFn(SequencerState.WAITING_FOR_TXS, this.
|
|
778
|
+
this.setStateFn(SequencerState.WAITING_FOR_TXS, this.targetSlot);
|
|
649
779
|
this.log.verbose(
|
|
650
|
-
`Waiting for enough txs to build block ${blockNumber} at index ${indexWithinCheckpoint} in slot ${this.
|
|
651
|
-
{ blockNumber, slot: this.
|
|
780
|
+
`Waiting for enough txs to build block ${blockNumber} at index ${indexWithinCheckpoint} in slot ${this.targetSlot} (have ${availableTxs} but need ${minTxs})`,
|
|
781
|
+
{ blockNumber, slot: this.targetSlot, indexWithinCheckpoint },
|
|
652
782
|
);
|
|
653
|
-
await
|
|
783
|
+
await this.waitForTxsPollingInterval();
|
|
654
784
|
availableTxs = await this.p2pClient.getPendingTxCount();
|
|
655
785
|
}
|
|
656
786
|
|
|
657
|
-
return { canStartBuilding: true, availableTxs };
|
|
787
|
+
return { canStartBuilding: true, availableTxs, minTxs };
|
|
658
788
|
}
|
|
659
789
|
|
|
660
790
|
/**
|
|
@@ -706,11 +836,28 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
706
836
|
|
|
707
837
|
collectedAttestationsCount = attestations.length;
|
|
708
838
|
|
|
839
|
+
// Trim attestations to minimum required to save L1 calldata gas
|
|
840
|
+
const localAddresses = this.validatorClient.getValidatorAddresses();
|
|
841
|
+
const trimmed = trimAttestations(
|
|
842
|
+
attestations,
|
|
843
|
+
numberOfRequiredAttestations,
|
|
844
|
+
this.attestorAddress,
|
|
845
|
+
localAddresses,
|
|
846
|
+
);
|
|
847
|
+
if (trimmed.length < attestations.length) {
|
|
848
|
+
this.log.debug(`Trimmed attestations from ${attestations.length} to ${trimmed.length} for L1 submission`);
|
|
849
|
+
}
|
|
850
|
+
|
|
709
851
|
// Rollup contract requires that the signatures are provided in the order of the committee
|
|
710
|
-
const sorted = orderAttestations(
|
|
852
|
+
const sorted = orderAttestations(trimmed, committee);
|
|
711
853
|
|
|
712
854
|
// Manipulate the attestations if we've been configured to do so
|
|
713
|
-
if (
|
|
855
|
+
if (
|
|
856
|
+
this.config.injectFakeAttestation ||
|
|
857
|
+
this.config.injectHighSValueAttestation ||
|
|
858
|
+
this.config.injectUnrecoverableSignatureAttestation ||
|
|
859
|
+
this.config.shuffleAttestationOrdering
|
|
860
|
+
) {
|
|
714
861
|
return this.manipulateAttestations(proposal.slotNumber, epoch, seed, committee, sorted);
|
|
715
862
|
}
|
|
716
863
|
|
|
@@ -739,7 +886,11 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
739
886
|
this.epochCache.computeProposerIndex(slotNumber, epoch, seed, BigInt(committee.length)),
|
|
740
887
|
);
|
|
741
888
|
|
|
742
|
-
if (
|
|
889
|
+
if (
|
|
890
|
+
this.config.injectFakeAttestation ||
|
|
891
|
+
this.config.injectHighSValueAttestation ||
|
|
892
|
+
this.config.injectUnrecoverableSignatureAttestation
|
|
893
|
+
) {
|
|
743
894
|
// Find non-empty attestations that are not from the proposer
|
|
744
895
|
const nonProposerIndices: number[] = [];
|
|
745
896
|
for (let i = 0; i < attestations.length; i++) {
|
|
@@ -749,8 +900,20 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
749
900
|
}
|
|
750
901
|
if (nonProposerIndices.length > 0) {
|
|
751
902
|
const targetIndex = nonProposerIndices[randomInt(nonProposerIndices.length)];
|
|
752
|
-
this.
|
|
753
|
-
|
|
903
|
+
if (this.config.injectHighSValueAttestation) {
|
|
904
|
+
this.log.warn(
|
|
905
|
+
`Injecting high-s value attestation in checkpoint for slot ${slotNumber} at index ${targetIndex}`,
|
|
906
|
+
);
|
|
907
|
+
unfreeze(attestations[targetIndex]).signature = flipSignature(attestations[targetIndex].signature);
|
|
908
|
+
} else if (this.config.injectUnrecoverableSignatureAttestation) {
|
|
909
|
+
this.log.warn(
|
|
910
|
+
`Injecting unrecoverable signature attestation in checkpoint for slot ${slotNumber} at index ${targetIndex}`,
|
|
911
|
+
);
|
|
912
|
+
unfreeze(attestations[targetIndex]).signature = generateUnrecoverableSignature();
|
|
913
|
+
} else {
|
|
914
|
+
this.log.warn(`Injecting fake attestation in checkpoint for slot ${slotNumber} at index ${targetIndex}`);
|
|
915
|
+
unfreeze(attestations[targetIndex]).signature = generateRecoverableSignature();
|
|
916
|
+
}
|
|
754
917
|
}
|
|
755
918
|
return new CommitteeAttestationsAndSigners(attestations);
|
|
756
919
|
}
|
|
@@ -759,11 +922,20 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
759
922
|
this.log.warn(`Shuffling attestation ordering in checkpoint for slot ${slotNumber} (proposer #${proposerIndex})`);
|
|
760
923
|
|
|
761
924
|
const shuffled = [...attestations];
|
|
762
|
-
|
|
763
|
-
|
|
764
|
-
|
|
765
|
-
|
|
766
|
-
|
|
925
|
+
|
|
926
|
+
// Find two non-proposer positions that both have non-empty signatures to swap.
|
|
927
|
+
// This ensures the bitmap doesn't change, so the MaliciousCommitteeAttestationsAndSigners
|
|
928
|
+
// signers array stays correctly aligned with L1's committee reconstruction.
|
|
929
|
+
const swappable: number[] = [];
|
|
930
|
+
for (let k = 0; k < shuffled.length; k++) {
|
|
931
|
+
if (!shuffled[k].signature.isEmpty() && k !== proposerIndex) {
|
|
932
|
+
swappable.push(k);
|
|
933
|
+
}
|
|
934
|
+
}
|
|
935
|
+
if (swappable.length >= 2) {
|
|
936
|
+
const [i, j] = [swappable[0], swappable[1]];
|
|
937
|
+
[shuffled[i], shuffled[j]] = [shuffled[j], shuffled[i]];
|
|
938
|
+
}
|
|
767
939
|
|
|
768
940
|
const signers = new CommitteeAttestationsAndSigners(attestations).getSigners();
|
|
769
941
|
return new MaliciousCommitteeAttestationsAndSigners(shuffled, signers);
|
|
@@ -779,7 +951,7 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
779
951
|
const failedTxData = failedTxs.map(fail => fail.tx);
|
|
780
952
|
const failedTxHashes = failedTxData.map(tx => tx.getTxHash());
|
|
781
953
|
this.log.verbose(`Dropping failed txs ${failedTxHashes.join(', ')}`);
|
|
782
|
-
await this.p2pClient.
|
|
954
|
+
await this.p2pClient.handleFailedExecution(failedTxHashes);
|
|
783
955
|
}
|
|
784
956
|
|
|
785
957
|
/**
|
|
@@ -806,27 +978,49 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
806
978
|
private async handleCheckpointEndAsFisherman(checkpoint: Checkpoint | undefined) {
|
|
807
979
|
// Perform L1 fee analysis before clearing requests
|
|
808
980
|
// The callback is invoked asynchronously after the next block is mined
|
|
809
|
-
const feeAnalysis = await this.publisher.analyzeL1Fees(this.
|
|
981
|
+
const feeAnalysis = await this.publisher.analyzeL1Fees(this.targetSlot, analysis =>
|
|
810
982
|
this.metrics.recordFishermanFeeAnalysis(analysis),
|
|
811
983
|
);
|
|
812
984
|
|
|
813
985
|
if (checkpoint) {
|
|
814
|
-
this.log.info(`Validation checkpoint building SUCCEEDED for slot ${this.
|
|
986
|
+
this.log.info(`Validation checkpoint building SUCCEEDED for slot ${this.targetSlot}`, {
|
|
815
987
|
...checkpoint.toCheckpointInfo(),
|
|
816
988
|
...checkpoint.getStats(),
|
|
817
989
|
feeAnalysisId: feeAnalysis?.id,
|
|
818
990
|
});
|
|
819
991
|
} else {
|
|
820
|
-
this.log.warn(`Validation block building FAILED for slot ${this.
|
|
821
|
-
slot: this.
|
|
992
|
+
this.log.warn(`Validation block building FAILED for slot ${this.targetSlot}`, {
|
|
993
|
+
slot: this.targetSlot,
|
|
822
994
|
feeAnalysisId: feeAnalysis?.id,
|
|
823
995
|
});
|
|
824
|
-
this.metrics.
|
|
996
|
+
this.metrics.recordCheckpointProposalFailed('block_build_failed');
|
|
825
997
|
}
|
|
826
998
|
|
|
827
999
|
this.publisher.clearPendingRequests();
|
|
828
1000
|
}
|
|
829
1001
|
|
|
1002
|
+
/**
|
|
1003
|
+
* Helper to handle HA double-signing errors. Returns true if the error was handled (caller should yield).
|
|
1004
|
+
*/
|
|
1005
|
+
private handleHASigningError(err: any, errorContext: string): boolean {
|
|
1006
|
+
if (err instanceof DutyAlreadySignedError) {
|
|
1007
|
+
this.log.info(`${errorContext} for slot ${this.targetSlot} already signed by another HA node, yielding`, {
|
|
1008
|
+
slot: this.targetSlot,
|
|
1009
|
+
signedByNode: err.signedByNode,
|
|
1010
|
+
});
|
|
1011
|
+
return true;
|
|
1012
|
+
}
|
|
1013
|
+
if (err instanceof SlashingProtectionError) {
|
|
1014
|
+
this.log.info(`${errorContext} for slot ${this.targetSlot} blocked by slashing protection, yielding`, {
|
|
1015
|
+
slot: this.targetSlot,
|
|
1016
|
+
existingMessageHash: err.existingMessageHash,
|
|
1017
|
+
attemptedMessageHash: err.attemptedMessageHash,
|
|
1018
|
+
});
|
|
1019
|
+
return true;
|
|
1020
|
+
}
|
|
1021
|
+
return false;
|
|
1022
|
+
}
|
|
1023
|
+
|
|
830
1024
|
/** Waits until a specific time within the current slot */
|
|
831
1025
|
@trackSpan('CheckpointProposalJob.waitUntilTimeInSlot')
|
|
832
1026
|
protected async waitUntilTimeInSlot(targetSecondsIntoSlot: number): Promise<void> {
|
|
@@ -835,6 +1029,11 @@ export class CheckpointProposalJob implements Traceable {
|
|
|
835
1029
|
await sleepUntil(new Date(targetTimestamp * 1000), this.dateProvider.nowAsDate());
|
|
836
1030
|
}
|
|
837
1031
|
|
|
1032
|
+
/** Waits the polling interval for transactions. Extracted for test overriding. */
|
|
1033
|
+
protected async waitForTxsPollingInterval(): Promise<void> {
|
|
1034
|
+
await sleep(TXS_POLLING_MS);
|
|
1035
|
+
}
|
|
1036
|
+
|
|
838
1037
|
private getSlotStartBuildTimestamp(): number {
|
|
839
1038
|
return getSlotStartBuildTimestamp(this.slot, this.l1Constants);
|
|
840
1039
|
}
|