@aztec/sequencer-client 0.0.1-commit.d3ec352c → 0.0.1-commit.e310a4c8

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 (96) hide show
  1. package/dest/client/sequencer-client.d.ts +12 -12
  2. package/dest/client/sequencer-client.d.ts.map +1 -1
  3. package/dest/client/sequencer-client.js +33 -26
  4. package/dest/config.d.ts +12 -5
  5. package/dest/config.d.ts.map +1 -1
  6. package/dest/config.js +73 -30
  7. package/dest/global_variable_builder/global_builder.d.ts +21 -12
  8. package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
  9. package/dest/global_variable_builder/global_builder.js +51 -41
  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/config.d.ts +7 -4
  14. package/dest/publisher/config.d.ts.map +1 -1
  15. package/dest/publisher/config.js +9 -3
  16. package/dest/publisher/sequencer-publisher-factory.d.ts +5 -4
  17. package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
  18. package/dest/publisher/sequencer-publisher-factory.js +1 -1
  19. package/dest/publisher/sequencer-publisher-metrics.d.ts +3 -3
  20. package/dest/publisher/sequencer-publisher-metrics.d.ts.map +1 -1
  21. package/dest/publisher/sequencer-publisher-metrics.js +23 -86
  22. package/dest/publisher/sequencer-publisher.d.ts +48 -41
  23. package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
  24. package/dest/publisher/sequencer-publisher.js +600 -129
  25. package/dest/sequencer/checkpoint_proposal_job.d.ts +96 -0
  26. package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -0
  27. package/dest/sequencer/checkpoint_proposal_job.js +1192 -0
  28. package/dest/sequencer/checkpoint_voter.d.ts +35 -0
  29. package/dest/sequencer/checkpoint_voter.d.ts.map +1 -0
  30. package/dest/sequencer/checkpoint_voter.js +109 -0
  31. package/dest/sequencer/config.d.ts +3 -2
  32. package/dest/sequencer/config.d.ts.map +1 -1
  33. package/dest/sequencer/events.d.ts +46 -0
  34. package/dest/sequencer/events.d.ts.map +1 -0
  35. package/dest/sequencer/events.js +1 -0
  36. package/dest/sequencer/index.d.ts +4 -2
  37. package/dest/sequencer/index.d.ts.map +1 -1
  38. package/dest/sequencer/index.js +3 -1
  39. package/dest/sequencer/metrics.d.ts +23 -3
  40. package/dest/sequencer/metrics.d.ts.map +1 -1
  41. package/dest/sequencer/metrics.js +143 -70
  42. package/dest/sequencer/sequencer.d.ts +107 -131
  43. package/dest/sequencer/sequencer.d.ts.map +1 -1
  44. package/dest/sequencer/sequencer.js +690 -602
  45. package/dest/sequencer/timetable.d.ts +54 -14
  46. package/dest/sequencer/timetable.d.ts.map +1 -1
  47. package/dest/sequencer/timetable.js +148 -59
  48. package/dest/sequencer/types.d.ts +3 -0
  49. package/dest/sequencer/types.d.ts.map +1 -0
  50. package/dest/sequencer/types.js +1 -0
  51. package/dest/sequencer/utils.d.ts +14 -8
  52. package/dest/sequencer/utils.d.ts.map +1 -1
  53. package/dest/sequencer/utils.js +7 -4
  54. package/dest/test/index.d.ts +4 -3
  55. package/dest/test/index.d.ts.map +1 -1
  56. package/dest/test/mock_checkpoint_builder.d.ts +95 -0
  57. package/dest/test/mock_checkpoint_builder.d.ts.map +1 -0
  58. package/dest/test/mock_checkpoint_builder.js +220 -0
  59. package/dest/test/utils.d.ts +53 -0
  60. package/dest/test/utils.d.ts.map +1 -0
  61. package/dest/test/utils.js +103 -0
  62. package/package.json +30 -28
  63. package/src/client/sequencer-client.ts +31 -42
  64. package/src/config.ts +78 -34
  65. package/src/global_variable_builder/global_builder.ts +63 -59
  66. package/src/index.ts +1 -7
  67. package/src/publisher/config.ts +12 -9
  68. package/src/publisher/sequencer-publisher-factory.ts +5 -4
  69. package/src/publisher/sequencer-publisher-metrics.ts +19 -71
  70. package/src/publisher/sequencer-publisher.ts +293 -170
  71. package/src/sequencer/README.md +531 -0
  72. package/src/sequencer/checkpoint_proposal_job.ts +874 -0
  73. package/src/sequencer/checkpoint_voter.ts +130 -0
  74. package/src/sequencer/config.ts +2 -1
  75. package/src/sequencer/events.ts +27 -0
  76. package/src/sequencer/index.ts +3 -1
  77. package/src/sequencer/metrics.ts +190 -78
  78. package/src/sequencer/sequencer.ts +430 -804
  79. package/src/sequencer/timetable.ts +173 -79
  80. package/src/sequencer/types.ts +6 -0
  81. package/src/sequencer/utils.ts +18 -9
  82. package/src/test/index.ts +3 -2
  83. package/src/test/mock_checkpoint_builder.ts +309 -0
  84. package/src/test/utils.ts +164 -0
  85. package/dest/sequencer/block_builder.d.ts +0 -28
  86. package/dest/sequencer/block_builder.d.ts.map +0 -1
  87. package/dest/sequencer/block_builder.js +0 -134
  88. package/dest/tx_validator/nullifier_cache.d.ts +0 -14
  89. package/dest/tx_validator/nullifier_cache.d.ts.map +0 -1
  90. package/dest/tx_validator/nullifier_cache.js +0 -24
  91. package/dest/tx_validator/tx_validator_factory.d.ts +0 -18
  92. package/dest/tx_validator/tx_validator_factory.d.ts.map +0 -1
  93. package/dest/tx_validator/tx_validator_factory.js +0 -53
  94. package/src/sequencer/block_builder.ts +0 -222
  95. package/src/tx_validator/nullifier_cache.ts +0 -30
  96. package/src/tx_validator/tx_validator_factory.ts +0 -133
@@ -1,9 +1,385 @@
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
- import { createBlobSinkClient } from '@aztec/blob-sink/client';
3
- import { FormattedViemError, MULTI_CALL_3_ADDRESS, Multicall3, RollupContract, WEI_CONST, formatViemError, tryExtractEvent } from '@aztec/ethereum';
375
+ import { MULTI_CALL_3_ADDRESS, Multicall3, RollupContract } from '@aztec/ethereum/contracts';
376
+ import { L1FeeAnalyzer } from '@aztec/ethereum/l1-fee-analysis';
377
+ import { MAX_L1_TX_LIMIT, WEI_CONST } from '@aztec/ethereum/l1-tx-utils';
378
+ import { FormattedViemError, formatViemError, mergeAbis, tryExtractEvent } from '@aztec/ethereum/utils';
4
379
  import { sumBigint } from '@aztec/foundation/bigint';
5
380
  import { toHex as toPaddedHex } from '@aztec/foundation/bigint-buffer';
6
- import { BlockNumber, CheckpointNumber, SlotNumber } from '@aztec/foundation/branded-types';
381
+ import { CheckpointNumber, SlotNumber } from '@aztec/foundation/branded-types';
382
+ import { pick } from '@aztec/foundation/collection';
7
383
  import { EthAddress } from '@aztec/foundation/eth-address';
8
384
  import { Signature } from '@aztec/foundation/eth-signature';
9
385
  import { createLogger } from '@aztec/foundation/log';
@@ -11,8 +387,8 @@ import { bufferToHex } from '@aztec/foundation/string';
11
387
  import { Timer } from '@aztec/foundation/timer';
12
388
  import { EmpireBaseAbi, ErrorsAbi, RollupAbi } from '@aztec/l1-artifacts';
13
389
  import { encodeSlashConsensusVotes } from '@aztec/slasher';
14
- import { CommitteeAttestation, CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
15
- import { getTelemetryClient } from '@aztec/telemetry-client';
390
+ import { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
391
+ import { getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
16
392
  import { encodeFunctionData, toHex } from 'viem';
17
393
  import { SequencerPublisherMetrics } from './sequencer-publisher-metrics.js';
18
394
  export const Actions = [
@@ -28,22 +404,40 @@ export const Actions = [
28
404
  ];
29
405
  // Sorting for actions such that invalidations go before proposals, and proposals go before votes
30
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');
31
408
  export class SequencerPublisher {
32
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
+ }
33
429
  interrupted;
34
430
  metrics;
35
431
  epochCache;
36
432
  governanceLog;
37
433
  slashingLog;
38
434
  lastActions;
435
+ isPayloadEmptyCache;
39
436
  log;
40
437
  ethereumSlotDuration;
41
- blobSinkClient;
438
+ blobClient;
42
439
  /** Address to use for simulations in fisherman mode (actual proposer's address) */ proposerAddressForSimulation;
43
- // @note - with blobs, the below estimate seems too large.
44
- // Total used for full block from int_l1_pub e2e test: 1m (of which 86k is 1x blob)
45
- // Total used for emptier block from above test: 429k (of which 84k is 1x blob)
46
- static PROPOSE_GAS_GUESS = 12_000_000n;
440
+ /** L1 fee analyzer for fisherman mode */ l1FeeAnalyzer;
47
441
  // A CALL to a cold address is 2700 gas
48
442
  static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
49
443
  // Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
@@ -53,23 +447,24 @@ export class SequencerPublisher {
53
447
  govProposerContract;
54
448
  slashingProposerContract;
55
449
  slashFactoryContract;
450
+ tracer;
56
451
  requests;
57
452
  constructor(config, deps){
58
453
  this.config = config;
59
- this.interrupted = false;
454
+ this.interrupted = (_initProto(this), false);
60
455
  this.governanceLog = createLogger('sequencer:publisher:governance');
61
456
  this.slashingLog = createLogger('sequencer:publisher:slashing');
62
457
  this.lastActions = {};
458
+ this.isPayloadEmptyCache = new Map();
63
459
  this.requests = [];
64
460
  this.log = deps.log ?? createLogger('sequencer:publisher');
65
461
  this.ethereumSlotDuration = BigInt(config.ethereumSlotDuration);
66
462
  this.epochCache = deps.epochCache;
67
463
  this.lastActions = deps.lastActions;
68
- this.blobSinkClient = deps.blobSinkClient ?? createBlobSinkClient(config, {
69
- logger: createLogger('sequencer:blob-sink:client')
70
- });
464
+ this.blobClient = deps.blobClient;
71
465
  const telemetry = deps.telemetry ?? getTelemetryClient();
72
466
  this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
467
+ this.tracer = telemetry.getTracer('SequencerPublisher');
73
468
  this.l1TxUtils = deps.l1TxUtils;
74
469
  this.rollupContract = deps.rollupContract;
75
470
  this.govProposerContract = deps.governanceProposerContract;
@@ -80,6 +475,10 @@ export class SequencerPublisher {
80
475
  this.slashingProposerContract = newSlashingProposer;
81
476
  });
82
477
  this.slashFactoryContract = deps.slashFactoryContract;
478
+ // Initialize L1 fee analyzer for fisherman mode
479
+ if (config.fishermanMode) {
480
+ this.l1FeeAnalyzer = new L1FeeAnalyzer(this.l1TxUtils.client, deps.dateProvider, createLogger('sequencer:publisher:fee-analyzer'));
481
+ }
83
482
  }
84
483
  getRollupContract() {
85
484
  return this.rollupContract;
@@ -88,6 +487,11 @@ export class SequencerPublisher {
88
487
  return this.l1TxUtils.getSenderAddress();
89
488
  }
90
489
  /**
490
+ * Gets the L1 fee analyzer instance (only available in fisherman mode)
491
+ */ getL1FeeAnalyzer() {
492
+ return this.l1FeeAnalyzer;
493
+ }
494
+ /**
91
495
  * Sets the proposer address to use for simulations in fisherman mode.
92
496
  * @param proposerAddress - The actual proposer's address to use for balance lookups in simulations
93
497
  */ setProposerAddressForSimulation(proposerAddress) {
@@ -109,6 +513,46 @@ export class SequencerPublisher {
109
513
  }
110
514
  }
111
515
  /**
516
+ * Analyzes L1 fees for the pending requests without sending them.
517
+ * This is used in fisherman mode to validate fee calculations.
518
+ * @param l2SlotNumber - The L2 slot number for this analysis
519
+ * @param onComplete - Optional callback to invoke when analysis completes (after block is mined)
520
+ * @returns The analysis result (incomplete until block mines), or undefined if no requests
521
+ */ async analyzeL1Fees(l2SlotNumber, onComplete) {
522
+ if (!this.l1FeeAnalyzer) {
523
+ this.log.warn('L1 fee analyzer not available (not in fisherman mode)');
524
+ return undefined;
525
+ }
526
+ const requestsToAnalyze = [
527
+ ...this.requests
528
+ ];
529
+ if (requestsToAnalyze.length === 0) {
530
+ this.log.debug('No requests to analyze for L1 fees');
531
+ return undefined;
532
+ }
533
+ // Extract blob config from requests (if any)
534
+ const blobConfigs = requestsToAnalyze.filter((request)=>request.blobConfig).map((request)=>request.blobConfig);
535
+ const blobConfig = blobConfigs[0];
536
+ // Get gas configs
537
+ const gasConfigs = requestsToAnalyze.filter((request)=>request.gasConfig).map((request)=>request.gasConfig);
538
+ const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
539
+ const gasLimit = gasLimits.length > 0 ? gasLimits.reduce((sum, g)=>sum + g, 0n) : 0n;
540
+ // Get the transaction requests
541
+ const l1Requests = requestsToAnalyze.map((r)=>r.request);
542
+ // Start the analysis
543
+ const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
544
+ this.log.info('Started L1 fee analysis', {
545
+ analysisId,
546
+ l2SlotNumber: l2SlotNumber.toString(),
547
+ requestCount: requestsToAnalyze.length,
548
+ hasBlobConfig: !!blobConfig,
549
+ gasLimit: gasLimit.toString(),
550
+ actions: requestsToAnalyze.map((r)=>r.action)
551
+ });
552
+ // Return the analysis result (will be incomplete until block mines)
553
+ return this.l1FeeAnalyzer.getAnalysis(analysisId);
554
+ }
555
+ /**
112
556
  * Sends all requests that are still valid.
113
557
  * @returns one of:
114
558
  * - A receipt and stats if the tx succeeded
@@ -119,7 +563,7 @@ export class SequencerPublisher {
119
563
  ...this.requests
120
564
  ];
121
565
  this.requests = [];
122
- if (this.interrupted) {
566
+ if (this.interrupted || requestsToProcess.length === 0) {
123
567
  return undefined;
124
568
  }
125
569
  const currentL2Slot = this.getCurrentL2Slot();
@@ -154,7 +598,16 @@ export class SequencerPublisher {
154
598
  const blobConfig = blobConfigs[0];
155
599
  // Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
156
600
  const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
157
- 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
+ }
158
611
  const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
159
612
  const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
160
613
  const txConfig = {
@@ -229,7 +682,7 @@ export class SequencerPublisher {
229
682
  'InvalidArchive'
230
683
  ];
231
684
  return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
232
- forcePendingCheckpointNumber: opts.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(opts.forcePendingBlockNumber) : undefined
685
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
233
686
  }).catch((err)=>{
234
687
  if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
235
688
  this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
@@ -257,12 +710,11 @@ export class SequencerPublisher {
257
710
  [],
258
711
  Signature.empty().toViemSignature(),
259
712
  `0x${'0'.repeat(64)}`,
260
- header.contentCommitment.blobsHash.toString(),
713
+ header.blobsHash.toString(),
261
714
  flags
262
715
  ];
263
716
  const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
264
- const optsForcePendingCheckpointNumber = opts?.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(opts.forcePendingBlockNumber) : undefined;
265
- const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(optsForcePendingCheckpointNumber);
717
+ const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
266
718
  let balance = 0n;
267
719
  if (this.config.fishermanMode) {
268
720
  // In fisherman mode, we can't know where the proposer is publishing from
@@ -289,34 +741,37 @@ export class SequencerPublisher {
289
741
  this.log.debug(`Simulated validateHeader`);
290
742
  }
291
743
  /**
292
- * Simulate making a call to invalidate a block with invalid attestations. Returns undefined if no need to invalidate.
293
- * @param block - The block to invalidate and the criteria for invalidation (as returned by the archiver)
294
- */ 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) {
295
747
  if (validationResult.valid) {
296
748
  return undefined;
297
749
  }
298
- const { reason, block } = validationResult;
299
- const blockNumber = block.blockNumber;
750
+ const { reason, checkpoint } = validationResult;
751
+ const checkpointNumber = checkpoint.checkpointNumber;
300
752
  const logData = {
301
- ...block,
753
+ ...checkpoint,
302
754
  reason
303
755
  };
304
- const currentBlockNumber = await this.rollupContract.getCheckpointNumber();
305
- if (currentBlockNumber < validationResult.block.blockNumber) {
306
- this.log.verbose(`Skipping block ${blockNumber} invalidation since it has already been removed from the pending chain`, {
307
- 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,
308
760
  ...logData
309
761
  });
310
762
  return undefined;
311
763
  }
312
- const request = this.buildInvalidateBlockRequest(validationResult);
313
- this.log.debug(`Simulating invalidate block ${blockNumber}`, {
764
+ const request = this.buildInvalidateCheckpointRequest(validationResult);
765
+ this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
314
766
  ...logData,
315
767
  request
316
768
  });
317
769
  try {
318
- const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, ErrorsAbi);
319
- 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`, {
320
775
  ...logData,
321
776
  request,
322
777
  gasUsed
@@ -324,88 +779,83 @@ export class SequencerPublisher {
324
779
  return {
325
780
  request,
326
781
  gasUsed,
327
- blockNumber,
328
- forcePendingBlockNumber: BlockNumber(blockNumber - 1),
782
+ checkpointNumber,
783
+ forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
329
784
  reason
330
785
  };
331
786
  } catch (err) {
332
787
  const viemError = formatViemError(err);
333
- // If the error is due to the block not being in the pending chain, and it was indeed removed by someone else,
334
- // we can safely ignore it and return undefined so we go ahead with block building.
335
- if (viemError.message?.includes('Rollup__BlockNotInPendingChain')) {
336
- 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`, {
337
792
  ...logData,
338
793
  request,
339
794
  error: viemError.message
340
795
  });
341
- const latestPendingBlockNumber = await this.rollupContract.getCheckpointNumber();
342
- if (latestPendingBlockNumber < blockNumber) {
343
- 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`, {
344
799
  ...logData
345
800
  });
346
801
  return undefined;
347
802
  } else {
348
- this.log.error(`Simulation for invalidate ${blockNumber} failed and it is still in pending chain`, viemError, logData);
349
- 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`, {
350
805
  cause: viemError
351
806
  });
352
807
  }
353
808
  }
354
- // Otherwise, throw. We cannot build the next block if we cannot invalidate the previous one.
355
- this.log.error(`Simulation for invalidate block ${blockNumber} failed`, viemError, logData);
356
- 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}`, {
357
812
  cause: viemError
358
813
  });
359
814
  }
360
815
  }
361
- buildInvalidateBlockRequest(validationResult) {
816
+ buildInvalidateCheckpointRequest(validationResult) {
362
817
  if (validationResult.valid) {
363
- throw new Error('Cannot invalidate a valid block');
818
+ throw new Error('Cannot invalidate a valid checkpoint');
364
819
  }
365
- const { block, committee, reason } = validationResult;
820
+ const { checkpoint, committee, reason } = validationResult;
366
821
  const logData = {
367
- ...block,
822
+ ...checkpoint,
368
823
  reason
369
824
  };
370
- this.log.debug(`Simulating invalidate block ${block.blockNumber}`, logData);
825
+ this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
371
826
  const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
372
827
  if (reason === 'invalid-attestation') {
373
- return this.rollupContract.buildInvalidateBadAttestationRequest(CheckpointNumber.fromBlockNumber(block.blockNumber), attestationsAndSigners, committee, validationResult.invalidIndex);
828
+ return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
374
829
  } else if (reason === 'insufficient-attestations') {
375
- return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(CheckpointNumber.fromBlockNumber(block.blockNumber), attestationsAndSigners, committee);
830
+ return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
376
831
  } else {
377
832
  const _ = reason;
378
833
  throw new Error(`Unknown reason for invalidation`);
379
834
  }
380
835
  }
381
- /**
382
- * @notice Will simulate `propose` to make sure that the block is valid for submission
383
- *
384
- * @dev Throws if unable to propose
385
- *
386
- * @param block - The block to propose
387
- * @param attestationData - The block's attestation data
388
- *
389
- */ async validateBlockForSubmission(block, attestationsAndSigners, attestationsAndSignersSignature, options) {
836
+ /** Simulates `propose` to make sure that the checkpoint is valid for submission */ async validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, options) {
390
837
  const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
838
+ // TODO(palla/mbps): This should not be needed, there's no flow where we propose with zero attestations. Or is there?
391
839
  // If we have no attestations, we still need to provide the empty attestations
392
840
  // so that the committee is recalculated correctly
393
- const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
394
- if (ignoreSignatures) {
395
- const { committee } = await this.epochCache.getCommittee(block.header.globalVariables.slotNumber);
396
- if (!committee) {
397
- this.log.warn(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
398
- throw new Error(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
399
- }
400
- attestationsAndSigners.attestations = committee.map((committeeMember)=>CommitteeAttestation.fromAddress(committeeMember));
401
- }
402
- const blobFields = block.getCheckpointBlobFields();
841
+ // const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
842
+ // if (ignoreSignatures) {
843
+ // const { committee } = await this.epochCache.getCommittee(block.header.globalVariables.slotNumber);
844
+ // if (!committee) {
845
+ // this.log.warn(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
846
+ // throw new Error(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
847
+ // }
848
+ // attestationsAndSigners.attestations = committee.map(committeeMember =>
849
+ // CommitteeAttestation.fromAddress(committeeMember),
850
+ // );
851
+ // }
852
+ const blobFields = checkpoint.toBlobFields();
403
853
  const blobs = getBlobsPerL1Block(blobFields);
404
854
  const blobInput = getPrefixedEthBlobCommitments(blobs);
405
855
  const args = [
406
856
  {
407
- header: block.getCheckpointHeader().toViem(),
408
- archive: toHex(block.archive.root.toBuffer()),
857
+ header: checkpoint.header.toViem(),
858
+ archive: toHex(checkpoint.archive.root.toBuffer()),
409
859
  oracleInput: {
410
860
  feeAssetPriceModifier: 0n
411
861
  }
@@ -432,9 +882,16 @@ export class SequencerPublisher {
432
882
  }
433
883
  const round = await base.computeRound(slotNumber);
434
884
  const roundInfo = await base.getRoundInfo(this.rollupContract.address, round);
885
+ if (roundInfo.quorumReached) {
886
+ return false;
887
+ }
435
888
  if (roundInfo.lastSignalSlot >= slotNumber) {
436
889
  return false;
437
890
  }
891
+ if (await this.isPayloadEmpty(payload)) {
892
+ this.log.warn(`Skipping vote cast for payload with empty code`);
893
+ return false;
894
+ }
438
895
  const cachedLastVote = this.lastActions[signalType];
439
896
  this.lastActions[signalType] = slotNumber;
440
897
  const action = signalType;
@@ -448,7 +905,10 @@ export class SequencerPublisher {
448
905
  try {
449
906
  await this.l1TxUtils.simulate(request, {
450
907
  time: timestamp
451
- }, [], ErrorsAbi);
908
+ }, [], mergeAbis([
909
+ request.abi ?? [],
910
+ ErrorsAbi
911
+ ]));
452
912
  this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
453
913
  request
454
914
  });
@@ -473,17 +933,27 @@ export class SequencerPublisher {
473
933
  payload: payload.toString()
474
934
  };
475
935
  if (!success) {
476
- this.log.error(`Signaling in [${action}] for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
936
+ this.log.error(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
477
937
  this.lastActions[signalType] = cachedLastVote;
478
938
  return false;
479
939
  } else {
480
- this.log.info(`Signaling in [${action}] for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
940
+ this.log.info(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
481
941
  return true;
482
942
  }
483
943
  }
484
944
  });
485
945
  return true;
486
946
  }
947
+ async isPayloadEmpty(payload) {
948
+ const key = payload.toString();
949
+ const cached = this.isPayloadEmptyCache.get(key);
950
+ if (cached) {
951
+ return cached;
952
+ }
953
+ const isEmpty = !await this.l1TxUtils.getCode(payload);
954
+ this.isPayloadEmptyCache.set(key, isEmpty);
955
+ return isEmpty;
956
+ }
487
957
  /**
488
958
  * Enqueues a governance castSignal transaction to cast a signal for a given slot number.
489
959
  * @param slotNumber - The slot number to cast a signal for.
@@ -579,19 +1049,13 @@ export class SequencerPublisher {
579
1049
  }
580
1050
  return true;
581
1051
  }
582
- /**
583
- * Proposes a L2 block on L1.
584
- *
585
- * @param block - L2 block to propose.
586
- * @returns True if the tx has been enqueued, throws otherwise. See #9315
587
- */ async enqueueProposeL2Block(block, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
588
- const checkpointHeader = block.getCheckpointHeader();
589
- const blobFields = block.getCheckpointBlobFields();
1052
+ /** Simulates and enqueues a proposal for a checkpoint on L1 */ async enqueueProposeCheckpoint(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
1053
+ const checkpointHeader = checkpoint.header;
1054
+ const blobFields = checkpoint.toBlobFields();
590
1055
  const blobs = getBlobsPerL1Block(blobFields);
591
1056
  const proposeTxArgs = {
592
1057
  header: checkpointHeader,
593
- archive: block.archive.root.toBuffer(),
594
- body: block.body.toBuffer(),
1058
+ archive: checkpoint.archive.root.toBuffer(),
595
1059
  blobs,
596
1060
  attestationsAndSigners,
597
1061
  attestationsAndSignersSignature
@@ -603,36 +1067,35 @@ export class SequencerPublisher {
603
1067
  // By simulation issue, I mean the fact that the block.timestamp is equal to the last block, not the next, which
604
1068
  // make time consistency checks break.
605
1069
  // TODO(palla): Check whether we're validating twice, once here and once within addProposeTx, since we call simulateProposeTx in both places.
606
- ts = await this.validateBlockForSubmission(block, attestationsAndSigners, attestationsAndSignersSignature, opts);
1070
+ ts = await this.validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts);
607
1071
  } catch (err) {
608
- this.log.error(`Block validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
609
- ...block.getStats(),
610
- slotNumber: block.header.globalVariables.slotNumber,
611
- forcePendingBlockNumber: opts.forcePendingBlockNumber
1072
+ this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
1073
+ ...checkpoint.getStats(),
1074
+ slotNumber: checkpoint.header.slotNumber,
1075
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
612
1076
  });
613
1077
  throw err;
614
1078
  }
615
- this.log.verbose(`Enqueuing block propose transaction`, {
616
- ...block.toBlockInfo(),
1079
+ this.log.verbose(`Enqueuing checkpoint propose transaction`, {
1080
+ ...checkpoint.toCheckpointInfo(),
617
1081
  ...opts
618
1082
  });
619
- await this.addProposeTx(block, proposeTxArgs, opts, ts);
620
- return true;
1083
+ await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
621
1084
  }
622
- enqueueInvalidateBlock(request, opts = {}) {
1085
+ enqueueInvalidateCheckpoint(request, opts = {}) {
623
1086
  if (!request) {
624
1087
  return;
625
1088
  }
626
1089
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
627
1090
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
628
- const { gasUsed, blockNumber } = request;
1091
+ const { gasUsed, checkpointNumber } = request;
629
1092
  const logData = {
630
1093
  gasUsed,
631
- blockNumber,
1094
+ checkpointNumber,
632
1095
  gasLimit,
633
1096
  opts
634
1097
  };
635
- this.log.verbose(`Enqueuing invalidate block request`, logData);
1098
+ this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
636
1099
  this.addRequest({
637
1100
  action: `invalidate-by-${request.reason}`,
638
1101
  request: request.request,
@@ -644,12 +1107,12 @@ export class SequencerPublisher {
644
1107
  checkSuccess: (_req, result)=>{
645
1108
  const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
646
1109
  if (!success) {
647
- this.log.warn(`Invalidate block ${request.blockNumber} failed`, {
1110
+ this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
648
1111
  ...result,
649
1112
  ...logData
650
1113
  });
651
1114
  } else {
652
- this.log.info(`Invalidate block ${request.blockNumber} succeeded`, {
1115
+ this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
653
1116
  ...result,
654
1117
  ...logData
655
1118
  });
@@ -672,27 +1135,37 @@ export class SequencerPublisher {
672
1135
  this.lastActions[action] = slotNumber;
673
1136
  this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
674
1137
  let gasUsed;
1138
+ const simulateAbi = mergeAbis([
1139
+ request.abi ?? [],
1140
+ ErrorsAbi
1141
+ ]);
675
1142
  try {
676
1143
  ({ gasUsed } = await this.l1TxUtils.simulate(request, {
677
1144
  time: timestamp
678
- }, [], ErrorsAbi)); // TODO(palla/slash): Check the timestamp logic
1145
+ }, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
679
1146
  this.log.verbose(`Simulation for ${action} succeeded`, {
680
1147
  ...logData,
681
1148
  request,
682
1149
  gasUsed
683
1150
  });
684
1151
  } catch (err) {
685
- const viemError = formatViemError(err);
1152
+ const viemError = formatViemError(err, simulateAbi);
686
1153
  this.log.error(`Simulation for ${action} at ${slotNumber} failed`, viemError, logData);
687
1154
  return false;
688
1155
  }
689
1156
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
690
1157
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
691
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
+ };
692
1165
  this.log.debug(`Enqueuing ${action}`, logData);
693
1166
  this.addRequest({
694
1167
  action,
695
- request,
1168
+ request: requestWithAbi,
696
1169
  gasConfig: {
697
1170
  gasLimit
698
1171
  },
@@ -799,8 +1272,7 @@ export class SequencerPublisher {
799
1272
  args
800
1273
  });
801
1274
  // override the pending checkpoint number if requested
802
- const optsForcePendingCheckpointNumber = options.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(options.forcePendingBlockNumber) : undefined;
803
- 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 ?? []);
804
1276
  const stateOverrides = [
805
1277
  {
806
1278
  address: this.rollupContract.address,
@@ -824,7 +1296,7 @@ export class SequencerPublisher {
824
1296
  const simulationResult = await this.l1TxUtils.simulate({
825
1297
  to: this.rollupContract.address,
826
1298
  data: rollupData,
827
- gas: SequencerPublisher.PROPOSE_GAS_GUESS,
1299
+ gas: MAX_L1_TX_LIMIT,
828
1300
  ...this.proposerAddressForSimulation && {
829
1301
  from: this.proposerAddressForSimulation.toString()
830
1302
  }
@@ -832,10 +1304,10 @@ export class SequencerPublisher {
832
1304
  // @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
833
1305
  time: timestamp + 1n,
834
1306
  // @note reth should have a 30m gas limit per block but throws errors that this tx is beyond limit so we increase here
835
- gasLimit: SequencerPublisher.PROPOSE_GAS_GUESS * 2n
1307
+ gasLimit: MAX_L1_TX_LIMIT * 2n
836
1308
  }, stateOverrides, RollupAbi, {
837
1309
  // @note fallback gas estimate to use if the node doesn't support simulation API
838
- fallbackGasEstimate: SequencerPublisher.PROPOSE_GAS_GUESS
1310
+ fallbackGasEstimate: MAX_L1_TX_LIMIT
839
1311
  }).catch((err)=>{
840
1312
  // In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
841
1313
  const viemError = formatViemError(err);
@@ -843,7 +1315,7 @@ export class SequencerPublisher {
843
1315
  this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
844
1316
  // Return a minimal simulation result with the fallback gas estimate
845
1317
  return {
846
- gasUsed: SequencerPublisher.PROPOSE_GAS_GUESS,
1318
+ gasUsed: MAX_L1_TX_LIMIT,
847
1319
  logs: []
848
1320
  };
849
1321
  }
@@ -855,24 +1327,25 @@ export class SequencerPublisher {
855
1327
  simulationResult
856
1328
  };
857
1329
  }
858
- async addProposeTx(block, encodedData, opts = {}, timestamp) {
1330
+ async addProposeTx(checkpoint, encodedData, opts = {}, timestamp) {
1331
+ const slot = checkpoint.header.slotNumber;
859
1332
  const timer = new Timer();
860
1333
  const kzg = Blob.getViemKzgInstance();
861
1334
  const { rollupData, simulationResult, blobEvaluationGas } = await this.prepareProposeTx(encodedData, timestamp, opts);
862
1335
  const startBlock = await this.l1TxUtils.getBlockNumber();
863
1336
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(simulationResult.gasUsed) * 64 / 63)) + blobEvaluationGas + SequencerPublisher.MULTICALL_OVERHEAD_GAS_GUESS);
864
- // Send the blobs to the blob sink preemptively. This helps in tests where the sequencer mistakingly thinks that the propose
865
- // tx fails but it does get mined. We make sure that the blobs are sent to the blob sink regardless of the tx outcome.
866
- void this.blobSinkClient.sendBlobsToBlobSink(encodedData.blobs).catch((_err)=>{
867
- this.log.error('Failed to send blobs to blob sink');
868
- });
1337
+ // Send the blobs to the blob client preemptively. This helps in tests where the sequencer mistakingly thinks that the propose
1338
+ // tx fails but it does get mined. We make sure that the blobs are sent to the blob client regardless of the tx outcome.
1339
+ void Promise.resolve().then(()=>this.blobClient.sendBlobsToFilestore(encodedData.blobs).catch((_err)=>{
1340
+ this.log.error('Failed to send blobs to blob client');
1341
+ }));
869
1342
  return this.addRequest({
870
1343
  action: 'propose',
871
1344
  request: {
872
1345
  to: this.rollupContract.address,
873
1346
  data: rollupData
874
1347
  },
875
- lastValidL2Slot: block.header.globalVariables.slotNumber,
1348
+ lastValidL2Slot: checkpoint.header.slotNumber,
876
1349
  gasConfig: {
877
1350
  ...opts,
878
1351
  gasLimit
@@ -900,25 +1373,23 @@ export class SequencerPublisher {
900
1373
  calldataGas,
901
1374
  calldataSize,
902
1375
  sender,
903
- ...block.getStats(),
1376
+ ...checkpoint.getStats(),
904
1377
  eventName: 'rollup-published-to-l1',
905
1378
  blobCount: encodedData.blobs.length,
906
1379
  inclusionBlocks
907
1380
  };
908
- this.log.info(`Published L2 block to L1 rollup contract`, {
1381
+ this.log.info(`Published checkpoint ${checkpoint.number} at slot ${slot} to rollup contract`, {
909
1382
  ...stats,
910
- ...block.getStats(),
911
- ...receipt
1383
+ ...checkpoint.getStats(),
1384
+ ...pick(receipt, 'transactionHash', 'blockHash')
912
1385
  });
913
1386
  this.metrics.recordProcessBlockTx(timer.ms(), publishStats);
914
1387
  return true;
915
1388
  } else {
916
1389
  this.metrics.recordFailedTx('process');
917
- this.log.error(`Rollup process tx failed: ${errorMsg ?? 'no error message'}`, undefined, {
918
- ...block.getStats(),
919
- receipt,
920
- txHash: receipt.transactionHash,
921
- slotNumber: block.header.globalVariables.slotNumber
1390
+ this.log.error(`Publishing checkpoint at slot ${slot} failed with ${errorMsg ?? 'no error message'}`, undefined, {
1391
+ ...checkpoint.getStats(),
1392
+ ...receipt
922
1393
  });
923
1394
  return false;
924
1395
  }