@aztec/sequencer-client 0.0.1-commit.6230efd → 0.0.1-commit.6a729f7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (72) hide show
  1. package/dest/client/sequencer-client.d.ts +4 -5
  2. package/dest/client/sequencer-client.d.ts.map +1 -1
  3. package/dest/client/sequencer-client.js +1 -1
  4. package/dest/config.d.ts +1 -1
  5. package/dest/config.d.ts.map +1 -1
  6. package/dest/config.js +6 -1
  7. package/dest/global_variable_builder/global_builder.d.ts +4 -4
  8. package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
  9. package/dest/global_variable_builder/global_builder.js +13 -13
  10. package/dest/index.d.ts +2 -3
  11. package/dest/index.d.ts.map +1 -1
  12. package/dest/index.js +1 -2
  13. package/dest/publisher/sequencer-publisher-metrics.d.ts +1 -1
  14. package/dest/publisher/sequencer-publisher-metrics.d.ts.map +1 -1
  15. package/dest/publisher/sequencer-publisher-metrics.js +23 -86
  16. package/dest/publisher/sequencer-publisher.d.ts +17 -17
  17. package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
  18. package/dest/publisher/sequencer-publisher.js +481 -67
  19. package/dest/sequencer/checkpoint_proposal_job.d.ts +34 -12
  20. package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -1
  21. package/dest/sequencer/checkpoint_proposal_job.js +605 -55
  22. package/dest/sequencer/checkpoint_voter.d.ts +3 -2
  23. package/dest/sequencer/checkpoint_voter.d.ts.map +1 -1
  24. package/dest/sequencer/checkpoint_voter.js +34 -10
  25. package/dest/sequencer/index.d.ts +1 -3
  26. package/dest/sequencer/index.d.ts.map +1 -1
  27. package/dest/sequencer/index.js +0 -2
  28. package/dest/sequencer/metrics.d.ts +4 -4
  29. package/dest/sequencer/metrics.d.ts.map +1 -1
  30. package/dest/sequencer/metrics.js +48 -129
  31. package/dest/sequencer/sequencer.d.ts +25 -15
  32. package/dest/sequencer/sequencer.d.ts.map +1 -1
  33. package/dest/sequencer/sequencer.js +486 -42
  34. package/dest/test/index.d.ts +2 -3
  35. package/dest/test/index.d.ts.map +1 -1
  36. package/dest/test/mock_checkpoint_builder.d.ts +23 -11
  37. package/dest/test/mock_checkpoint_builder.d.ts.map +1 -1
  38. package/dest/test/mock_checkpoint_builder.js +50 -9
  39. package/dest/test/utils.d.ts +13 -9
  40. package/dest/test/utils.d.ts.map +1 -1
  41. package/dest/test/utils.js +26 -17
  42. package/package.json +30 -28
  43. package/src/client/sequencer-client.ts +4 -5
  44. package/src/config.ts +5 -0
  45. package/src/global_variable_builder/global_builder.ts +13 -13
  46. package/src/index.ts +1 -9
  47. package/src/publisher/sequencer-publisher-metrics.ts +17 -69
  48. package/src/publisher/sequencer-publisher.ts +118 -91
  49. package/src/sequencer/checkpoint_proposal_job.ts +253 -85
  50. package/src/sequencer/checkpoint_voter.ts +32 -7
  51. package/src/sequencer/index.ts +0 -2
  52. package/src/sequencer/metrics.ts +48 -138
  53. package/src/sequencer/sequencer.ts +125 -42
  54. package/src/test/index.ts +1 -2
  55. package/src/test/mock_checkpoint_builder.ts +91 -29
  56. package/src/test/utils.ts +55 -28
  57. package/dest/sequencer/block_builder.d.ts +0 -26
  58. package/dest/sequencer/block_builder.d.ts.map +0 -1
  59. package/dest/sequencer/block_builder.js +0 -129
  60. package/dest/sequencer/checkpoint_builder.d.ts +0 -63
  61. package/dest/sequencer/checkpoint_builder.d.ts.map +0 -1
  62. package/dest/sequencer/checkpoint_builder.js +0 -131
  63. package/dest/tx_validator/nullifier_cache.d.ts +0 -14
  64. package/dest/tx_validator/nullifier_cache.d.ts.map +0 -1
  65. package/dest/tx_validator/nullifier_cache.js +0 -24
  66. package/dest/tx_validator/tx_validator_factory.d.ts +0 -18
  67. package/dest/tx_validator/tx_validator_factory.d.ts.map +0 -1
  68. package/dest/tx_validator/tx_validator_factory.js +0 -53
  69. package/src/sequencer/block_builder.ts +0 -217
  70. package/src/sequencer/checkpoint_builder.ts +0 -217
  71. package/src/tx_validator/nullifier_cache.ts +0 -30
  72. package/src/tx_validator/tx_validator_factory.ts +0 -133
@@ -1,11 +1,384 @@
1
+ function applyDecs2203RFactory() {
2
+ function createAddInitializerMethod(initializers, decoratorFinishedRef) {
3
+ return function addInitializer(initializer) {
4
+ assertNotFinished(decoratorFinishedRef, "addInitializer");
5
+ assertCallable(initializer, "An initializer");
6
+ initializers.push(initializer);
7
+ };
8
+ }
9
+ function memberDec(dec, name, desc, initializers, kind, isStatic, isPrivate, metadata, value) {
10
+ var kindStr;
11
+ switch(kind){
12
+ case 1:
13
+ kindStr = "accessor";
14
+ break;
15
+ case 2:
16
+ kindStr = "method";
17
+ break;
18
+ case 3:
19
+ kindStr = "getter";
20
+ break;
21
+ case 4:
22
+ kindStr = "setter";
23
+ break;
24
+ default:
25
+ kindStr = "field";
26
+ }
27
+ var ctx = {
28
+ kind: kindStr,
29
+ name: isPrivate ? "#" + name : name,
30
+ static: isStatic,
31
+ private: isPrivate,
32
+ metadata: metadata
33
+ };
34
+ var decoratorFinishedRef = {
35
+ v: false
36
+ };
37
+ ctx.addInitializer = createAddInitializerMethod(initializers, decoratorFinishedRef);
38
+ var get, set;
39
+ if (kind === 0) {
40
+ if (isPrivate) {
41
+ get = desc.get;
42
+ set = desc.set;
43
+ } else {
44
+ get = function() {
45
+ return this[name];
46
+ };
47
+ set = function(v) {
48
+ this[name] = v;
49
+ };
50
+ }
51
+ } else if (kind === 2) {
52
+ get = function() {
53
+ return desc.value;
54
+ };
55
+ } else {
56
+ if (kind === 1 || kind === 3) {
57
+ get = function() {
58
+ return desc.get.call(this);
59
+ };
60
+ }
61
+ if (kind === 1 || kind === 4) {
62
+ set = function(v) {
63
+ desc.set.call(this, v);
64
+ };
65
+ }
66
+ }
67
+ ctx.access = get && set ? {
68
+ get: get,
69
+ set: set
70
+ } : get ? {
71
+ get: get
72
+ } : {
73
+ set: set
74
+ };
75
+ try {
76
+ return dec(value, ctx);
77
+ } finally{
78
+ decoratorFinishedRef.v = true;
79
+ }
80
+ }
81
+ function assertNotFinished(decoratorFinishedRef, fnName) {
82
+ if (decoratorFinishedRef.v) {
83
+ throw new Error("attempted to call " + fnName + " after decoration was finished");
84
+ }
85
+ }
86
+ function assertCallable(fn, hint) {
87
+ if (typeof fn !== "function") {
88
+ throw new TypeError(hint + " must be a function");
89
+ }
90
+ }
91
+ function assertValidReturnValue(kind, value) {
92
+ var type = typeof value;
93
+ if (kind === 1) {
94
+ if (type !== "object" || value === null) {
95
+ throw new TypeError("accessor decorators must return an object with get, set, or init properties or void 0");
96
+ }
97
+ if (value.get !== undefined) {
98
+ assertCallable(value.get, "accessor.get");
99
+ }
100
+ if (value.set !== undefined) {
101
+ assertCallable(value.set, "accessor.set");
102
+ }
103
+ if (value.init !== undefined) {
104
+ assertCallable(value.init, "accessor.init");
105
+ }
106
+ } else if (type !== "function") {
107
+ var hint;
108
+ if (kind === 0) {
109
+ hint = "field";
110
+ } else if (kind === 10) {
111
+ hint = "class";
112
+ } else {
113
+ hint = "method";
114
+ }
115
+ throw new TypeError(hint + " decorators must return a function or void 0");
116
+ }
117
+ }
118
+ function applyMemberDec(ret, base, decInfo, name, kind, isStatic, isPrivate, initializers, metadata) {
119
+ var decs = decInfo[0];
120
+ var desc, init, value;
121
+ if (isPrivate) {
122
+ if (kind === 0 || kind === 1) {
123
+ desc = {
124
+ get: decInfo[3],
125
+ set: decInfo[4]
126
+ };
127
+ } else if (kind === 3) {
128
+ desc = {
129
+ get: decInfo[3]
130
+ };
131
+ } else if (kind === 4) {
132
+ desc = {
133
+ set: decInfo[3]
134
+ };
135
+ } else {
136
+ desc = {
137
+ value: decInfo[3]
138
+ };
139
+ }
140
+ } else if (kind !== 0) {
141
+ desc = Object.getOwnPropertyDescriptor(base, name);
142
+ }
143
+ if (kind === 1) {
144
+ value = {
145
+ get: desc.get,
146
+ set: desc.set
147
+ };
148
+ } else if (kind === 2) {
149
+ value = desc.value;
150
+ } else if (kind === 3) {
151
+ value = desc.get;
152
+ } else if (kind === 4) {
153
+ value = desc.set;
154
+ }
155
+ var newValue, get, set;
156
+ if (typeof decs === "function") {
157
+ newValue = memberDec(decs, name, desc, initializers, kind, isStatic, isPrivate, metadata, value);
158
+ if (newValue !== void 0) {
159
+ assertValidReturnValue(kind, newValue);
160
+ if (kind === 0) {
161
+ init = newValue;
162
+ } else if (kind === 1) {
163
+ init = newValue.init;
164
+ get = newValue.get || value.get;
165
+ set = newValue.set || value.set;
166
+ value = {
167
+ get: get,
168
+ set: set
169
+ };
170
+ } else {
171
+ value = newValue;
172
+ }
173
+ }
174
+ } else {
175
+ for(var i = decs.length - 1; i >= 0; i--){
176
+ var dec = decs[i];
177
+ newValue = memberDec(dec, name, desc, initializers, kind, isStatic, isPrivate, metadata, value);
178
+ if (newValue !== void 0) {
179
+ assertValidReturnValue(kind, newValue);
180
+ var newInit;
181
+ if (kind === 0) {
182
+ newInit = newValue;
183
+ } else if (kind === 1) {
184
+ newInit = newValue.init;
185
+ get = newValue.get || value.get;
186
+ set = newValue.set || value.set;
187
+ value = {
188
+ get: get,
189
+ set: set
190
+ };
191
+ } else {
192
+ value = newValue;
193
+ }
194
+ if (newInit !== void 0) {
195
+ if (init === void 0) {
196
+ init = newInit;
197
+ } else if (typeof init === "function") {
198
+ init = [
199
+ init,
200
+ newInit
201
+ ];
202
+ } else {
203
+ init.push(newInit);
204
+ }
205
+ }
206
+ }
207
+ }
208
+ }
209
+ if (kind === 0 || kind === 1) {
210
+ if (init === void 0) {
211
+ init = function(instance, init) {
212
+ return init;
213
+ };
214
+ } else if (typeof init !== "function") {
215
+ var ownInitializers = init;
216
+ init = function(instance, init) {
217
+ var value = init;
218
+ for(var i = 0; i < ownInitializers.length; i++){
219
+ value = ownInitializers[i].call(instance, value);
220
+ }
221
+ return value;
222
+ };
223
+ } else {
224
+ var originalInitializer = init;
225
+ init = function(instance, init) {
226
+ return originalInitializer.call(instance, init);
227
+ };
228
+ }
229
+ ret.push(init);
230
+ }
231
+ if (kind !== 0) {
232
+ if (kind === 1) {
233
+ desc.get = value.get;
234
+ desc.set = value.set;
235
+ } else if (kind === 2) {
236
+ desc.value = value;
237
+ } else if (kind === 3) {
238
+ desc.get = value;
239
+ } else if (kind === 4) {
240
+ desc.set = value;
241
+ }
242
+ if (isPrivate) {
243
+ if (kind === 1) {
244
+ ret.push(function(instance, args) {
245
+ return value.get.call(instance, args);
246
+ });
247
+ ret.push(function(instance, args) {
248
+ return value.set.call(instance, args);
249
+ });
250
+ } else if (kind === 2) {
251
+ ret.push(value);
252
+ } else {
253
+ ret.push(function(instance, args) {
254
+ return value.call(instance, args);
255
+ });
256
+ }
257
+ } else {
258
+ Object.defineProperty(base, name, desc);
259
+ }
260
+ }
261
+ }
262
+ function applyMemberDecs(Class, decInfos, metadata) {
263
+ var ret = [];
264
+ var protoInitializers;
265
+ var staticInitializers;
266
+ var existingProtoNonFields = new Map();
267
+ var existingStaticNonFields = new Map();
268
+ for(var i = 0; i < decInfos.length; i++){
269
+ var decInfo = decInfos[i];
270
+ if (!Array.isArray(decInfo)) continue;
271
+ var kind = decInfo[1];
272
+ var name = decInfo[2];
273
+ var isPrivate = decInfo.length > 3;
274
+ var isStatic = kind >= 5;
275
+ var base;
276
+ var initializers;
277
+ if (isStatic) {
278
+ base = Class;
279
+ kind = kind - 5;
280
+ staticInitializers = staticInitializers || [];
281
+ initializers = staticInitializers;
282
+ } else {
283
+ base = Class.prototype;
284
+ protoInitializers = protoInitializers || [];
285
+ initializers = protoInitializers;
286
+ }
287
+ if (kind !== 0 && !isPrivate) {
288
+ var existingNonFields = isStatic ? existingStaticNonFields : existingProtoNonFields;
289
+ var existingKind = existingNonFields.get(name) || 0;
290
+ if (existingKind === true || existingKind === 3 && kind !== 4 || existingKind === 4 && kind !== 3) {
291
+ throw new Error("Attempted to decorate a public method/accessor that has the same name as a previously decorated public method/accessor. This is not currently supported by the decorators plugin. Property name was: " + name);
292
+ } else if (!existingKind && kind > 2) {
293
+ existingNonFields.set(name, kind);
294
+ } else {
295
+ existingNonFields.set(name, true);
296
+ }
297
+ }
298
+ applyMemberDec(ret, base, decInfo, name, kind, isStatic, isPrivate, initializers, metadata);
299
+ }
300
+ pushInitializers(ret, protoInitializers);
301
+ pushInitializers(ret, staticInitializers);
302
+ return ret;
303
+ }
304
+ function pushInitializers(ret, initializers) {
305
+ if (initializers) {
306
+ ret.push(function(instance) {
307
+ for(var i = 0; i < initializers.length; i++){
308
+ initializers[i].call(instance);
309
+ }
310
+ return instance;
311
+ });
312
+ }
313
+ }
314
+ function applyClassDecs(targetClass, classDecs, metadata) {
315
+ if (classDecs.length > 0) {
316
+ var initializers = [];
317
+ var newClass = targetClass;
318
+ var name = targetClass.name;
319
+ for(var i = classDecs.length - 1; i >= 0; i--){
320
+ var decoratorFinishedRef = {
321
+ v: false
322
+ };
323
+ try {
324
+ var nextNewClass = classDecs[i](newClass, {
325
+ kind: "class",
326
+ name: name,
327
+ addInitializer: createAddInitializerMethod(initializers, decoratorFinishedRef),
328
+ metadata
329
+ });
330
+ } finally{
331
+ decoratorFinishedRef.v = true;
332
+ }
333
+ if (nextNewClass !== undefined) {
334
+ assertValidReturnValue(10, nextNewClass);
335
+ newClass = nextNewClass;
336
+ }
337
+ }
338
+ return [
339
+ defineMetadata(newClass, metadata),
340
+ function() {
341
+ for(var i = 0; i < initializers.length; i++){
342
+ initializers[i].call(newClass);
343
+ }
344
+ }
345
+ ];
346
+ }
347
+ }
348
+ function defineMetadata(Class, metadata) {
349
+ return Object.defineProperty(Class, Symbol.metadata || Symbol.for("Symbol.metadata"), {
350
+ configurable: true,
351
+ enumerable: true,
352
+ value: metadata
353
+ });
354
+ }
355
+ return function applyDecs2203R(targetClass, memberDecs, classDecs, parentClass) {
356
+ if (parentClass !== void 0) {
357
+ var parentMetadata = parentClass[Symbol.metadata || Symbol.for("Symbol.metadata")];
358
+ }
359
+ var metadata = Object.create(parentMetadata === void 0 ? null : parentMetadata);
360
+ var e = applyMemberDecs(targetClass, memberDecs, metadata);
361
+ if (!classDecs.length) defineMetadata(targetClass, metadata);
362
+ return {
363
+ e: e,
364
+ get c () {
365
+ return applyClassDecs(targetClass, classDecs, metadata);
366
+ }
367
+ };
368
+ };
369
+ }
370
+ function _apply_decs_2203_r(targetClass, memberDecs, classDecs, parentClass) {
371
+ return (_apply_decs_2203_r = applyDecs2203RFactory())(targetClass, memberDecs, classDecs, parentClass);
372
+ }
373
+ var _dec, _dec1, _dec2, _initProto;
1
374
  import { Blob, getBlobsPerL1Block, getPrefixedEthBlobCommitments } from '@aztec/blob-lib';
2
375
  import { MULTI_CALL_3_ADDRESS, Multicall3, RollupContract } from '@aztec/ethereum/contracts';
3
376
  import { L1FeeAnalyzer } from '@aztec/ethereum/l1-fee-analysis';
4
- import { WEI_CONST } from '@aztec/ethereum/l1-tx-utils';
5
- import { FormattedViemError, formatViemError, tryExtractEvent } from '@aztec/ethereum/utils';
377
+ import { MAX_L1_TX_LIMIT, WEI_CONST } from '@aztec/ethereum/l1-tx-utils';
378
+ import { FormattedViemError, formatViemError, mergeAbis, tryExtractEvent } from '@aztec/ethereum/utils';
6
379
  import { sumBigint } from '@aztec/foundation/bigint';
7
380
  import { toHex as toPaddedHex } from '@aztec/foundation/bigint-buffer';
8
- import { BlockNumber, CheckpointNumber, SlotNumber } from '@aztec/foundation/branded-types';
381
+ import { CheckpointNumber, SlotNumber } from '@aztec/foundation/branded-types';
9
382
  import { pick } from '@aztec/foundation/collection';
10
383
  import { EthAddress } from '@aztec/foundation/eth-address';
11
384
  import { Signature } from '@aztec/foundation/eth-signature';
@@ -15,7 +388,7 @@ import { Timer } from '@aztec/foundation/timer';
15
388
  import { EmpireBaseAbi, ErrorsAbi, RollupAbi } from '@aztec/l1-artifacts';
16
389
  import { encodeSlashConsensusVotes } from '@aztec/slasher';
17
390
  import { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
18
- import { getTelemetryClient } from '@aztec/telemetry-client';
391
+ import { getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
19
392
  import { encodeFunctionData, toHex } from 'viem';
20
393
  import { SequencerPublisherMetrics } from './sequencer-publisher-metrics.js';
21
394
  export const Actions = [
@@ -31,8 +404,28 @@ export const Actions = [
31
404
  ];
32
405
  // Sorting for actions such that invalidations go before proposals, and proposals go before votes
33
406
  export const compareActions = (a, b)=>Actions.indexOf(a) - Actions.indexOf(b);
407
+ _dec = trackSpan('SequencerPublisher.sendRequests'), _dec1 = trackSpan('SequencerPublisher.validateBlockHeader'), _dec2 = trackSpan('SequencerPublisher.validateCheckpointForSubmission');
34
408
  export class SequencerPublisher {
35
409
  config;
410
+ static{
411
+ ({ e: [_initProto] } = _apply_decs_2203_r(this, [
412
+ [
413
+ _dec,
414
+ 2,
415
+ "sendRequests"
416
+ ],
417
+ [
418
+ _dec1,
419
+ 2,
420
+ "validateBlockHeader"
421
+ ],
422
+ [
423
+ _dec2,
424
+ 2,
425
+ "validateCheckpointForSubmission"
426
+ ]
427
+ ], []));
428
+ }
36
429
  interrupted;
37
430
  metrics;
38
431
  epochCache;
@@ -45,10 +438,6 @@ export class SequencerPublisher {
45
438
  blobClient;
46
439
  /** Address to use for simulations in fisherman mode (actual proposer's address) */ proposerAddressForSimulation;
47
440
  /** L1 fee analyzer for fisherman mode */ l1FeeAnalyzer;
48
- // @note - with blobs, the below estimate seems too large.
49
- // Total used for full block from int_l1_pub e2e test: 1m (of which 86k is 1x blob)
50
- // Total used for emptier block from above test: 429k (of which 84k is 1x blob)
51
- static PROPOSE_GAS_GUESS = 12_000_000n;
52
441
  // A CALL to a cold address is 2700 gas
53
442
  static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
54
443
  // Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
@@ -58,10 +447,11 @@ export class SequencerPublisher {
58
447
  govProposerContract;
59
448
  slashingProposerContract;
60
449
  slashFactoryContract;
450
+ tracer;
61
451
  requests;
62
452
  constructor(config, deps){
63
453
  this.config = config;
64
- this.interrupted = false;
454
+ this.interrupted = (_initProto(this), false);
65
455
  this.governanceLog = createLogger('sequencer:publisher:governance');
66
456
  this.slashingLog = createLogger('sequencer:publisher:slashing');
67
457
  this.lastActions = {};
@@ -74,6 +464,7 @@ export class SequencerPublisher {
74
464
  this.blobClient = deps.blobClient;
75
465
  const telemetry = deps.telemetry ?? getTelemetryClient();
76
466
  this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
467
+ this.tracer = telemetry.getTracer('SequencerPublisher');
77
468
  this.l1TxUtils = deps.l1TxUtils;
78
469
  this.rollupContract = deps.rollupContract;
79
470
  this.govProposerContract = deps.governanceProposerContract;
@@ -149,7 +540,7 @@ export class SequencerPublisher {
149
540
  // Get the transaction requests
150
541
  const l1Requests = requestsToAnalyze.map((r)=>r.request);
151
542
  // Start the analysis
152
- const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : SequencerPublisher.PROPOSE_GAS_GUESS, l1Requests, blobConfig, onComplete);
543
+ const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
153
544
  this.log.info('Started L1 fee analysis', {
154
545
  analysisId,
155
546
  l2SlotNumber: l2SlotNumber.toString(),
@@ -207,7 +598,16 @@ export class SequencerPublisher {
207
598
  const blobConfig = blobConfigs[0];
208
599
  // Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
209
600
  const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
210
- const gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
601
+ let gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
602
+ // Cap at L1 block gas limit so the node accepts the tx ("gas limit too high" otherwise).
603
+ const maxGas = MAX_L1_TX_LIMIT;
604
+ if (gasLimit !== undefined && gasLimit > maxGas) {
605
+ this.log.debug('Capping bundled tx gas limit to L1 max', {
606
+ requested: gasLimit,
607
+ capped: maxGas
608
+ });
609
+ gasLimit = maxGas;
610
+ }
211
611
  const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
212
612
  const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
213
613
  const txConfig = {
@@ -282,7 +682,7 @@ export class SequencerPublisher {
282
682
  'InvalidArchive'
283
683
  ];
284
684
  return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
285
- forcePendingCheckpointNumber: opts.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(opts.forcePendingBlockNumber) : undefined
685
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
286
686
  }).catch((err)=>{
287
687
  if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
288
688
  this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
@@ -310,12 +710,11 @@ export class SequencerPublisher {
310
710
  [],
311
711
  Signature.empty().toViemSignature(),
312
712
  `0x${'0'.repeat(64)}`,
313
- header.contentCommitment.blobsHash.toString(),
713
+ header.blobsHash.toString(),
314
714
  flags
315
715
  ];
316
716
  const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
317
- const optsForcePendingCheckpointNumber = opts?.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(opts.forcePendingBlockNumber) : undefined;
318
- const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(optsForcePendingCheckpointNumber);
717
+ const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
319
718
  let balance = 0n;
320
719
  if (this.config.fishermanMode) {
321
720
  // In fisherman mode, we can't know where the proposer is publishing from
@@ -342,34 +741,37 @@ export class SequencerPublisher {
342
741
  this.log.debug(`Simulated validateHeader`);
343
742
  }
344
743
  /**
345
- * Simulate making a call to invalidate a block with invalid attestations. Returns undefined if no need to invalidate.
346
- * @param block - The block to invalidate and the criteria for invalidation (as returned by the archiver)
347
- */ async simulateInvalidateBlock(validationResult) {
744
+ * Simulate making a call to invalidate a checkpoint with invalid attestations. Returns undefined if no need to invalidate.
745
+ * @param validationResult - The validation result indicating which checkpoint to invalidate (as returned by the archiver)
746
+ */ async simulateInvalidateCheckpoint(validationResult) {
348
747
  if (validationResult.valid) {
349
748
  return undefined;
350
749
  }
351
- const { reason, block } = validationResult;
352
- const blockNumber = block.blockNumber;
750
+ const { reason, checkpoint } = validationResult;
751
+ const checkpointNumber = checkpoint.checkpointNumber;
353
752
  const logData = {
354
- ...block,
753
+ ...checkpoint,
355
754
  reason
356
755
  };
357
- const currentBlockNumber = await this.rollupContract.getCheckpointNumber();
358
- if (currentBlockNumber < validationResult.block.blockNumber) {
359
- this.log.verbose(`Skipping block ${blockNumber} invalidation since it has already been removed from the pending chain`, {
360
- currentBlockNumber,
756
+ const currentCheckpointNumber = await this.rollupContract.getCheckpointNumber();
757
+ if (currentCheckpointNumber < checkpointNumber) {
758
+ this.log.verbose(`Skipping checkpoint ${checkpointNumber} invalidation since it has already been removed from the pending chain`, {
759
+ currentCheckpointNumber,
361
760
  ...logData
362
761
  });
363
762
  return undefined;
364
763
  }
365
- const request = this.buildInvalidateBlockRequest(validationResult);
366
- this.log.debug(`Simulating invalidate block ${blockNumber}`, {
764
+ const request = this.buildInvalidateCheckpointRequest(validationResult);
765
+ this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
367
766
  ...logData,
368
767
  request
369
768
  });
370
769
  try {
371
- const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, ErrorsAbi);
372
- this.log.verbose(`Simulation for invalidate block ${blockNumber} succeeded`, {
770
+ const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, mergeAbis([
771
+ request.abi ?? [],
772
+ ErrorsAbi
773
+ ]));
774
+ this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} succeeded`, {
373
775
  ...logData,
374
776
  request,
375
777
  gasUsed
@@ -377,55 +779,55 @@ export class SequencerPublisher {
377
779
  return {
378
780
  request,
379
781
  gasUsed,
380
- blockNumber,
381
- forcePendingBlockNumber: BlockNumber(blockNumber - 1),
782
+ checkpointNumber,
783
+ forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
382
784
  reason
383
785
  };
384
786
  } catch (err) {
385
787
  const viemError = formatViemError(err);
386
- // If the error is due to the block not being in the pending chain, and it was indeed removed by someone else,
387
- // we can safely ignore it and return undefined so we go ahead with block building.
388
- if (viemError.message?.includes('Rollup__BlockNotInPendingChain')) {
389
- this.log.verbose(`Simulation for invalidate block ${blockNumber} failed due to block not being in pending chain`, {
788
+ // If the error is due to the checkpoint not being in the pending chain, and it was indeed removed by someone else,
789
+ // we can safely ignore it and return undefined so we go ahead with checkpoint building.
790
+ if (viemError.message?.includes('Rollup__CheckpointNotInPendingChain')) {
791
+ this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} failed due to checkpoint not being in pending chain`, {
390
792
  ...logData,
391
793
  request,
392
794
  error: viemError.message
393
795
  });
394
- const latestPendingBlockNumber = await this.rollupContract.getCheckpointNumber();
395
- if (latestPendingBlockNumber < blockNumber) {
396
- this.log.verbose(`Block number ${blockNumber} has already been invalidated`, {
796
+ const latestPendingCheckpointNumber = await this.rollupContract.getCheckpointNumber();
797
+ if (latestPendingCheckpointNumber < checkpointNumber) {
798
+ this.log.verbose(`Checkpoint ${checkpointNumber} has already been invalidated`, {
397
799
  ...logData
398
800
  });
399
801
  return undefined;
400
802
  } else {
401
- this.log.error(`Simulation for invalidate ${blockNumber} failed and it is still in pending chain`, viemError, logData);
402
- throw new Error(`Failed to simulate invalidate block ${blockNumber} while it is still in pending chain`, {
803
+ this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed and it is still in pending chain`, viemError, logData);
804
+ throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber} while it is still in pending chain`, {
403
805
  cause: viemError
404
806
  });
405
807
  }
406
808
  }
407
- // Otherwise, throw. We cannot build the next block if we cannot invalidate the previous one.
408
- this.log.error(`Simulation for invalidate block ${blockNumber} failed`, viemError, logData);
409
- throw new Error(`Failed to simulate invalidate block ${blockNumber}`, {
809
+ // Otherwise, throw. We cannot build the next checkpoint if we cannot invalidate the previous one.
810
+ this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed`, viemError, logData);
811
+ throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber}`, {
410
812
  cause: viemError
411
813
  });
412
814
  }
413
815
  }
414
- buildInvalidateBlockRequest(validationResult) {
816
+ buildInvalidateCheckpointRequest(validationResult) {
415
817
  if (validationResult.valid) {
416
- throw new Error('Cannot invalidate a valid block');
818
+ throw new Error('Cannot invalidate a valid checkpoint');
417
819
  }
418
- const { block, committee, reason } = validationResult;
820
+ const { checkpoint, committee, reason } = validationResult;
419
821
  const logData = {
420
- ...block,
822
+ ...checkpoint,
421
823
  reason
422
824
  };
423
- this.log.debug(`Simulating invalidate block ${block.blockNumber}`, logData);
825
+ this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
424
826
  const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
425
827
  if (reason === 'invalid-attestation') {
426
- return this.rollupContract.buildInvalidateBadAttestationRequest(CheckpointNumber.fromBlockNumber(block.blockNumber), attestationsAndSigners, committee, validationResult.invalidIndex);
828
+ return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
427
829
  } else if (reason === 'insufficient-attestations') {
428
- return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(CheckpointNumber.fromBlockNumber(block.blockNumber), attestationsAndSigners, committee);
830
+ return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
429
831
  } else {
430
832
  const _ = reason;
431
833
  throw new Error(`Unknown reason for invalidation`);
@@ -503,7 +905,10 @@ export class SequencerPublisher {
503
905
  try {
504
906
  await this.l1TxUtils.simulate(request, {
505
907
  time: timestamp
506
- }, [], ErrorsAbi);
908
+ }, [], mergeAbis([
909
+ request.abi ?? [],
910
+ ErrorsAbi
911
+ ]));
507
912
  this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
508
913
  request
509
914
  });
@@ -667,7 +1072,7 @@ export class SequencerPublisher {
667
1072
  this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
668
1073
  ...checkpoint.getStats(),
669
1074
  slotNumber: checkpoint.header.slotNumber,
670
- forcePendingBlockNumber: opts.forcePendingBlockNumber
1075
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
671
1076
  });
672
1077
  throw err;
673
1078
  }
@@ -677,20 +1082,20 @@ export class SequencerPublisher {
677
1082
  });
678
1083
  await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
679
1084
  }
680
- enqueueInvalidateBlock(request, opts = {}) {
1085
+ enqueueInvalidateCheckpoint(request, opts = {}) {
681
1086
  if (!request) {
682
1087
  return;
683
1088
  }
684
1089
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
685
1090
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
686
- const { gasUsed, blockNumber } = request;
1091
+ const { gasUsed, checkpointNumber } = request;
687
1092
  const logData = {
688
1093
  gasUsed,
689
- blockNumber,
1094
+ checkpointNumber,
690
1095
  gasLimit,
691
1096
  opts
692
1097
  };
693
- this.log.verbose(`Enqueuing invalidate block request`, logData);
1098
+ this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
694
1099
  this.addRequest({
695
1100
  action: `invalidate-by-${request.reason}`,
696
1101
  request: request.request,
@@ -702,12 +1107,12 @@ export class SequencerPublisher {
702
1107
  checkSuccess: (_req, result)=>{
703
1108
  const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
704
1109
  if (!success) {
705
- this.log.warn(`Invalidate block ${request.blockNumber} failed`, {
1110
+ this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
706
1111
  ...result,
707
1112
  ...logData
708
1113
  });
709
1114
  } else {
710
- this.log.info(`Invalidate block ${request.blockNumber} succeeded`, {
1115
+ this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
711
1116
  ...result,
712
1117
  ...logData
713
1118
  });
@@ -730,27 +1135,37 @@ export class SequencerPublisher {
730
1135
  this.lastActions[action] = slotNumber;
731
1136
  this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
732
1137
  let gasUsed;
1138
+ const simulateAbi = mergeAbis([
1139
+ request.abi ?? [],
1140
+ ErrorsAbi
1141
+ ]);
733
1142
  try {
734
1143
  ({ gasUsed } = await this.l1TxUtils.simulate(request, {
735
1144
  time: timestamp
736
- }, [], ErrorsAbi)); // TODO(palla/slash): Check the timestamp logic
1145
+ }, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
737
1146
  this.log.verbose(`Simulation for ${action} succeeded`, {
738
1147
  ...logData,
739
1148
  request,
740
1149
  gasUsed
741
1150
  });
742
1151
  } catch (err) {
743
- const viemError = formatViemError(err);
1152
+ const viemError = formatViemError(err, simulateAbi);
744
1153
  this.log.error(`Simulation for ${action} at ${slotNumber} failed`, viemError, logData);
745
1154
  return false;
746
1155
  }
747
1156
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
748
1157
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
749
1158
  logData.gasLimit = gasLimit;
1159
+ // Store the ABI used for simulation on the request so Multicall3.forward can decode errors
1160
+ // when the tx is sent and a revert is diagnosed via simulation.
1161
+ const requestWithAbi = {
1162
+ ...request,
1163
+ abi: simulateAbi
1164
+ };
750
1165
  this.log.debug(`Enqueuing ${action}`, logData);
751
1166
  this.addRequest({
752
1167
  action,
753
- request,
1168
+ request: requestWithAbi,
754
1169
  gasConfig: {
755
1170
  gasLimit
756
1171
  },
@@ -857,8 +1272,7 @@ export class SequencerPublisher {
857
1272
  args
858
1273
  });
859
1274
  // override the pending checkpoint number if requested
860
- const optsForcePendingCheckpointNumber = options.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(options.forcePendingBlockNumber) : undefined;
861
- const forcePendingCheckpointNumberStateDiff = (optsForcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(optsForcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
1275
+ const forcePendingCheckpointNumberStateDiff = (options.forcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
862
1276
  const stateOverrides = [
863
1277
  {
864
1278
  address: this.rollupContract.address,
@@ -882,7 +1296,7 @@ export class SequencerPublisher {
882
1296
  const simulationResult = await this.l1TxUtils.simulate({
883
1297
  to: this.rollupContract.address,
884
1298
  data: rollupData,
885
- gas: SequencerPublisher.PROPOSE_GAS_GUESS,
1299
+ gas: MAX_L1_TX_LIMIT,
886
1300
  ...this.proposerAddressForSimulation && {
887
1301
  from: this.proposerAddressForSimulation.toString()
888
1302
  }
@@ -890,10 +1304,10 @@ export class SequencerPublisher {
890
1304
  // @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
891
1305
  time: timestamp + 1n,
892
1306
  // @note reth should have a 30m gas limit per block but throws errors that this tx is beyond limit so we increase here
893
- gasLimit: SequencerPublisher.PROPOSE_GAS_GUESS * 2n
1307
+ gasLimit: MAX_L1_TX_LIMIT * 2n
894
1308
  }, stateOverrides, RollupAbi, {
895
1309
  // @note fallback gas estimate to use if the node doesn't support simulation API
896
- fallbackGasEstimate: SequencerPublisher.PROPOSE_GAS_GUESS
1310
+ fallbackGasEstimate: MAX_L1_TX_LIMIT
897
1311
  }).catch((err)=>{
898
1312
  // In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
899
1313
  const viemError = formatViemError(err);
@@ -901,7 +1315,7 @@ export class SequencerPublisher {
901
1315
  this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
902
1316
  // Return a minimal simulation result with the fallback gas estimate
903
1317
  return {
904
- gasUsed: SequencerPublisher.PROPOSE_GAS_GUESS,
1318
+ gasUsed: MAX_L1_TX_LIMIT,
905
1319
  logs: []
906
1320
  };
907
1321
  }