@aztec/sequencer-client 0.0.1-commit.21caa21 → 0.0.1-commit.21ecf947b

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 -6
  5. package/dest/config.d.ts.map +1 -1
  6. package/dest/config.js +71 -32
  7. package/dest/global_variable_builder/global_builder.d.ts +22 -13
  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 +56 -42
  23. package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
  24. package/dest/publisher/sequencer-publisher.js +615 -133
  25. package/dest/sequencer/checkpoint_proposal_job.d.ts +102 -0
  26. package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -0
  27. package/dest/sequencer/checkpoint_proposal_job.js +1193 -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 +27 -3
  40. package/dest/sequencer/metrics.d.ts.map +1 -1
  41. package/dest/sequencer/metrics.js +188 -66
  42. package/dest/sequencer/sequencer.d.ts +109 -131
  43. package/dest/sequencer/sequencer.d.ts.map +1 -1
  44. package/dest/sequencer/sequencer.js +700 -606
  45. package/dest/sequencer/timetable.d.ts +51 -14
  46. package/dest/sequencer/timetable.d.ts.map +1 -1
  47. package/dest/sequencer/timetable.js +145 -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 +97 -0
  57. package/dest/test/mock_checkpoint_builder.d.ts.map +1 -0
  58. package/dest/test/mock_checkpoint_builder.js +222 -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 +104 -0
  62. package/package.json +32 -30
  63. package/src/client/sequencer-client.ts +31 -42
  64. package/src/config.ts +78 -36
  65. package/src/global_variable_builder/global_builder.ts +65 -61
  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 +327 -166
  71. package/src/sequencer/README.md +531 -0
  72. package/src/sequencer/checkpoint_proposal_job.ts +882 -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 +236 -76
  78. package/src/sequencer/sequencer.ts +445 -813
  79. package/src/sequencer/timetable.ts +175 -80
  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 +320 -0
  84. package/src/test/utils.ts +167 -0
  85. package/dest/sequencer/block_builder.d.ts +0 -27
  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 -17
  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 -132
@@ -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 { FeeAssetPriceOracle, 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 { 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,41 @@ 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;
441
+ /** Fee asset price oracle for computing price modifiers from Uniswap V4 */ feeAssetPriceOracle;
47
442
  // A CALL to a cold address is 2700 gas
48
443
  static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
49
444
  // Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
@@ -53,23 +448,24 @@ export class SequencerPublisher {
53
448
  govProposerContract;
54
449
  slashingProposerContract;
55
450
  slashFactoryContract;
451
+ tracer;
56
452
  requests;
57
453
  constructor(config, deps){
58
454
  this.config = config;
59
- this.interrupted = false;
455
+ this.interrupted = (_initProto(this), false);
60
456
  this.governanceLog = createLogger('sequencer:publisher:governance');
61
457
  this.slashingLog = createLogger('sequencer:publisher:slashing');
62
458
  this.lastActions = {};
459
+ this.isPayloadEmptyCache = new Map();
63
460
  this.requests = [];
64
461
  this.log = deps.log ?? createLogger('sequencer:publisher');
65
462
  this.ethereumSlotDuration = BigInt(config.ethereumSlotDuration);
66
463
  this.epochCache = deps.epochCache;
67
464
  this.lastActions = deps.lastActions;
68
- this.blobSinkClient = deps.blobSinkClient ?? createBlobSinkClient(config, {
69
- logger: createLogger('sequencer:blob-sink:client')
70
- });
465
+ this.blobClient = deps.blobClient;
71
466
  const telemetry = deps.telemetry ?? getTelemetryClient();
72
467
  this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
468
+ this.tracer = telemetry.getTracer('SequencerPublisher');
73
469
  this.l1TxUtils = deps.l1TxUtils;
74
470
  this.rollupContract = deps.rollupContract;
75
471
  this.govProposerContract = deps.governanceProposerContract;
@@ -80,14 +476,31 @@ export class SequencerPublisher {
80
476
  this.slashingProposerContract = newSlashingProposer;
81
477
  });
82
478
  this.slashFactoryContract = deps.slashFactoryContract;
479
+ // Initialize L1 fee analyzer for fisherman mode
480
+ if (config.fishermanMode) {
481
+ this.l1FeeAnalyzer = new L1FeeAnalyzer(this.l1TxUtils.client, deps.dateProvider, createLogger('sequencer:publisher:fee-analyzer'));
482
+ }
483
+ // Initialize fee asset price oracle
484
+ this.feeAssetPriceOracle = new FeeAssetPriceOracle(this.l1TxUtils.client, this.rollupContract, createLogger('sequencer:publisher:price-oracle'));
83
485
  }
84
486
  getRollupContract() {
85
487
  return this.rollupContract;
86
488
  }
489
+ /**
490
+ * Gets the fee asset price modifier from the oracle.
491
+ * Returns 0n if the oracle query fails.
492
+ */ getFeeAssetPriceModifier() {
493
+ return this.feeAssetPriceOracle.computePriceModifier();
494
+ }
87
495
  getSenderAddress() {
88
496
  return this.l1TxUtils.getSenderAddress();
89
497
  }
90
498
  /**
499
+ * Gets the L1 fee analyzer instance (only available in fisherman mode)
500
+ */ getL1FeeAnalyzer() {
501
+ return this.l1FeeAnalyzer;
502
+ }
503
+ /**
91
504
  * Sets the proposer address to use for simulations in fisherman mode.
92
505
  * @param proposerAddress - The actual proposer's address to use for balance lookups in simulations
93
506
  */ setProposerAddressForSimulation(proposerAddress) {
@@ -109,6 +522,46 @@ export class SequencerPublisher {
109
522
  }
110
523
  }
111
524
  /**
525
+ * Analyzes L1 fees for the pending requests without sending them.
526
+ * This is used in fisherman mode to validate fee calculations.
527
+ * @param l2SlotNumber - The L2 slot number for this analysis
528
+ * @param onComplete - Optional callback to invoke when analysis completes (after block is mined)
529
+ * @returns The analysis result (incomplete until block mines), or undefined if no requests
530
+ */ async analyzeL1Fees(l2SlotNumber, onComplete) {
531
+ if (!this.l1FeeAnalyzer) {
532
+ this.log.warn('L1 fee analyzer not available (not in fisherman mode)');
533
+ return undefined;
534
+ }
535
+ const requestsToAnalyze = [
536
+ ...this.requests
537
+ ];
538
+ if (requestsToAnalyze.length === 0) {
539
+ this.log.debug('No requests to analyze for L1 fees');
540
+ return undefined;
541
+ }
542
+ // Extract blob config from requests (if any)
543
+ const blobConfigs = requestsToAnalyze.filter((request)=>request.blobConfig).map((request)=>request.blobConfig);
544
+ const blobConfig = blobConfigs[0];
545
+ // Get gas configs
546
+ const gasConfigs = requestsToAnalyze.filter((request)=>request.gasConfig).map((request)=>request.gasConfig);
547
+ const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
548
+ const gasLimit = gasLimits.length > 0 ? gasLimits.reduce((sum, g)=>sum + g, 0n) : 0n;
549
+ // Get the transaction requests
550
+ const l1Requests = requestsToAnalyze.map((r)=>r.request);
551
+ // Start the analysis
552
+ const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
553
+ this.log.info('Started L1 fee analysis', {
554
+ analysisId,
555
+ l2SlotNumber: l2SlotNumber.toString(),
556
+ requestCount: requestsToAnalyze.length,
557
+ hasBlobConfig: !!blobConfig,
558
+ gasLimit: gasLimit.toString(),
559
+ actions: requestsToAnalyze.map((r)=>r.action)
560
+ });
561
+ // Return the analysis result (will be incomplete until block mines)
562
+ return this.l1FeeAnalyzer.getAnalysis(analysisId);
563
+ }
564
+ /**
112
565
  * Sends all requests that are still valid.
113
566
  * @returns one of:
114
567
  * - A receipt and stats if the tx succeeded
@@ -119,7 +572,7 @@ export class SequencerPublisher {
119
572
  ...this.requests
120
573
  ];
121
574
  this.requests = [];
122
- if (this.interrupted) {
575
+ if (this.interrupted || requestsToProcess.length === 0) {
123
576
  return undefined;
124
577
  }
125
578
  const currentL2Slot = this.getCurrentL2Slot();
@@ -154,7 +607,16 @@ export class SequencerPublisher {
154
607
  const blobConfig = blobConfigs[0];
155
608
  // Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
156
609
  const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
157
- const gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
610
+ let gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
611
+ // Cap at L1 block gas limit so the node accepts the tx ("gas limit too high" otherwise).
612
+ const maxGas = MAX_L1_TX_LIMIT;
613
+ if (gasLimit !== undefined && gasLimit > maxGas) {
614
+ this.log.debug('Capping bundled tx gas limit to L1 max', {
615
+ requested: gasLimit,
616
+ capped: maxGas
617
+ });
618
+ gasLimit = maxGas;
619
+ }
158
620
  const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
159
621
  const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
160
622
  const txConfig = {
@@ -229,7 +691,7 @@ export class SequencerPublisher {
229
691
  'InvalidArchive'
230
692
  ];
231
693
  return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
232
- forcePendingCheckpointNumber: opts.forcePendingBlockNumber
694
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
233
695
  }).catch((err)=>{
234
696
  if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
235
697
  this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
@@ -257,11 +719,11 @@ export class SequencerPublisher {
257
719
  [],
258
720
  Signature.empty().toViemSignature(),
259
721
  `0x${'0'.repeat(64)}`,
260
- header.contentCommitment.blobsHash.toString(),
722
+ header.blobsHash.toString(),
261
723
  flags
262
724
  ];
263
725
  const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
264
- const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingBlockNumber);
726
+ const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
265
727
  let balance = 0n;
266
728
  if (this.config.fishermanMode) {
267
729
  // In fisherman mode, we can't know where the proposer is publishing from
@@ -288,34 +750,37 @@ export class SequencerPublisher {
288
750
  this.log.debug(`Simulated validateHeader`);
289
751
  }
290
752
  /**
291
- * Simulate making a call to invalidate a block with invalid attestations. Returns undefined if no need to invalidate.
292
- * @param block - The block to invalidate and the criteria for invalidation (as returned by the archiver)
293
- */ async simulateInvalidateBlock(validationResult) {
753
+ * Simulate making a call to invalidate a checkpoint with invalid attestations. Returns undefined if no need to invalidate.
754
+ * @param validationResult - The validation result indicating which checkpoint to invalidate (as returned by the archiver)
755
+ */ async simulateInvalidateCheckpoint(validationResult) {
294
756
  if (validationResult.valid) {
295
757
  return undefined;
296
758
  }
297
- const { reason, block } = validationResult;
298
- const blockNumber = block.blockNumber;
759
+ const { reason, checkpoint } = validationResult;
760
+ const checkpointNumber = checkpoint.checkpointNumber;
299
761
  const logData = {
300
- ...block,
762
+ ...checkpoint,
301
763
  reason
302
764
  };
303
- const currentBlockNumber = await this.rollupContract.getCheckpointNumber();
304
- if (currentBlockNumber < validationResult.block.blockNumber) {
305
- this.log.verbose(`Skipping block ${blockNumber} invalidation since it has already been removed from the pending chain`, {
306
- currentBlockNumber,
765
+ const currentCheckpointNumber = await this.rollupContract.getCheckpointNumber();
766
+ if (currentCheckpointNumber < checkpointNumber) {
767
+ this.log.verbose(`Skipping checkpoint ${checkpointNumber} invalidation since it has already been removed from the pending chain`, {
768
+ currentCheckpointNumber,
307
769
  ...logData
308
770
  });
309
771
  return undefined;
310
772
  }
311
- const request = this.buildInvalidateBlockRequest(validationResult);
312
- this.log.debug(`Simulating invalidate block ${blockNumber}`, {
773
+ const request = this.buildInvalidateCheckpointRequest(validationResult);
774
+ this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
313
775
  ...logData,
314
776
  request
315
777
  });
316
778
  try {
317
- const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, ErrorsAbi);
318
- this.log.verbose(`Simulation for invalidate block ${blockNumber} succeeded`, {
779
+ const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, mergeAbis([
780
+ request.abi ?? [],
781
+ ErrorsAbi
782
+ ]));
783
+ this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} succeeded`, {
319
784
  ...logData,
320
785
  request,
321
786
  gasUsed
@@ -323,90 +788,85 @@ export class SequencerPublisher {
323
788
  return {
324
789
  request,
325
790
  gasUsed,
326
- blockNumber,
327
- forcePendingBlockNumber: blockNumber - 1,
791
+ checkpointNumber,
792
+ forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
328
793
  reason
329
794
  };
330
795
  } catch (err) {
331
796
  const viemError = formatViemError(err);
332
- // If the error is due to the block not being in the pending chain, and it was indeed removed by someone else,
333
- // we can safely ignore it and return undefined so we go ahead with block building.
334
- if (viemError.message?.includes('Rollup__BlockNotInPendingChain')) {
335
- this.log.verbose(`Simulation for invalidate block ${blockNumber} failed due to block not being in pending chain`, {
797
+ // If the error is due to the checkpoint not being in the pending chain, and it was indeed removed by someone else,
798
+ // we can safely ignore it and return undefined so we go ahead with checkpoint building.
799
+ if (viemError.message?.includes('Rollup__CheckpointNotInPendingChain')) {
800
+ this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} failed due to checkpoint not being in pending chain`, {
336
801
  ...logData,
337
802
  request,
338
803
  error: viemError.message
339
804
  });
340
- const latestPendingBlockNumber = await this.rollupContract.getCheckpointNumber();
341
- if (latestPendingBlockNumber < blockNumber) {
342
- this.log.verbose(`Block number ${blockNumber} has already been invalidated`, {
805
+ const latestPendingCheckpointNumber = await this.rollupContract.getCheckpointNumber();
806
+ if (latestPendingCheckpointNumber < checkpointNumber) {
807
+ this.log.verbose(`Checkpoint ${checkpointNumber} has already been invalidated`, {
343
808
  ...logData
344
809
  });
345
810
  return undefined;
346
811
  } else {
347
- this.log.error(`Simulation for invalidate ${blockNumber} failed and it is still in pending chain`, viemError, logData);
348
- throw new Error(`Failed to simulate invalidate block ${blockNumber} while it is still in pending chain`, {
812
+ this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed and it is still in pending chain`, viemError, logData);
813
+ throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber} while it is still in pending chain`, {
349
814
  cause: viemError
350
815
  });
351
816
  }
352
817
  }
353
- // Otherwise, throw. We cannot build the next block if we cannot invalidate the previous one.
354
- this.log.error(`Simulation for invalidate block ${blockNumber} failed`, viemError, logData);
355
- throw new Error(`Failed to simulate invalidate block ${blockNumber}`, {
818
+ // Otherwise, throw. We cannot build the next checkpoint if we cannot invalidate the previous one.
819
+ this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed`, viemError, logData);
820
+ throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber}`, {
356
821
  cause: viemError
357
822
  });
358
823
  }
359
824
  }
360
- buildInvalidateBlockRequest(validationResult) {
825
+ buildInvalidateCheckpointRequest(validationResult) {
361
826
  if (validationResult.valid) {
362
- throw new Error('Cannot invalidate a valid block');
827
+ throw new Error('Cannot invalidate a valid checkpoint');
363
828
  }
364
- const { block, committee, reason } = validationResult;
829
+ const { checkpoint, committee, reason } = validationResult;
365
830
  const logData = {
366
- ...block,
831
+ ...checkpoint,
367
832
  reason
368
833
  };
369
- this.log.debug(`Simulating invalidate block ${block.blockNumber}`, logData);
834
+ this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
370
835
  const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
371
836
  if (reason === 'invalid-attestation') {
372
- return this.rollupContract.buildInvalidateBadAttestationRequest(block.blockNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
837
+ return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
373
838
  } else if (reason === 'insufficient-attestations') {
374
- return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(block.blockNumber, attestationsAndSigners, committee);
839
+ return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
375
840
  } else {
376
841
  const _ = reason;
377
842
  throw new Error(`Unknown reason for invalidation`);
378
843
  }
379
844
  }
380
- /**
381
- * @notice Will simulate `propose` to make sure that the block is valid for submission
382
- *
383
- * @dev Throws if unable to propose
384
- *
385
- * @param block - The block to propose
386
- * @param attestationData - The block's attestation data
387
- *
388
- */ async validateBlockForSubmission(block, attestationsAndSigners, attestationsAndSignersSignature, options) {
845
+ /** Simulates `propose` to make sure that the checkpoint is valid for submission */ async validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, options) {
389
846
  const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
847
+ // TODO(palla/mbps): This should not be needed, there's no flow where we propose with zero attestations. Or is there?
390
848
  // If we have no attestations, we still need to provide the empty attestations
391
849
  // so that the committee is recalculated correctly
392
- const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
393
- if (ignoreSignatures) {
394
- const { committee } = await this.epochCache.getCommittee(block.header.globalVariables.slotNumber);
395
- if (!committee) {
396
- this.log.warn(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
397
- throw new Error(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
398
- }
399
- attestationsAndSigners.attestations = committee.map((committeeMember)=>CommitteeAttestation.fromAddress(committeeMember));
400
- }
401
- const blobFields = block.getCheckpointBlobFields();
850
+ // const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
851
+ // if (ignoreSignatures) {
852
+ // const { committee } = await this.epochCache.getCommittee(block.header.globalVariables.slotNumber);
853
+ // if (!committee) {
854
+ // this.log.warn(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
855
+ // throw new Error(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
856
+ // }
857
+ // attestationsAndSigners.attestations = committee.map(committeeMember =>
858
+ // CommitteeAttestation.fromAddress(committeeMember),
859
+ // );
860
+ // }
861
+ const blobFields = checkpoint.toBlobFields();
402
862
  const blobs = getBlobsPerL1Block(blobFields);
403
863
  const blobInput = getPrefixedEthBlobCommitments(blobs);
404
864
  const args = [
405
865
  {
406
- header: block.getCheckpointHeader().toViem(),
407
- archive: toHex(block.archive.root.toBuffer()),
866
+ header: checkpoint.header.toViem(),
867
+ archive: toHex(checkpoint.archive.root.toBuffer()),
408
868
  oracleInput: {
409
- feeAssetPriceModifier: 0n
869
+ feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
410
870
  }
411
871
  },
412
872
  attestationsAndSigners.getPackedAttestations(),
@@ -431,9 +891,16 @@ export class SequencerPublisher {
431
891
  }
432
892
  const round = await base.computeRound(slotNumber);
433
893
  const roundInfo = await base.getRoundInfo(this.rollupContract.address, round);
894
+ if (roundInfo.quorumReached) {
895
+ return false;
896
+ }
434
897
  if (roundInfo.lastSignalSlot >= slotNumber) {
435
898
  return false;
436
899
  }
900
+ if (await this.isPayloadEmpty(payload)) {
901
+ this.log.warn(`Skipping vote cast for payload with empty code`);
902
+ return false;
903
+ }
437
904
  const cachedLastVote = this.lastActions[signalType];
438
905
  this.lastActions[signalType] = slotNumber;
439
906
  const action = signalType;
@@ -447,7 +914,10 @@ export class SequencerPublisher {
447
914
  try {
448
915
  await this.l1TxUtils.simulate(request, {
449
916
  time: timestamp
450
- }, [], ErrorsAbi);
917
+ }, [], mergeAbis([
918
+ request.abi ?? [],
919
+ ErrorsAbi
920
+ ]));
451
921
  this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
452
922
  request
453
923
  });
@@ -472,17 +942,27 @@ export class SequencerPublisher {
472
942
  payload: payload.toString()
473
943
  };
474
944
  if (!success) {
475
- this.log.error(`Signaling in [${action}] for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
945
+ this.log.error(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
476
946
  this.lastActions[signalType] = cachedLastVote;
477
947
  return false;
478
948
  } else {
479
- this.log.info(`Signaling in [${action}] for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
949
+ this.log.info(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
480
950
  return true;
481
951
  }
482
952
  }
483
953
  });
484
954
  return true;
485
955
  }
956
+ async isPayloadEmpty(payload) {
957
+ const key = payload.toString();
958
+ const cached = this.isPayloadEmptyCache.get(key);
959
+ if (cached) {
960
+ return cached;
961
+ }
962
+ const isEmpty = !await this.l1TxUtils.getCode(payload);
963
+ this.isPayloadEmptyCache.set(key, isEmpty);
964
+ return isEmpty;
965
+ }
486
966
  /**
487
967
  * Enqueues a governance castSignal transaction to cast a signal for a given slot number.
488
968
  * @param slotNumber - The slot number to cast a signal for.
@@ -578,22 +1058,17 @@ export class SequencerPublisher {
578
1058
  }
579
1059
  return true;
580
1060
  }
581
- /**
582
- * Proposes a L2 block on L1.
583
- *
584
- * @param block - L2 block to propose.
585
- * @returns True if the tx has been enqueued, throws otherwise. See #9315
586
- */ async enqueueProposeL2Block(block, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
587
- const checkpointHeader = block.getCheckpointHeader();
588
- const blobFields = block.getCheckpointBlobFields();
1061
+ /** Simulates and enqueues a proposal for a checkpoint on L1 */ async enqueueProposeCheckpoint(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
1062
+ const checkpointHeader = checkpoint.header;
1063
+ const blobFields = checkpoint.toBlobFields();
589
1064
  const blobs = getBlobsPerL1Block(blobFields);
590
1065
  const proposeTxArgs = {
591
1066
  header: checkpointHeader,
592
- archive: block.archive.root.toBuffer(),
593
- body: block.body.toBuffer(),
1067
+ archive: checkpoint.archive.root.toBuffer(),
594
1068
  blobs,
595
1069
  attestationsAndSigners,
596
- attestationsAndSignersSignature
1070
+ attestationsAndSignersSignature,
1071
+ feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
597
1072
  };
598
1073
  let ts;
599
1074
  try {
@@ -602,36 +1077,35 @@ export class SequencerPublisher {
602
1077
  // By simulation issue, I mean the fact that the block.timestamp is equal to the last block, not the next, which
603
1078
  // make time consistency checks break.
604
1079
  // TODO(palla): Check whether we're validating twice, once here and once within addProposeTx, since we call simulateProposeTx in both places.
605
- ts = await this.validateBlockForSubmission(block, attestationsAndSigners, attestationsAndSignersSignature, opts);
1080
+ ts = await this.validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts);
606
1081
  } catch (err) {
607
- this.log.error(`Block validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
608
- ...block.getStats(),
609
- slotNumber: block.header.globalVariables.slotNumber,
610
- forcePendingBlockNumber: opts.forcePendingBlockNumber
1082
+ this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
1083
+ ...checkpoint.getStats(),
1084
+ slotNumber: checkpoint.header.slotNumber,
1085
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
611
1086
  });
612
1087
  throw err;
613
1088
  }
614
- this.log.verbose(`Enqueuing block propose transaction`, {
615
- ...block.toBlockInfo(),
1089
+ this.log.verbose(`Enqueuing checkpoint propose transaction`, {
1090
+ ...checkpoint.toCheckpointInfo(),
616
1091
  ...opts
617
1092
  });
618
- await this.addProposeTx(block, proposeTxArgs, opts, ts);
619
- return true;
1093
+ await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
620
1094
  }
621
- enqueueInvalidateBlock(request, opts = {}) {
1095
+ enqueueInvalidateCheckpoint(request, opts = {}) {
622
1096
  if (!request) {
623
1097
  return;
624
1098
  }
625
1099
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
626
1100
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
627
- const { gasUsed, blockNumber } = request;
1101
+ const { gasUsed, checkpointNumber } = request;
628
1102
  const logData = {
629
1103
  gasUsed,
630
- blockNumber,
1104
+ checkpointNumber,
631
1105
  gasLimit,
632
1106
  opts
633
1107
  };
634
- this.log.verbose(`Enqueuing invalidate block request`, logData);
1108
+ this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
635
1109
  this.addRequest({
636
1110
  action: `invalidate-by-${request.reason}`,
637
1111
  request: request.request,
@@ -643,12 +1117,12 @@ export class SequencerPublisher {
643
1117
  checkSuccess: (_req, result)=>{
644
1118
  const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
645
1119
  if (!success) {
646
- this.log.warn(`Invalidate block ${request.blockNumber} failed`, {
1120
+ this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
647
1121
  ...result,
648
1122
  ...logData
649
1123
  });
650
1124
  } else {
651
- this.log.info(`Invalidate block ${request.blockNumber} succeeded`, {
1125
+ this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
652
1126
  ...result,
653
1127
  ...logData
654
1128
  });
@@ -671,27 +1145,37 @@ export class SequencerPublisher {
671
1145
  this.lastActions[action] = slotNumber;
672
1146
  this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
673
1147
  let gasUsed;
1148
+ const simulateAbi = mergeAbis([
1149
+ request.abi ?? [],
1150
+ ErrorsAbi
1151
+ ]);
674
1152
  try {
675
1153
  ({ gasUsed } = await this.l1TxUtils.simulate(request, {
676
1154
  time: timestamp
677
- }, [], ErrorsAbi)); // TODO(palla/slash): Check the timestamp logic
1155
+ }, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
678
1156
  this.log.verbose(`Simulation for ${action} succeeded`, {
679
1157
  ...logData,
680
1158
  request,
681
1159
  gasUsed
682
1160
  });
683
1161
  } catch (err) {
684
- const viemError = formatViemError(err);
1162
+ const viemError = formatViemError(err, simulateAbi);
685
1163
  this.log.error(`Simulation for ${action} at ${slotNumber} failed`, viemError, logData);
686
1164
  return false;
687
1165
  }
688
1166
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
689
1167
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
690
1168
  logData.gasLimit = gasLimit;
1169
+ // Store the ABI used for simulation on the request so Multicall3.forward can decode errors
1170
+ // when the tx is sent and a revert is diagnosed via simulation.
1171
+ const requestWithAbi = {
1172
+ ...request,
1173
+ abi: simulateAbi
1174
+ };
691
1175
  this.log.debug(`Enqueuing ${action}`, logData);
692
1176
  this.addRequest({
693
1177
  action,
694
- request,
1178
+ request: requestWithAbi,
695
1179
  gasConfig: {
696
1180
  gasLimit
697
1181
  },
@@ -769,8 +1253,7 @@ export class SequencerPublisher {
769
1253
  header: encodedData.header.toViem(),
770
1254
  archive: toHex(encodedData.archive),
771
1255
  oracleInput: {
772
- // We are currently not modifying these. See #9963
773
- feeAssetPriceModifier: 0n
1256
+ feeAssetPriceModifier: encodedData.feeAssetPriceModifier
774
1257
  }
775
1258
  },
776
1259
  encodedData.attestationsAndSigners.getPackedAttestations(),
@@ -797,8 +1280,8 @@ export class SequencerPublisher {
797
1280
  functionName: 'propose',
798
1281
  args
799
1282
  });
800
- // override the pending block number if requested
801
- const forcePendingBlockNumberStateDiff = (options.forcePendingBlockNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingBlockNumber) : []).flatMap((override)=>override.stateDiff ?? []);
1283
+ // override the pending checkpoint number if requested
1284
+ const forcePendingCheckpointNumberStateDiff = (options.forcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
802
1285
  const stateOverrides = [
803
1286
  {
804
1287
  address: this.rollupContract.address,
@@ -808,7 +1291,7 @@ export class SequencerPublisher {
808
1291
  slot: toPaddedHex(RollupContract.checkBlobStorageSlot, true),
809
1292
  value: toPaddedHex(0n, true)
810
1293
  },
811
- ...forcePendingBlockNumberStateDiff
1294
+ ...forcePendingCheckpointNumberStateDiff
812
1295
  ]
813
1296
  }
814
1297
  ];
@@ -822,7 +1305,7 @@ export class SequencerPublisher {
822
1305
  const simulationResult = await this.l1TxUtils.simulate({
823
1306
  to: this.rollupContract.address,
824
1307
  data: rollupData,
825
- gas: SequencerPublisher.PROPOSE_GAS_GUESS,
1308
+ gas: MAX_L1_TX_LIMIT,
826
1309
  ...this.proposerAddressForSimulation && {
827
1310
  from: this.proposerAddressForSimulation.toString()
828
1311
  }
@@ -830,10 +1313,10 @@ export class SequencerPublisher {
830
1313
  // @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
831
1314
  time: timestamp + 1n,
832
1315
  // @note reth should have a 30m gas limit per block but throws errors that this tx is beyond limit so we increase here
833
- gasLimit: SequencerPublisher.PROPOSE_GAS_GUESS * 2n
1316
+ gasLimit: MAX_L1_TX_LIMIT * 2n
834
1317
  }, stateOverrides, RollupAbi, {
835
1318
  // @note fallback gas estimate to use if the node doesn't support simulation API
836
- fallbackGasEstimate: SequencerPublisher.PROPOSE_GAS_GUESS
1319
+ fallbackGasEstimate: MAX_L1_TX_LIMIT
837
1320
  }).catch((err)=>{
838
1321
  // In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
839
1322
  const viemError = formatViemError(err);
@@ -841,7 +1324,7 @@ export class SequencerPublisher {
841
1324
  this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
842
1325
  // Return a minimal simulation result with the fallback gas estimate
843
1326
  return {
844
- gasUsed: SequencerPublisher.PROPOSE_GAS_GUESS,
1327
+ gasUsed: MAX_L1_TX_LIMIT,
845
1328
  logs: []
846
1329
  };
847
1330
  }
@@ -853,24 +1336,25 @@ export class SequencerPublisher {
853
1336
  simulationResult
854
1337
  };
855
1338
  }
856
- async addProposeTx(block, encodedData, opts = {}, timestamp) {
1339
+ async addProposeTx(checkpoint, encodedData, opts = {}, timestamp) {
1340
+ const slot = checkpoint.header.slotNumber;
857
1341
  const timer = new Timer();
858
1342
  const kzg = Blob.getViemKzgInstance();
859
1343
  const { rollupData, simulationResult, blobEvaluationGas } = await this.prepareProposeTx(encodedData, timestamp, opts);
860
1344
  const startBlock = await this.l1TxUtils.getBlockNumber();
861
1345
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(simulationResult.gasUsed) * 64 / 63)) + blobEvaluationGas + SequencerPublisher.MULTICALL_OVERHEAD_GAS_GUESS);
862
- // Send the blobs to the blob sink preemptively. This helps in tests where the sequencer mistakingly thinks that the propose
863
- // tx fails but it does get mined. We make sure that the blobs are sent to the blob sink regardless of the tx outcome.
864
- void this.blobSinkClient.sendBlobsToBlobSink(encodedData.blobs).catch((_err)=>{
865
- this.log.error('Failed to send blobs to blob sink');
866
- });
1346
+ // Send the blobs to the blob client preemptively. This helps in tests where the sequencer mistakingly thinks that the propose
1347
+ // tx fails but it does get mined. We make sure that the blobs are sent to the blob client regardless of the tx outcome.
1348
+ void Promise.resolve().then(()=>this.blobClient.sendBlobsToFilestore(encodedData.blobs).catch((_err)=>{
1349
+ this.log.error('Failed to send blobs to blob client');
1350
+ }));
867
1351
  return this.addRequest({
868
1352
  action: 'propose',
869
1353
  request: {
870
1354
  to: this.rollupContract.address,
871
1355
  data: rollupData
872
1356
  },
873
- lastValidL2Slot: block.header.globalVariables.slotNumber,
1357
+ lastValidL2Slot: checkpoint.header.slotNumber,
874
1358
  gasConfig: {
875
1359
  ...opts,
876
1360
  gasLimit
@@ -898,25 +1382,23 @@ export class SequencerPublisher {
898
1382
  calldataGas,
899
1383
  calldataSize,
900
1384
  sender,
901
- ...block.getStats(),
1385
+ ...checkpoint.getStats(),
902
1386
  eventName: 'rollup-published-to-l1',
903
1387
  blobCount: encodedData.blobs.length,
904
1388
  inclusionBlocks
905
1389
  };
906
- this.log.info(`Published L2 block to L1 rollup contract`, {
1390
+ this.log.info(`Published checkpoint ${checkpoint.number} at slot ${slot} to rollup contract`, {
907
1391
  ...stats,
908
- ...block.getStats(),
909
- ...receipt
1392
+ ...checkpoint.getStats(),
1393
+ ...pick(receipt, 'transactionHash', 'blockHash')
910
1394
  });
911
1395
  this.metrics.recordProcessBlockTx(timer.ms(), publishStats);
912
1396
  return true;
913
1397
  } else {
914
1398
  this.metrics.recordFailedTx('process');
915
- this.log.error(`Rollup process tx failed: ${errorMsg ?? 'no error message'}`, undefined, {
916
- ...block.getStats(),
917
- receipt,
918
- txHash: receipt.transactionHash,
919
- slotNumber: block.header.globalVariables.slotNumber
1399
+ this.log.error(`Publishing checkpoint at slot ${slot} failed with ${errorMsg ?? 'no error message'}`, undefined, {
1400
+ ...checkpoint.getStats(),
1401
+ ...receipt
920
1402
  });
921
1403
  return false;
922
1404
  }