@aztec/sequencer-client 0.0.1-commit.fce3e4f → 0.0.1-commit.ffe5b04ea

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 (115) hide show
  1. package/dest/client/sequencer-client.d.ts +32 -16
  2. package/dest/client/sequencer-client.d.ts.map +1 -1
  3. package/dest/client/sequencer-client.js +118 -28
  4. package/dest/config.d.ts +33 -8
  5. package/dest/config.d.ts.map +1 -1
  6. package/dest/config.js +99 -44
  7. package/dest/global_variable_builder/global_builder.d.ts +20 -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 +41 -20
  14. package/dest/publisher/config.d.ts.map +1 -1
  15. package/dest/publisher/config.js +109 -39
  16. package/dest/publisher/index.d.ts +2 -1
  17. package/dest/publisher/index.d.ts.map +1 -1
  18. package/dest/publisher/l1_tx_failed_store/factory.d.ts +11 -0
  19. package/dest/publisher/l1_tx_failed_store/factory.d.ts.map +1 -0
  20. package/dest/publisher/l1_tx_failed_store/factory.js +22 -0
  21. package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts +59 -0
  22. package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts.map +1 -0
  23. package/dest/publisher/l1_tx_failed_store/failed_tx_store.js +1 -0
  24. package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts +15 -0
  25. package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts.map +1 -0
  26. package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.js +34 -0
  27. package/dest/publisher/l1_tx_failed_store/index.d.ts +4 -0
  28. package/dest/publisher/l1_tx_failed_store/index.d.ts.map +1 -0
  29. package/dest/publisher/l1_tx_failed_store/index.js +2 -0
  30. package/dest/publisher/sequencer-publisher-factory.d.ts +15 -6
  31. package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
  32. package/dest/publisher/sequencer-publisher-factory.js +28 -3
  33. package/dest/publisher/sequencer-publisher-metrics.d.ts +3 -3
  34. package/dest/publisher/sequencer-publisher-metrics.d.ts.map +1 -1
  35. package/dest/publisher/sequencer-publisher-metrics.js +23 -86
  36. package/dest/publisher/sequencer-publisher.d.ts +73 -47
  37. package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
  38. package/dest/publisher/sequencer-publisher.js +888 -146
  39. package/dest/sequencer/checkpoint_proposal_job.d.ts +100 -0
  40. package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -0
  41. package/dest/sequencer/checkpoint_proposal_job.js +1244 -0
  42. package/dest/sequencer/checkpoint_voter.d.ts +35 -0
  43. package/dest/sequencer/checkpoint_voter.d.ts.map +1 -0
  44. package/dest/sequencer/checkpoint_voter.js +109 -0
  45. package/dest/sequencer/config.d.ts +3 -2
  46. package/dest/sequencer/config.d.ts.map +1 -1
  47. package/dest/sequencer/events.d.ts +46 -0
  48. package/dest/sequencer/events.d.ts.map +1 -0
  49. package/dest/sequencer/events.js +1 -0
  50. package/dest/sequencer/index.d.ts +4 -2
  51. package/dest/sequencer/index.d.ts.map +1 -1
  52. package/dest/sequencer/index.js +3 -1
  53. package/dest/sequencer/metrics.d.ts +38 -6
  54. package/dest/sequencer/metrics.d.ts.map +1 -1
  55. package/dest/sequencer/metrics.js +216 -72
  56. package/dest/sequencer/sequencer.d.ts +122 -133
  57. package/dest/sequencer/sequencer.d.ts.map +1 -1
  58. package/dest/sequencer/sequencer.js +717 -625
  59. package/dest/sequencer/timetable.d.ts +54 -16
  60. package/dest/sequencer/timetable.d.ts.map +1 -1
  61. package/dest/sequencer/timetable.js +147 -62
  62. package/dest/sequencer/types.d.ts +6 -0
  63. package/dest/sequencer/types.d.ts.map +1 -0
  64. package/dest/sequencer/types.js +1 -0
  65. package/dest/sequencer/utils.d.ts +14 -8
  66. package/dest/sequencer/utils.d.ts.map +1 -1
  67. package/dest/sequencer/utils.js +7 -4
  68. package/dest/test/index.d.ts +6 -7
  69. package/dest/test/index.d.ts.map +1 -1
  70. package/dest/test/mock_checkpoint_builder.d.ts +95 -0
  71. package/dest/test/mock_checkpoint_builder.d.ts.map +1 -0
  72. package/dest/test/mock_checkpoint_builder.js +231 -0
  73. package/dest/test/utils.d.ts +53 -0
  74. package/dest/test/utils.d.ts.map +1 -0
  75. package/dest/test/utils.js +104 -0
  76. package/package.json +32 -30
  77. package/src/client/sequencer-client.ts +158 -52
  78. package/src/config.ts +114 -54
  79. package/src/global_variable_builder/global_builder.ts +65 -61
  80. package/src/index.ts +1 -7
  81. package/src/publisher/config.ts +131 -50
  82. package/src/publisher/index.ts +3 -0
  83. package/src/publisher/l1_tx_failed_store/factory.ts +32 -0
  84. package/src/publisher/l1_tx_failed_store/failed_tx_store.ts +55 -0
  85. package/src/publisher/l1_tx_failed_store/file_store_failed_tx_store.ts +46 -0
  86. package/src/publisher/l1_tx_failed_store/index.ts +3 -0
  87. package/src/publisher/sequencer-publisher-factory.ts +43 -10
  88. package/src/publisher/sequencer-publisher-metrics.ts +19 -71
  89. package/src/publisher/sequencer-publisher.ts +587 -191
  90. package/src/sequencer/README.md +531 -0
  91. package/src/sequencer/checkpoint_proposal_job.ts +960 -0
  92. package/src/sequencer/checkpoint_voter.ts +130 -0
  93. package/src/sequencer/config.ts +2 -1
  94. package/src/sequencer/events.ts +27 -0
  95. package/src/sequencer/index.ts +3 -1
  96. package/src/sequencer/metrics.ts +268 -82
  97. package/src/sequencer/sequencer.ts +464 -831
  98. package/src/sequencer/timetable.ts +178 -83
  99. package/src/sequencer/types.ts +9 -0
  100. package/src/sequencer/utils.ts +18 -9
  101. package/src/test/index.ts +5 -6
  102. package/src/test/mock_checkpoint_builder.ts +323 -0
  103. package/src/test/utils.ts +167 -0
  104. package/dest/sequencer/block_builder.d.ts +0 -27
  105. package/dest/sequencer/block_builder.d.ts.map +0 -1
  106. package/dest/sequencer/block_builder.js +0 -134
  107. package/dest/tx_validator/nullifier_cache.d.ts +0 -14
  108. package/dest/tx_validator/nullifier_cache.d.ts.map +0 -1
  109. package/dest/tx_validator/nullifier_cache.js +0 -24
  110. package/dest/tx_validator/tx_validator_factory.d.ts +0 -17
  111. package/dest/tx_validator/tx_validator_factory.d.ts.map +0 -1
  112. package/dest/tx_validator/tx_validator_factory.js +0 -53
  113. package/src/sequencer/block_builder.ts +0 -222
  114. package/src/tx_validator/nullifier_cache.ts +0 -30
  115. package/src/tx_validator/tx_validator_factory.ts +0 -132
@@ -1,19 +1,398 @@
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';
383
+ import { TimeoutError } from '@aztec/foundation/error';
7
384
  import { EthAddress } from '@aztec/foundation/eth-address';
8
385
  import { Signature } from '@aztec/foundation/eth-signature';
9
386
  import { createLogger } from '@aztec/foundation/log';
387
+ import { makeBackoff, retry } from '@aztec/foundation/retry';
10
388
  import { bufferToHex } from '@aztec/foundation/string';
11
389
  import { Timer } from '@aztec/foundation/timer';
12
390
  import { EmpireBaseAbi, ErrorsAbi, RollupAbi } from '@aztec/l1-artifacts';
13
391
  import { encodeSlashConsensusVotes } from '@aztec/slasher';
14
- import { CommitteeAttestation, CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
15
- import { getTelemetryClient } from '@aztec/telemetry-client';
16
- import { encodeFunctionData, toHex } from 'viem';
392
+ import { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
393
+ import { getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
394
+ import { encodeFunctionData, keccak256, multicall3Abi, toHex } from 'viem';
395
+ import { createL1TxFailedStore } from './l1_tx_failed_store/index.js';
17
396
  import { SequencerPublisherMetrics } from './sequencer-publisher-metrics.js';
18
397
  export const Actions = [
19
398
  'invalidate-by-invalid-attestation',
@@ -28,22 +407,44 @@ export const Actions = [
28
407
  ];
29
408
  // Sorting for actions such that invalidations go before proposals, and proposals go before votes
30
409
  export const compareActions = (a, b)=>Actions.indexOf(a) - Actions.indexOf(b);
410
+ _dec = trackSpan('SequencerPublisher.sendRequests'), _dec1 = trackSpan('SequencerPublisher.validateBlockHeader'), _dec2 = trackSpan('SequencerPublisher.validateCheckpointForSubmission');
31
411
  export class SequencerPublisher {
32
412
  config;
413
+ static{
414
+ ({ e: [_initProto] } = _apply_decs_2203_r(this, [
415
+ [
416
+ _dec,
417
+ 2,
418
+ "sendRequests"
419
+ ],
420
+ [
421
+ _dec1,
422
+ 2,
423
+ "validateBlockHeader"
424
+ ],
425
+ [
426
+ _dec2,
427
+ 2,
428
+ "validateCheckpointForSubmission"
429
+ ]
430
+ ], []));
431
+ }
33
432
  interrupted;
34
433
  metrics;
35
434
  epochCache;
435
+ failedTxStore;
36
436
  governanceLog;
37
437
  slashingLog;
38
438
  lastActions;
439
+ isPayloadEmptyCache;
440
+ payloadProposedCache;
39
441
  log;
40
442
  ethereumSlotDuration;
41
- blobSinkClient;
443
+ blobClient;
42
444
  /** 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;
445
+ /** Optional callback to obtain a replacement publisher when the current one fails to send. */ getNextPublisher;
446
+ /** L1 fee analyzer for fisherman mode */ l1FeeAnalyzer;
447
+ /** Fee asset price oracle for computing price modifiers from Uniswap V4 */ feeAssetPriceOracle;
47
448
  // A CALL to a cold address is 2700 gas
48
449
  static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
49
450
  // Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
@@ -53,24 +454,27 @@ export class SequencerPublisher {
53
454
  govProposerContract;
54
455
  slashingProposerContract;
55
456
  slashFactoryContract;
457
+ tracer;
56
458
  requests;
57
459
  constructor(config, deps){
58
460
  this.config = config;
59
- this.interrupted = false;
461
+ this.interrupted = (_initProto(this), false);
60
462
  this.governanceLog = createLogger('sequencer:publisher:governance');
61
463
  this.slashingLog = createLogger('sequencer:publisher:slashing');
62
464
  this.lastActions = {};
465
+ this.isPayloadEmptyCache = new Map();
466
+ this.payloadProposedCache = new Set();
63
467
  this.requests = [];
64
468
  this.log = deps.log ?? createLogger('sequencer:publisher');
65
469
  this.ethereumSlotDuration = BigInt(config.ethereumSlotDuration);
66
470
  this.epochCache = deps.epochCache;
67
471
  this.lastActions = deps.lastActions;
68
- this.blobSinkClient = deps.blobSinkClient ?? createBlobSinkClient(config, {
69
- logger: createLogger('sequencer:blob-sink:client')
70
- });
472
+ this.blobClient = deps.blobClient;
71
473
  const telemetry = deps.telemetry ?? getTelemetryClient();
72
474
  this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
475
+ this.tracer = telemetry.getTracer('SequencerPublisher');
73
476
  this.l1TxUtils = deps.l1TxUtils;
477
+ this.getNextPublisher = deps.getNextPublisher;
74
478
  this.rollupContract = deps.rollupContract;
75
479
  this.govProposerContract = deps.governanceProposerContract;
76
480
  this.slashingProposerContract = deps.slashingProposerContract;
@@ -80,14 +484,49 @@ export class SequencerPublisher {
80
484
  this.slashingProposerContract = newSlashingProposer;
81
485
  });
82
486
  this.slashFactoryContract = deps.slashFactoryContract;
487
+ // Initialize L1 fee analyzer for fisherman mode
488
+ if (config.fishermanMode) {
489
+ this.l1FeeAnalyzer = new L1FeeAnalyzer(this.l1TxUtils.client, deps.dateProvider, createLogger('sequencer:publisher:fee-analyzer'));
490
+ }
491
+ // Initialize fee asset price oracle
492
+ this.feeAssetPriceOracle = new FeeAssetPriceOracle(this.l1TxUtils.client, this.rollupContract, createLogger('sequencer:publisher:price-oracle'));
493
+ // Initialize failed L1 tx store (optional, for test networks)
494
+ this.failedTxStore = createL1TxFailedStore(config.l1TxFailedStore, this.log);
495
+ }
496
+ /**
497
+ * Backs up a failed L1 transaction to the configured store for debugging.
498
+ * Does nothing if no store is configured.
499
+ */ backupFailedTx(failedTx) {
500
+ if (!this.failedTxStore) {
501
+ return;
502
+ }
503
+ const tx = {
504
+ ...failedTx,
505
+ timestamp: Date.now()
506
+ };
507
+ // Fire and forget - don't block on backup
508
+ void this.failedTxStore.then((store)=>store?.saveFailedTx(tx)).catch((err)=>{
509
+ this.log.warn(`Failed to backup failed L1 tx to store`, err);
510
+ });
83
511
  }
84
512
  getRollupContract() {
85
513
  return this.rollupContract;
86
514
  }
515
+ /**
516
+ * Gets the fee asset price modifier from the oracle.
517
+ * Returns 0n if the oracle query fails.
518
+ */ getFeeAssetPriceModifier() {
519
+ return this.feeAssetPriceOracle.computePriceModifier();
520
+ }
87
521
  getSenderAddress() {
88
522
  return this.l1TxUtils.getSenderAddress();
89
523
  }
90
524
  /**
525
+ * Gets the L1 fee analyzer instance (only available in fisherman mode)
526
+ */ getL1FeeAnalyzer() {
527
+ return this.l1FeeAnalyzer;
528
+ }
529
+ /**
91
530
  * Sets the proposer address to use for simulations in fisherman mode.
92
531
  * @param proposerAddress - The actual proposer's address to use for balance lookups in simulations
93
532
  */ setProposerAddressForSimulation(proposerAddress) {
@@ -109,6 +548,46 @@ export class SequencerPublisher {
109
548
  }
110
549
  }
111
550
  /**
551
+ * Analyzes L1 fees for the pending requests without sending them.
552
+ * This is used in fisherman mode to validate fee calculations.
553
+ * @param l2SlotNumber - The L2 slot number for this analysis
554
+ * @param onComplete - Optional callback to invoke when analysis completes (after block is mined)
555
+ * @returns The analysis result (incomplete until block mines), or undefined if no requests
556
+ */ async analyzeL1Fees(l2SlotNumber, onComplete) {
557
+ if (!this.l1FeeAnalyzer) {
558
+ this.log.warn('L1 fee analyzer not available (not in fisherman mode)');
559
+ return undefined;
560
+ }
561
+ const requestsToAnalyze = [
562
+ ...this.requests
563
+ ];
564
+ if (requestsToAnalyze.length === 0) {
565
+ this.log.debug('No requests to analyze for L1 fees');
566
+ return undefined;
567
+ }
568
+ // Extract blob config from requests (if any)
569
+ const blobConfigs = requestsToAnalyze.filter((request)=>request.blobConfig).map((request)=>request.blobConfig);
570
+ const blobConfig = blobConfigs[0];
571
+ // Get gas configs
572
+ const gasConfigs = requestsToAnalyze.filter((request)=>request.gasConfig).map((request)=>request.gasConfig);
573
+ const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
574
+ const gasLimit = gasLimits.length > 0 ? gasLimits.reduce((sum, g)=>sum + g, 0n) : 0n;
575
+ // Get the transaction requests
576
+ const l1Requests = requestsToAnalyze.map((r)=>r.request);
577
+ // Start the analysis
578
+ const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
579
+ this.log.info('Started L1 fee analysis', {
580
+ analysisId,
581
+ l2SlotNumber: l2SlotNumber.toString(),
582
+ requestCount: requestsToAnalyze.length,
583
+ hasBlobConfig: !!blobConfig,
584
+ gasLimit: gasLimit.toString(),
585
+ actions: requestsToAnalyze.map((r)=>r.action)
586
+ });
587
+ // Return the analysis result (will be incomplete until block mines)
588
+ return this.l1FeeAnalyzer.getAnalysis(analysisId);
589
+ }
590
+ /**
112
591
  * Sends all requests that are still valid.
113
592
  * @returns one of:
114
593
  * - A receipt and stats if the tx succeeded
@@ -119,7 +598,7 @@ export class SequencerPublisher {
119
598
  ...this.requests
120
599
  ];
121
600
  this.requests = [];
122
- if (this.interrupted) {
601
+ if (this.interrupted || requestsToProcess.length === 0) {
123
602
  return undefined;
124
603
  }
125
604
  const currentL2Slot = this.getCurrentL2Slot();
@@ -154,7 +633,16 @@ export class SequencerPublisher {
154
633
  const blobConfig = blobConfigs[0];
155
634
  // Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
156
635
  const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
157
- const gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
636
+ let gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
637
+ // Cap at L1 block gas limit so the node accepts the tx ("gas limit too high" otherwise).
638
+ const maxGas = MAX_L1_TX_LIMIT;
639
+ if (gasLimit !== undefined && gasLimit > maxGas) {
640
+ this.log.debug('Capping bundled tx gas limit to L1 max', {
641
+ requested: gasLimit,
642
+ capped: maxGas
643
+ });
644
+ gasLimit = maxGas;
645
+ }
158
646
  const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
159
647
  const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
160
648
  const txConfig = {
@@ -165,12 +653,34 @@ export class SequencerPublisher {
165
653
  // This ensures the committee gets precomputed correctly
166
654
  validRequests.sort((a, b)=>compareActions(a.action, b.action));
167
655
  try {
656
+ // Capture context for failed tx backup before sending
657
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
658
+ const multicallData = encodeFunctionData({
659
+ abi: multicall3Abi,
660
+ functionName: 'aggregate3',
661
+ args: [
662
+ validRequests.map((r)=>({
663
+ target: r.request.to,
664
+ callData: r.request.data,
665
+ allowFailure: true
666
+ }))
667
+ ]
668
+ });
669
+ const blobDataHex = blobConfig?.blobs?.map((b)=>toHex(b));
670
+ const txContext = {
671
+ multicallData,
672
+ blobData: blobDataHex,
673
+ l1BlockNumber
674
+ };
168
675
  this.log.debug('Forwarding transactions', {
169
676
  validRequests: validRequests.map((request)=>request.action),
170
677
  txConfig
171
678
  });
172
- const result = await Multicall3.forward(validRequests.map((request)=>request.request), this.l1TxUtils, txConfig, blobConfig, this.rollupContract.address, this.log);
173
- const { successfulActions = [], failedActions = [] } = this.callbackBundledTransactions(validRequests, result);
679
+ const result = await this.forwardWithPublisherRotation(validRequests, txConfig, blobConfig);
680
+ if (result === undefined) {
681
+ return undefined;
682
+ }
683
+ const { successfulActions = [], failedActions = [] } = this.callbackBundledTransactions(validRequests, result, txContext);
174
684
  return {
175
685
  result,
176
686
  expiredActions,
@@ -190,10 +700,67 @@ export class SequencerPublisher {
190
700
  }
191
701
  }
192
702
  }
193
- callbackBundledTransactions(requests, result) {
703
+ /**
704
+ * Forwards transactions via Multicall3, rotating to the next available publisher if a send
705
+ * failure occurs (i.e. the tx never reached the chain).
706
+ * On-chain reverts and simulation errors are returned as-is without rotation.
707
+ */ async forwardWithPublisherRotation(validRequests, txConfig, blobConfig) {
708
+ const triedAddresses = [];
709
+ let currentPublisher = this.l1TxUtils;
710
+ while(true){
711
+ triedAddresses.push(currentPublisher.getSenderAddress());
712
+ try {
713
+ const result = await Multicall3.forward(validRequests.map((r)=>r.request), currentPublisher, txConfig, blobConfig, this.rollupContract.address, this.log);
714
+ this.l1TxUtils = currentPublisher;
715
+ return result;
716
+ } catch (err) {
717
+ if (err instanceof TimeoutError) {
718
+ throw err;
719
+ }
720
+ const viemError = formatViemError(err);
721
+ if (!this.getNextPublisher) {
722
+ this.log.error('Failed to publish bundled transactions', viemError);
723
+ return undefined;
724
+ }
725
+ this.log.warn(`Publisher ${currentPublisher.getSenderAddress()} failed to send, rotating to next publisher`, viemError);
726
+ const nextPublisher = await this.getNextPublisher([
727
+ ...triedAddresses
728
+ ]);
729
+ if (!nextPublisher) {
730
+ this.log.error('All available publishers exhausted, failed to publish bundled transactions');
731
+ return undefined;
732
+ }
733
+ currentPublisher = nextPublisher;
734
+ }
735
+ }
736
+ }
737
+ callbackBundledTransactions(requests, result, txContext) {
194
738
  const actionsListStr = requests.map((r)=>r.action).join(', ');
195
739
  if (result instanceof FormattedViemError) {
196
740
  this.log.error(`Failed to publish bundled transactions (${actionsListStr})`, result);
741
+ this.backupFailedTx({
742
+ id: keccak256(txContext.multicallData),
743
+ failureType: 'send-error',
744
+ request: {
745
+ to: MULTI_CALL_3_ADDRESS,
746
+ data: txContext.multicallData
747
+ },
748
+ blobData: txContext.blobData,
749
+ l1BlockNumber: txContext.l1BlockNumber.toString(),
750
+ error: {
751
+ message: result.message,
752
+ name: result.name
753
+ },
754
+ context: {
755
+ actions: requests.map((r)=>r.action),
756
+ requests: requests.map((r)=>({
757
+ action: r.action,
758
+ to: r.request.to,
759
+ data: r.request.data
760
+ })),
761
+ sender: this.getSenderAddress().toString()
762
+ }
763
+ });
197
764
  return {
198
765
  failedActions: requests.map((r)=>r.action)
199
766
  };
@@ -211,6 +778,37 @@ export class SequencerPublisher {
211
778
  failedActions.push(request.action);
212
779
  }
213
780
  }
781
+ // Single backup for the whole reverted tx
782
+ if (failedActions.length > 0 && result?.receipt?.status === 'reverted') {
783
+ this.backupFailedTx({
784
+ id: result.receipt.transactionHash,
785
+ failureType: 'revert',
786
+ request: {
787
+ to: MULTI_CALL_3_ADDRESS,
788
+ data: txContext.multicallData
789
+ },
790
+ blobData: txContext.blobData,
791
+ l1BlockNumber: result.receipt.blockNumber.toString(),
792
+ receipt: {
793
+ transactionHash: result.receipt.transactionHash,
794
+ blockNumber: result.receipt.blockNumber.toString(),
795
+ gasUsed: (result.receipt.gasUsed ?? 0n).toString(),
796
+ status: 'reverted'
797
+ },
798
+ error: {
799
+ message: result.errorMsg ?? 'Transaction reverted'
800
+ },
801
+ context: {
802
+ actions: failedActions,
803
+ requests: requests.filter((r)=>failedActions.includes(r.action)).map((r)=>({
804
+ action: r.action,
805
+ to: r.request.to,
806
+ data: r.request.data
807
+ })),
808
+ sender: this.getSenderAddress().toString()
809
+ }
810
+ });
811
+ }
214
812
  return {
215
813
  successfulActions,
216
814
  failedActions
@@ -229,7 +827,7 @@ export class SequencerPublisher {
229
827
  'InvalidArchive'
230
828
  ];
231
829
  return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
232
- forcePendingCheckpointNumber: opts.forcePendingBlockNumber
830
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
233
831
  }).catch((err)=>{
234
832
  if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
235
833
  this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
@@ -257,11 +855,11 @@ export class SequencerPublisher {
257
855
  [],
258
856
  Signature.empty().toViemSignature(),
259
857
  `0x${'0'.repeat(64)}`,
260
- header.contentCommitment.blobsHash.toString(),
858
+ header.blobsHash.toString(),
261
859
  flags
262
860
  ];
263
861
  const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
264
- const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingBlockNumber);
862
+ const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
265
863
  let balance = 0n;
266
864
  if (this.config.fishermanMode) {
267
865
  // In fisherman mode, we can't know where the proposer is publishing from
@@ -288,34 +886,38 @@ export class SequencerPublisher {
288
886
  this.log.debug(`Simulated validateHeader`);
289
887
  }
290
888
  /**
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) {
889
+ * Simulate making a call to invalidate a checkpoint with invalid attestations. Returns undefined if no need to invalidate.
890
+ * @param validationResult - The validation result indicating which checkpoint to invalidate (as returned by the archiver)
891
+ */ async simulateInvalidateCheckpoint(validationResult) {
294
892
  if (validationResult.valid) {
295
893
  return undefined;
296
894
  }
297
- const { reason, block } = validationResult;
298
- const blockNumber = block.blockNumber;
895
+ const { reason, checkpoint } = validationResult;
896
+ const checkpointNumber = checkpoint.checkpointNumber;
299
897
  const logData = {
300
- ...block,
898
+ ...checkpoint,
301
899
  reason
302
900
  };
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,
901
+ const currentCheckpointNumber = await this.rollupContract.getCheckpointNumber();
902
+ if (currentCheckpointNumber < checkpointNumber) {
903
+ this.log.verbose(`Skipping checkpoint ${checkpointNumber} invalidation since it has already been removed from the pending chain`, {
904
+ currentCheckpointNumber,
307
905
  ...logData
308
906
  });
309
907
  return undefined;
310
908
  }
311
- const request = this.buildInvalidateBlockRequest(validationResult);
312
- this.log.debug(`Simulating invalidate block ${blockNumber}`, {
909
+ const request = this.buildInvalidateCheckpointRequest(validationResult);
910
+ this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
313
911
  ...logData,
314
912
  request
315
913
  });
914
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
316
915
  try {
317
- const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, ErrorsAbi);
318
- this.log.verbose(`Simulation for invalidate block ${blockNumber} succeeded`, {
916
+ const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, mergeAbis([
917
+ request.abi ?? [],
918
+ ErrorsAbi
919
+ ]));
920
+ this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} succeeded`, {
319
921
  ...logData,
320
922
  request,
321
923
  gasUsed
@@ -323,90 +925,92 @@ export class SequencerPublisher {
323
925
  return {
324
926
  request,
325
927
  gasUsed,
326
- blockNumber,
327
- forcePendingBlockNumber: blockNumber - 1,
928
+ checkpointNumber,
929
+ forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
328
930
  reason
329
931
  };
330
932
  } catch (err) {
331
933
  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`, {
934
+ // If the error is due to the checkpoint not being in the pending chain, and it was indeed removed by someone else,
935
+ // we can safely ignore it and return undefined so we go ahead with checkpoint building.
936
+ if (viemError.message?.includes('Rollup__CheckpointNotInPendingChain')) {
937
+ this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} failed due to checkpoint not being in pending chain`, {
336
938
  ...logData,
337
939
  request,
338
940
  error: viemError.message
339
941
  });
340
- const latestPendingBlockNumber = await this.rollupContract.getCheckpointNumber();
341
- if (latestPendingBlockNumber < blockNumber) {
342
- this.log.verbose(`Block number ${blockNumber} has already been invalidated`, {
942
+ const latestPendingCheckpointNumber = await this.rollupContract.getCheckpointNumber();
943
+ if (latestPendingCheckpointNumber < checkpointNumber) {
944
+ this.log.verbose(`Checkpoint ${checkpointNumber} has already been invalidated`, {
343
945
  ...logData
344
946
  });
345
947
  return undefined;
346
948
  } 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`, {
949
+ this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed and it is still in pending chain`, viemError, logData);
950
+ throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber} while it is still in pending chain`, {
349
951
  cause: viemError
350
952
  });
351
953
  }
352
954
  }
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}`, {
955
+ // Otherwise, throw. We cannot build the next checkpoint if we cannot invalidate the previous one.
956
+ this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed`, viemError, logData);
957
+ this.backupFailedTx({
958
+ id: keccak256(request.data),
959
+ failureType: 'simulation',
960
+ request: {
961
+ to: request.to,
962
+ data: request.data,
963
+ value: request.value?.toString()
964
+ },
965
+ l1BlockNumber: l1BlockNumber.toString(),
966
+ error: {
967
+ message: viemError.message,
968
+ name: viemError.name
969
+ },
970
+ context: {
971
+ actions: [
972
+ `invalidate-${reason}`
973
+ ],
974
+ checkpointNumber,
975
+ sender: this.getSenderAddress().toString()
976
+ }
977
+ });
978
+ throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber}`, {
356
979
  cause: viemError
357
980
  });
358
981
  }
359
982
  }
360
- buildInvalidateBlockRequest(validationResult) {
983
+ buildInvalidateCheckpointRequest(validationResult) {
361
984
  if (validationResult.valid) {
362
- throw new Error('Cannot invalidate a valid block');
985
+ throw new Error('Cannot invalidate a valid checkpoint');
363
986
  }
364
- const { block, committee, reason } = validationResult;
987
+ const { checkpoint, committee, reason } = validationResult;
365
988
  const logData = {
366
- ...block,
989
+ ...checkpoint,
367
990
  reason
368
991
  };
369
- this.log.debug(`Simulating invalidate block ${block.blockNumber}`, logData);
992
+ this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
370
993
  const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
371
994
  if (reason === 'invalid-attestation') {
372
- return this.rollupContract.buildInvalidateBadAttestationRequest(block.blockNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
995
+ return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
373
996
  } else if (reason === 'insufficient-attestations') {
374
- return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(block.blockNumber, attestationsAndSigners, committee);
997
+ return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
375
998
  } else {
376
999
  const _ = reason;
377
1000
  throw new Error(`Unknown reason for invalidation`);
378
1001
  }
379
1002
  }
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) {
1003
+ /** Simulates `propose` to make sure that the checkpoint is valid for submission */ async validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, options) {
389
1004
  const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
390
- // If we have no attestations, we still need to provide the empty attestations
391
- // 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();
402
- const blobs = getBlobsPerL1Block(blobFields);
1005
+ const blobFields = checkpoint.toBlobFields();
1006
+ const blobs = await getBlobsPerL1Block(blobFields);
403
1007
  const blobInput = getPrefixedEthBlobCommitments(blobs);
404
1008
  const args = [
405
1009
  {
406
- header: block.getCheckpointHeader().toViem(),
407
- archive: toHex(block.archive.root.toBuffer()),
1010
+ header: checkpoint.header.toViem(),
1011
+ archive: toHex(checkpoint.archive.root.toBuffer()),
408
1012
  oracleInput: {
409
- feeAssetPriceModifier: 0n
1013
+ feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
410
1014
  }
411
1015
  },
412
1016
  attestationsAndSigners.getPackedAttestations(),
@@ -431,9 +1035,38 @@ export class SequencerPublisher {
431
1035
  }
432
1036
  const round = await base.computeRound(slotNumber);
433
1037
  const roundInfo = await base.getRoundInfo(this.rollupContract.address, round);
1038
+ if (roundInfo.quorumReached) {
1039
+ return false;
1040
+ }
434
1041
  if (roundInfo.lastSignalSlot >= slotNumber) {
435
1042
  return false;
436
1043
  }
1044
+ if (await this.isPayloadEmpty(payload)) {
1045
+ this.log.warn(`Skipping vote cast for payload with empty code`);
1046
+ return false;
1047
+ }
1048
+ // Check if payload was already submitted to governance
1049
+ const cacheKey = payload.toString();
1050
+ if (!this.payloadProposedCache.has(cacheKey)) {
1051
+ try {
1052
+ const l1StartBlock = await this.rollupContract.getL1StartBlock();
1053
+ const proposed = await retry(()=>base.hasPayloadBeenProposed(payload.toString(), l1StartBlock), 'Check if payload was proposed', makeBackoff([
1054
+ 0,
1055
+ 1,
1056
+ 2
1057
+ ]), this.log, true);
1058
+ if (proposed) {
1059
+ this.payloadProposedCache.add(cacheKey);
1060
+ }
1061
+ } catch (err) {
1062
+ this.log.warn(`Failed to check if payload ${payload} was proposed after retries, skipping signal`, err);
1063
+ return false;
1064
+ }
1065
+ }
1066
+ if (this.payloadProposedCache.has(cacheKey)) {
1067
+ this.log.info(`Payload ${payload} was already proposed to governance, stopping signals`);
1068
+ return false;
1069
+ }
437
1070
  const cachedLastVote = this.lastActions[signalType];
438
1071
  this.lastActions[signalType] = slotNumber;
439
1072
  const action = signalType;
@@ -444,15 +1077,41 @@ export class SequencerPublisher {
444
1077
  signer: this.l1TxUtils.client.account?.address,
445
1078
  lastValidL2Slot: slotNumber
446
1079
  });
1080
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
447
1081
  try {
448
1082
  await this.l1TxUtils.simulate(request, {
449
1083
  time: timestamp
450
- }, [], ErrorsAbi);
1084
+ }, [], mergeAbis([
1085
+ request.abi ?? [],
1086
+ ErrorsAbi
1087
+ ]));
451
1088
  this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
452
1089
  request
453
1090
  });
454
1091
  } catch (err) {
455
- this.log.error(`Failed simulation for ${action} at slot ${slotNumber} (enqueuing the action anyway)`, err);
1092
+ const viemError = formatViemError(err);
1093
+ this.log.error(`Failed simulation for ${action} at slot ${slotNumber} (enqueuing the action anyway)`, viemError);
1094
+ this.backupFailedTx({
1095
+ id: keccak256(request.data),
1096
+ failureType: 'simulation',
1097
+ request: {
1098
+ to: request.to,
1099
+ data: request.data,
1100
+ value: request.value?.toString()
1101
+ },
1102
+ l1BlockNumber: l1BlockNumber.toString(),
1103
+ error: {
1104
+ message: viemError.message,
1105
+ name: viemError.name
1106
+ },
1107
+ context: {
1108
+ actions: [
1109
+ action
1110
+ ],
1111
+ slot: slotNumber,
1112
+ sender: this.getSenderAddress().toString()
1113
+ }
1114
+ });
456
1115
  // Yes, we enqueue the request anyway, in case there was a bug with the simulation itself
457
1116
  }
458
1117
  // TODO(palla/slash): All votes (governance and slashing) should txTimeoutAt at the end of the slot.
@@ -472,17 +1131,27 @@ export class SequencerPublisher {
472
1131
  payload: payload.toString()
473
1132
  };
474
1133
  if (!success) {
475
- this.log.error(`Signaling in [${action}] for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
1134
+ this.log.error(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
476
1135
  this.lastActions[signalType] = cachedLastVote;
477
1136
  return false;
478
1137
  } else {
479
- this.log.info(`Signaling in [${action}] for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
1138
+ this.log.info(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
480
1139
  return true;
481
1140
  }
482
1141
  }
483
1142
  });
484
1143
  return true;
485
1144
  }
1145
+ async isPayloadEmpty(payload) {
1146
+ const key = payload.toString();
1147
+ const cached = this.isPayloadEmptyCache.get(key);
1148
+ if (cached) {
1149
+ return cached;
1150
+ }
1151
+ const isEmpty = !await this.l1TxUtils.getCode(payload);
1152
+ this.isPayloadEmptyCache.set(key, isEmpty);
1153
+ return isEmpty;
1154
+ }
486
1155
  /**
487
1156
  * Enqueues a governance castSignal transaction to cast a signal for a given slot number.
488
1157
  * @param slotNumber - The slot number to cast a signal for.
@@ -578,22 +1247,17 @@ export class SequencerPublisher {
578
1247
  }
579
1248
  return true;
580
1249
  }
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();
589
- const blobs = getBlobsPerL1Block(blobFields);
1250
+ /** Simulates and enqueues a proposal for a checkpoint on L1 */ async enqueueProposeCheckpoint(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
1251
+ const checkpointHeader = checkpoint.header;
1252
+ const blobFields = checkpoint.toBlobFields();
1253
+ const blobs = await getBlobsPerL1Block(blobFields);
590
1254
  const proposeTxArgs = {
591
1255
  header: checkpointHeader,
592
- archive: block.archive.root.toBuffer(),
593
- body: block.body.toBuffer(),
1256
+ archive: checkpoint.archive.root.toBuffer(),
594
1257
  blobs,
595
1258
  attestationsAndSigners,
596
- attestationsAndSignersSignature
1259
+ attestationsAndSignersSignature,
1260
+ feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
597
1261
  };
598
1262
  let ts;
599
1263
  try {
@@ -602,36 +1266,35 @@ export class SequencerPublisher {
602
1266
  // By simulation issue, I mean the fact that the block.timestamp is equal to the last block, not the next, which
603
1267
  // make time consistency checks break.
604
1268
  // 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);
1269
+ ts = await this.validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts);
606
1270
  } 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
1271
+ this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
1272
+ ...checkpoint.getStats(),
1273
+ slotNumber: checkpoint.header.slotNumber,
1274
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
611
1275
  });
612
1276
  throw err;
613
1277
  }
614
- this.log.verbose(`Enqueuing block propose transaction`, {
615
- ...block.toBlockInfo(),
1278
+ this.log.verbose(`Enqueuing checkpoint propose transaction`, {
1279
+ ...checkpoint.toCheckpointInfo(),
616
1280
  ...opts
617
1281
  });
618
- await this.addProposeTx(block, proposeTxArgs, opts, ts);
619
- return true;
1282
+ await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
620
1283
  }
621
- enqueueInvalidateBlock(request, opts = {}) {
1284
+ enqueueInvalidateCheckpoint(request, opts = {}) {
622
1285
  if (!request) {
623
1286
  return;
624
1287
  }
625
1288
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
626
1289
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
627
- const { gasUsed, blockNumber } = request;
1290
+ const { gasUsed, checkpointNumber } = request;
628
1291
  const logData = {
629
1292
  gasUsed,
630
- blockNumber,
1293
+ checkpointNumber,
631
1294
  gasLimit,
632
1295
  opts
633
1296
  };
634
- this.log.verbose(`Enqueuing invalidate block request`, logData);
1297
+ this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
635
1298
  this.addRequest({
636
1299
  action: `invalidate-by-${request.reason}`,
637
1300
  request: request.request,
@@ -643,12 +1306,12 @@ export class SequencerPublisher {
643
1306
  checkSuccess: (_req, result)=>{
644
1307
  const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
645
1308
  if (!success) {
646
- this.log.warn(`Invalidate block ${request.blockNumber} failed`, {
1309
+ this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
647
1310
  ...result,
648
1311
  ...logData
649
1312
  });
650
1313
  } else {
651
- this.log.info(`Invalidate block ${request.blockNumber} succeeded`, {
1314
+ this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
652
1315
  ...result,
653
1316
  ...logData
654
1317
  });
@@ -670,28 +1333,60 @@ export class SequencerPublisher {
670
1333
  const cachedLastActionSlot = this.lastActions[action];
671
1334
  this.lastActions[action] = slotNumber;
672
1335
  this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
1336
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
673
1337
  let gasUsed;
1338
+ const simulateAbi = mergeAbis([
1339
+ request.abi ?? [],
1340
+ ErrorsAbi
1341
+ ]);
674
1342
  try {
675
1343
  ({ gasUsed } = await this.l1TxUtils.simulate(request, {
676
1344
  time: timestamp
677
- }, [], ErrorsAbi)); // TODO(palla/slash): Check the timestamp logic
1345
+ }, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
678
1346
  this.log.verbose(`Simulation for ${action} succeeded`, {
679
1347
  ...logData,
680
1348
  request,
681
1349
  gasUsed
682
1350
  });
683
1351
  } catch (err) {
684
- const viemError = formatViemError(err);
1352
+ const viemError = formatViemError(err, simulateAbi);
685
1353
  this.log.error(`Simulation for ${action} at ${slotNumber} failed`, viemError, logData);
1354
+ this.backupFailedTx({
1355
+ id: keccak256(request.data),
1356
+ failureType: 'simulation',
1357
+ request: {
1358
+ to: request.to,
1359
+ data: request.data,
1360
+ value: request.value?.toString()
1361
+ },
1362
+ l1BlockNumber: l1BlockNumber.toString(),
1363
+ error: {
1364
+ message: viemError.message,
1365
+ name: viemError.name
1366
+ },
1367
+ context: {
1368
+ actions: [
1369
+ action
1370
+ ],
1371
+ slot: slotNumber,
1372
+ sender: this.getSenderAddress().toString()
1373
+ }
1374
+ });
686
1375
  return false;
687
1376
  }
688
1377
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
689
1378
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
690
1379
  logData.gasLimit = gasLimit;
1380
+ // Store the ABI used for simulation on the request so Multicall3.forward can decode errors
1381
+ // when the tx is sent and a revert is diagnosed via simulation.
1382
+ const requestWithAbi = {
1383
+ ...request,
1384
+ abi: simulateAbi
1385
+ };
691
1386
  this.log.debug(`Enqueuing ${action}`, logData);
692
1387
  this.addRequest({
693
1388
  action,
694
- request,
1389
+ request: requestWithAbi,
695
1390
  gasConfig: {
696
1391
  gasLimit
697
1392
  },
@@ -755,10 +1450,38 @@ export class SequencerPublisher {
755
1450
  }, {}, {
756
1451
  blobs: encodedData.blobs.map((b)=>b.data),
757
1452
  kzg
758
- }).catch((err)=>{
759
- const { message, metaMessages } = formatViemError(err);
760
- this.log.error(`Failed to validate blobs`, message, {
761
- metaMessages
1453
+ }).catch(async (err)=>{
1454
+ const viemError = formatViemError(err);
1455
+ this.log.error(`Failed to validate blobs`, viemError.message, {
1456
+ metaMessages: viemError.metaMessages
1457
+ });
1458
+ const validateBlobsData = encodeFunctionData({
1459
+ abi: RollupAbi,
1460
+ functionName: 'validateBlobs',
1461
+ args: [
1462
+ blobInput
1463
+ ]
1464
+ });
1465
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
1466
+ this.backupFailedTx({
1467
+ id: keccak256(validateBlobsData),
1468
+ failureType: 'simulation',
1469
+ request: {
1470
+ to: this.rollupContract.address,
1471
+ data: validateBlobsData
1472
+ },
1473
+ blobData: encodedData.blobs.map((b)=>toHex(b.data)),
1474
+ l1BlockNumber: l1BlockNumber.toString(),
1475
+ error: {
1476
+ message: viemError.message,
1477
+ name: viemError.name
1478
+ },
1479
+ context: {
1480
+ actions: [
1481
+ 'validate-blobs'
1482
+ ],
1483
+ sender: this.getSenderAddress().toString()
1484
+ }
762
1485
  });
763
1486
  throw new Error('Failed to validate blobs');
764
1487
  });
@@ -769,8 +1492,7 @@ export class SequencerPublisher {
769
1492
  header: encodedData.header.toViem(),
770
1493
  archive: toHex(encodedData.archive),
771
1494
  oracleInput: {
772
- // We are currently not modifying these. See #9963
773
- feeAssetPriceModifier: 0n
1495
+ feeAssetPriceModifier: encodedData.feeAssetPriceModifier
774
1496
  }
775
1497
  },
776
1498
  encodedData.attestationsAndSigners.getPackedAttestations(),
@@ -797,8 +1519,8 @@ export class SequencerPublisher {
797
1519
  functionName: 'propose',
798
1520
  args
799
1521
  });
800
- // override the pending block number if requested
801
- const forcePendingBlockNumberStateDiff = (options.forcePendingBlockNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingBlockNumber) : []).flatMap((override)=>override.stateDiff ?? []);
1522
+ // override the pending checkpoint number if requested
1523
+ const forcePendingCheckpointNumberStateDiff = (options.forcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
802
1524
  const stateOverrides = [
803
1525
  {
804
1526
  address: this.rollupContract.address,
@@ -808,7 +1530,7 @@ export class SequencerPublisher {
808
1530
  slot: toPaddedHex(RollupContract.checkBlobStorageSlot, true),
809
1531
  value: toPaddedHex(0n, true)
810
1532
  },
811
- ...forcePendingBlockNumberStateDiff
1533
+ ...forcePendingCheckpointNumberStateDiff
812
1534
  ]
813
1535
  }
814
1536
  ];
@@ -819,10 +1541,11 @@ export class SequencerPublisher {
819
1541
  balance: 10n * WEI_CONST * WEI_CONST
820
1542
  });
821
1543
  }
1544
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
822
1545
  const simulationResult = await this.l1TxUtils.simulate({
823
1546
  to: this.rollupContract.address,
824
1547
  data: rollupData,
825
- gas: SequencerPublisher.PROPOSE_GAS_GUESS,
1548
+ gas: MAX_L1_TX_LIMIT,
826
1549
  ...this.proposerAddressForSimulation && {
827
1550
  from: this.proposerAddressForSimulation.toString()
828
1551
  }
@@ -830,10 +1553,10 @@ export class SequencerPublisher {
830
1553
  // @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
831
1554
  time: timestamp + 1n,
832
1555
  // @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
1556
+ gasLimit: MAX_L1_TX_LIMIT * 2n
834
1557
  }, stateOverrides, RollupAbi, {
835
1558
  // @note fallback gas estimate to use if the node doesn't support simulation API
836
- fallbackGasEstimate: SequencerPublisher.PROPOSE_GAS_GUESS
1559
+ fallbackGasEstimate: MAX_L1_TX_LIMIT
837
1560
  }).catch((err)=>{
838
1561
  // In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
839
1562
  const viemError = formatViemError(err);
@@ -841,11 +1564,31 @@ export class SequencerPublisher {
841
1564
  this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
842
1565
  // Return a minimal simulation result with the fallback gas estimate
843
1566
  return {
844
- gasUsed: SequencerPublisher.PROPOSE_GAS_GUESS,
1567
+ gasUsed: MAX_L1_TX_LIMIT,
845
1568
  logs: []
846
1569
  };
847
1570
  }
848
1571
  this.log.error(`Failed to simulate propose tx`, viemError);
1572
+ this.backupFailedTx({
1573
+ id: keccak256(rollupData),
1574
+ failureType: 'simulation',
1575
+ request: {
1576
+ to: this.rollupContract.address,
1577
+ data: rollupData
1578
+ },
1579
+ l1BlockNumber: l1BlockNumber.toString(),
1580
+ error: {
1581
+ message: viemError.message,
1582
+ name: viemError.name
1583
+ },
1584
+ context: {
1585
+ actions: [
1586
+ 'propose'
1587
+ ],
1588
+ slot: Number(args[0].header.slotNumber),
1589
+ sender: this.getSenderAddress().toString()
1590
+ }
1591
+ });
849
1592
  throw err;
850
1593
  });
851
1594
  return {
@@ -853,24 +1596,25 @@ export class SequencerPublisher {
853
1596
  simulationResult
854
1597
  };
855
1598
  }
856
- async addProposeTx(block, encodedData, opts = {}, timestamp) {
1599
+ async addProposeTx(checkpoint, encodedData, opts = {}, timestamp) {
1600
+ const slot = checkpoint.header.slotNumber;
857
1601
  const timer = new Timer();
858
1602
  const kzg = Blob.getViemKzgInstance();
859
1603
  const { rollupData, simulationResult, blobEvaluationGas } = await this.prepareProposeTx(encodedData, timestamp, opts);
860
1604
  const startBlock = await this.l1TxUtils.getBlockNumber();
861
1605
  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
- });
1606
+ // Send the blobs to the blob client preemptively. This helps in tests where the sequencer mistakingly thinks that the propose
1607
+ // tx fails but it does get mined. We make sure that the blobs are sent to the blob client regardless of the tx outcome.
1608
+ void Promise.resolve().then(()=>this.blobClient.sendBlobsToFilestore(encodedData.blobs).catch((_err)=>{
1609
+ this.log.error('Failed to send blobs to blob client');
1610
+ }));
867
1611
  return this.addRequest({
868
1612
  action: 'propose',
869
1613
  request: {
870
1614
  to: this.rollupContract.address,
871
1615
  data: rollupData
872
1616
  },
873
- lastValidL2Slot: block.header.globalVariables.slotNumber,
1617
+ lastValidL2Slot: checkpoint.header.slotNumber,
874
1618
  gasConfig: {
875
1619
  ...opts,
876
1620
  gasLimit
@@ -898,25 +1642,23 @@ export class SequencerPublisher {
898
1642
  calldataGas,
899
1643
  calldataSize,
900
1644
  sender,
901
- ...block.getStats(),
1645
+ ...checkpoint.getStats(),
902
1646
  eventName: 'rollup-published-to-l1',
903
1647
  blobCount: encodedData.blobs.length,
904
1648
  inclusionBlocks
905
1649
  };
906
- this.log.info(`Published L2 block to L1 rollup contract`, {
1650
+ this.log.info(`Published checkpoint ${checkpoint.number} at slot ${slot} to rollup contract`, {
907
1651
  ...stats,
908
- ...block.getStats(),
909
- ...receipt
1652
+ ...checkpoint.getStats(),
1653
+ ...pick(receipt, 'transactionHash', 'blockHash')
910
1654
  });
911
1655
  this.metrics.recordProcessBlockTx(timer.ms(), publishStats);
912
1656
  return true;
913
1657
  } else {
914
1658
  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
1659
+ this.log.error(`Publishing checkpoint at slot ${slot} failed with ${errorMsg ?? 'no error message'}`, undefined, {
1660
+ ...checkpoint.getStats(),
1661
+ ...receipt
920
1662
  });
921
1663
  return false;
922
1664
  }