@aztec/sequencer-client 0.0.1-commit.03f7ef2 → 0.0.1-commit.0658669b3

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 (103) hide show
  1. package/dest/client/sequencer-client.d.ts +15 -11
  2. package/dest/client/sequencer-client.d.ts.map +1 -1
  3. package/dest/client/sequencer-client.js +15 -4
  4. package/dest/config.d.ts +3 -4
  5. package/dest/config.d.ts.map +1 -1
  6. package/dest/config.js +22 -12
  7. package/dest/global_variable_builder/global_builder.d.ts +5 -7
  8. package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
  9. package/dest/global_variable_builder/global_builder.js +13 -13
  10. package/dest/index.d.ts +2 -3
  11. package/dest/index.d.ts.map +1 -1
  12. package/dest/index.js +1 -2
  13. package/dest/publisher/config.d.ts +35 -17
  14. package/dest/publisher/config.d.ts.map +1 -1
  15. package/dest/publisher/config.js +106 -42
  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 +12 -4
  31. package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
  32. package/dest/publisher/sequencer-publisher-factory.js +13 -2
  33. package/dest/publisher/sequencer-publisher-metrics.d.ts +1 -1
  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 +40 -25
  37. package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
  38. package/dest/publisher/sequencer-publisher.js +740 -100
  39. package/dest/sequencer/checkpoint_proposal_job.d.ts +40 -12
  40. package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -1
  41. package/dest/sequencer/checkpoint_proposal_job.js +643 -64
  42. package/dest/sequencer/checkpoint_voter.d.ts +3 -2
  43. package/dest/sequencer/checkpoint_voter.d.ts.map +1 -1
  44. package/dest/sequencer/checkpoint_voter.js +34 -10
  45. package/dest/sequencer/index.d.ts +1 -3
  46. package/dest/sequencer/index.d.ts.map +1 -1
  47. package/dest/sequencer/index.js +0 -2
  48. package/dest/sequencer/metrics.d.ts +19 -7
  49. package/dest/sequencer/metrics.d.ts.map +1 -1
  50. package/dest/sequencer/metrics.js +131 -141
  51. package/dest/sequencer/sequencer.d.ts +38 -18
  52. package/dest/sequencer/sequencer.d.ts.map +1 -1
  53. package/dest/sequencer/sequencer.js +513 -66
  54. package/dest/sequencer/timetable.d.ts +1 -4
  55. package/dest/sequencer/timetable.d.ts.map +1 -1
  56. package/dest/sequencer/timetable.js +1 -4
  57. package/dest/test/index.d.ts +4 -7
  58. package/dest/test/index.d.ts.map +1 -1
  59. package/dest/test/mock_checkpoint_builder.d.ts +25 -11
  60. package/dest/test/mock_checkpoint_builder.d.ts.map +1 -1
  61. package/dest/test/mock_checkpoint_builder.js +52 -9
  62. package/dest/test/utils.d.ts +13 -9
  63. package/dest/test/utils.d.ts.map +1 -1
  64. package/dest/test/utils.js +27 -17
  65. package/package.json +30 -28
  66. package/src/client/sequencer-client.ts +29 -12
  67. package/src/config.ts +31 -19
  68. package/src/global_variable_builder/global_builder.ts +14 -14
  69. package/src/index.ts +1 -9
  70. package/src/publisher/config.ts +121 -43
  71. package/src/publisher/index.ts +3 -0
  72. package/src/publisher/l1_tx_failed_store/factory.ts +32 -0
  73. package/src/publisher/l1_tx_failed_store/failed_tx_store.ts +55 -0
  74. package/src/publisher/l1_tx_failed_store/file_store_failed_tx_store.ts +46 -0
  75. package/src/publisher/l1_tx_failed_store/index.ts +3 -0
  76. package/src/publisher/sequencer-publisher-factory.ts +24 -7
  77. package/src/publisher/sequencer-publisher-metrics.ts +17 -69
  78. package/src/publisher/sequencer-publisher.ts +361 -130
  79. package/src/sequencer/checkpoint_proposal_job.ts +316 -93
  80. package/src/sequencer/checkpoint_voter.ts +32 -7
  81. package/src/sequencer/index.ts +0 -2
  82. package/src/sequencer/metrics.ts +132 -148
  83. package/src/sequencer/sequencer.ts +159 -68
  84. package/src/sequencer/timetable.ts +6 -5
  85. package/src/test/index.ts +3 -6
  86. package/src/test/mock_checkpoint_builder.ts +102 -29
  87. package/src/test/utils.ts +58 -28
  88. package/dest/sequencer/block_builder.d.ts +0 -26
  89. package/dest/sequencer/block_builder.d.ts.map +0 -1
  90. package/dest/sequencer/block_builder.js +0 -129
  91. package/dest/sequencer/checkpoint_builder.d.ts +0 -63
  92. package/dest/sequencer/checkpoint_builder.d.ts.map +0 -1
  93. package/dest/sequencer/checkpoint_builder.js +0 -131
  94. package/dest/tx_validator/nullifier_cache.d.ts +0 -14
  95. package/dest/tx_validator/nullifier_cache.d.ts.map +0 -1
  96. package/dest/tx_validator/nullifier_cache.js +0 -24
  97. package/dest/tx_validator/tx_validator_factory.d.ts +0 -18
  98. package/dest/tx_validator/tx_validator_factory.d.ts.map +0 -1
  99. package/dest/tx_validator/tx_validator_factory.js +0 -53
  100. package/src/sequencer/block_builder.ts +0 -217
  101. package/src/sequencer/checkpoint_builder.ts +0 -217
  102. package/src/tx_validator/nullifier_cache.ts +0 -30
  103. package/src/tx_validator/tx_validator_factory.ts +0 -133
@@ -1,23 +1,397 @@
1
- import { createBlobClient } from '@aztec/blob-client/client';
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;
2
374
  import { Blob, getBlobsPerL1Block, getPrefixedEthBlobCommitments } from '@aztec/blob-lib';
3
- import { MULTI_CALL_3_ADDRESS, Multicall3, RollupContract } from '@aztec/ethereum/contracts';
375
+ import { FeeAssetPriceOracle, MULTI_CALL_3_ADDRESS, Multicall3, RollupContract } from '@aztec/ethereum/contracts';
4
376
  import { L1FeeAnalyzer } from '@aztec/ethereum/l1-fee-analysis';
5
- import { WEI_CONST } from '@aztec/ethereum/l1-tx-utils';
6
- import { FormattedViemError, formatViemError, tryExtractEvent } from '@aztec/ethereum/utils';
377
+ import { MAX_L1_TX_LIMIT, WEI_CONST } from '@aztec/ethereum/l1-tx-utils';
378
+ import { FormattedViemError, formatViemError, mergeAbis, tryExtractEvent } from '@aztec/ethereum/utils';
7
379
  import { sumBigint } from '@aztec/foundation/bigint';
8
380
  import { toHex as toPaddedHex } from '@aztec/foundation/bigint-buffer';
9
- import { BlockNumber, CheckpointNumber, SlotNumber } from '@aztec/foundation/branded-types';
381
+ import { CheckpointNumber, SlotNumber } from '@aztec/foundation/branded-types';
10
382
  import { pick } from '@aztec/foundation/collection';
11
383
  import { EthAddress } from '@aztec/foundation/eth-address';
12
384
  import { Signature } from '@aztec/foundation/eth-signature';
13
385
  import { createLogger } from '@aztec/foundation/log';
386
+ import { makeBackoff, retry } from '@aztec/foundation/retry';
14
387
  import { bufferToHex } from '@aztec/foundation/string';
15
388
  import { Timer } from '@aztec/foundation/timer';
16
389
  import { EmpireBaseAbi, ErrorsAbi, RollupAbi } from '@aztec/l1-artifacts';
17
390
  import { encodeSlashConsensusVotes } from '@aztec/slasher';
18
391
  import { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
19
- import { getTelemetryClient } from '@aztec/telemetry-client';
20
- import { encodeFunctionData, toHex } from 'viem';
392
+ import { getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
393
+ import { encodeFunctionData, keccak256, multicall3Abi, toHex } from 'viem';
394
+ import { createL1TxFailedStore } from './l1_tx_failed_store/index.js';
21
395
  import { SequencerPublisherMetrics } from './sequencer-publisher-metrics.js';
22
396
  export const Actions = [
23
397
  'invalidate-by-invalid-attestation',
@@ -32,24 +406,43 @@ export const Actions = [
32
406
  ];
33
407
  // Sorting for actions such that invalidations go before proposals, and proposals go before votes
34
408
  export const compareActions = (a, b)=>Actions.indexOf(a) - Actions.indexOf(b);
409
+ _dec = trackSpan('SequencerPublisher.sendRequests'), _dec1 = trackSpan('SequencerPublisher.validateBlockHeader'), _dec2 = trackSpan('SequencerPublisher.validateCheckpointForSubmission');
35
410
  export class SequencerPublisher {
36
411
  config;
412
+ static{
413
+ ({ e: [_initProto] } = _apply_decs_2203_r(this, [
414
+ [
415
+ _dec,
416
+ 2,
417
+ "sendRequests"
418
+ ],
419
+ [
420
+ _dec1,
421
+ 2,
422
+ "validateBlockHeader"
423
+ ],
424
+ [
425
+ _dec2,
426
+ 2,
427
+ "validateCheckpointForSubmission"
428
+ ]
429
+ ], []));
430
+ }
37
431
  interrupted;
38
432
  metrics;
39
433
  epochCache;
434
+ failedTxStore;
40
435
  governanceLog;
41
436
  slashingLog;
42
437
  lastActions;
43
438
  isPayloadEmptyCache;
439
+ payloadProposedCache;
44
440
  log;
45
441
  ethereumSlotDuration;
46
442
  blobClient;
47
443
  /** Address to use for simulations in fisherman mode (actual proposer's address) */ proposerAddressForSimulation;
48
444
  /** L1 fee analyzer for fisherman mode */ l1FeeAnalyzer;
49
- // @note - with blobs, the below estimate seems too large.
50
- // Total used for full block from int_l1_pub e2e test: 1m (of which 86k is 1x blob)
51
- // Total used for emptier block from above test: 429k (of which 84k is 1x blob)
52
- static PROPOSE_GAS_GUESS = 12_000_000n;
445
+ /** Fee asset price oracle for computing price modifiers from Uniswap V4 */ feeAssetPriceOracle;
53
446
  // A CALL to a cold address is 2700 gas
54
447
  static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
55
448
  // Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
@@ -59,24 +452,25 @@ export class SequencerPublisher {
59
452
  govProposerContract;
60
453
  slashingProposerContract;
61
454
  slashFactoryContract;
455
+ tracer;
62
456
  requests;
63
457
  constructor(config, deps){
64
458
  this.config = config;
65
- this.interrupted = false;
459
+ this.interrupted = (_initProto(this), false);
66
460
  this.governanceLog = createLogger('sequencer:publisher:governance');
67
461
  this.slashingLog = createLogger('sequencer:publisher:slashing');
68
462
  this.lastActions = {};
69
463
  this.isPayloadEmptyCache = new Map();
464
+ this.payloadProposedCache = new Set();
70
465
  this.requests = [];
71
466
  this.log = deps.log ?? createLogger('sequencer:publisher');
72
467
  this.ethereumSlotDuration = BigInt(config.ethereumSlotDuration);
73
468
  this.epochCache = deps.epochCache;
74
469
  this.lastActions = deps.lastActions;
75
- this.blobClient = deps.blobClient ?? createBlobClient(config, {
76
- logger: createLogger('sequencer:blob-client:client')
77
- });
470
+ this.blobClient = deps.blobClient;
78
471
  const telemetry = deps.telemetry ?? getTelemetryClient();
79
472
  this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
473
+ this.tracer = telemetry.getTracer('SequencerPublisher');
80
474
  this.l1TxUtils = deps.l1TxUtils;
81
475
  this.rollupContract = deps.rollupContract;
82
476
  this.govProposerContract = deps.governanceProposerContract;
@@ -91,10 +485,36 @@ export class SequencerPublisher {
91
485
  if (config.fishermanMode) {
92
486
  this.l1FeeAnalyzer = new L1FeeAnalyzer(this.l1TxUtils.client, deps.dateProvider, createLogger('sequencer:publisher:fee-analyzer'));
93
487
  }
488
+ // Initialize fee asset price oracle
489
+ this.feeAssetPriceOracle = new FeeAssetPriceOracle(this.l1TxUtils.client, this.rollupContract, createLogger('sequencer:publisher:price-oracle'));
490
+ // Initialize failed L1 tx store (optional, for test networks)
491
+ this.failedTxStore = createL1TxFailedStore(config.l1TxFailedStore, this.log);
492
+ }
493
+ /**
494
+ * Backs up a failed L1 transaction to the configured store for debugging.
495
+ * Does nothing if no store is configured.
496
+ */ backupFailedTx(failedTx) {
497
+ if (!this.failedTxStore) {
498
+ return;
499
+ }
500
+ const tx = {
501
+ ...failedTx,
502
+ timestamp: Date.now()
503
+ };
504
+ // Fire and forget - don't block on backup
505
+ void this.failedTxStore.then((store)=>store?.saveFailedTx(tx)).catch((err)=>{
506
+ this.log.warn(`Failed to backup failed L1 tx to store`, err);
507
+ });
94
508
  }
95
509
  getRollupContract() {
96
510
  return this.rollupContract;
97
511
  }
512
+ /**
513
+ * Gets the fee asset price modifier from the oracle.
514
+ * Returns 0n if the oracle query fails.
515
+ */ getFeeAssetPriceModifier() {
516
+ return this.feeAssetPriceOracle.computePriceModifier();
517
+ }
98
518
  getSenderAddress() {
99
519
  return this.l1TxUtils.getSenderAddress();
100
520
  }
@@ -152,7 +572,7 @@ export class SequencerPublisher {
152
572
  // Get the transaction requests
153
573
  const l1Requests = requestsToAnalyze.map((r)=>r.request);
154
574
  // Start the analysis
155
- const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : SequencerPublisher.PROPOSE_GAS_GUESS, l1Requests, blobConfig, onComplete);
575
+ const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
156
576
  this.log.info('Started L1 fee analysis', {
157
577
  analysisId,
158
578
  l2SlotNumber: l2SlotNumber.toString(),
@@ -210,7 +630,16 @@ export class SequencerPublisher {
210
630
  const blobConfig = blobConfigs[0];
211
631
  // Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
212
632
  const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
213
- const gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
633
+ let gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
634
+ // Cap at L1 block gas limit so the node accepts the tx ("gas limit too high" otherwise).
635
+ const maxGas = MAX_L1_TX_LIMIT;
636
+ if (gasLimit !== undefined && gasLimit > maxGas) {
637
+ this.log.debug('Capping bundled tx gas limit to L1 max', {
638
+ requested: gasLimit,
639
+ capped: maxGas
640
+ });
641
+ gasLimit = maxGas;
642
+ }
214
643
  const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
215
644
  const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
216
645
  const txConfig = {
@@ -221,12 +650,31 @@ export class SequencerPublisher {
221
650
  // This ensures the committee gets precomputed correctly
222
651
  validRequests.sort((a, b)=>compareActions(a.action, b.action));
223
652
  try {
653
+ // Capture context for failed tx backup before sending
654
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
655
+ const multicallData = encodeFunctionData({
656
+ abi: multicall3Abi,
657
+ functionName: 'aggregate3',
658
+ args: [
659
+ validRequests.map((r)=>({
660
+ target: r.request.to,
661
+ callData: r.request.data,
662
+ allowFailure: true
663
+ }))
664
+ ]
665
+ });
666
+ const blobDataHex = blobConfig?.blobs?.map((b)=>toHex(b));
224
667
  this.log.debug('Forwarding transactions', {
225
668
  validRequests: validRequests.map((request)=>request.action),
226
669
  txConfig
227
670
  });
228
671
  const result = await Multicall3.forward(validRequests.map((request)=>request.request), this.l1TxUtils, txConfig, blobConfig, this.rollupContract.address, this.log);
229
- const { successfulActions = [], failedActions = [] } = this.callbackBundledTransactions(validRequests, result);
672
+ const txContext = {
673
+ multicallData,
674
+ blobData: blobDataHex,
675
+ l1BlockNumber
676
+ };
677
+ const { successfulActions = [], failedActions = [] } = this.callbackBundledTransactions(validRequests, result, txContext);
230
678
  return {
231
679
  result,
232
680
  expiredActions,
@@ -246,10 +694,33 @@ export class SequencerPublisher {
246
694
  }
247
695
  }
248
696
  }
249
- callbackBundledTransactions(requests, result) {
697
+ callbackBundledTransactions(requests, result, txContext) {
250
698
  const actionsListStr = requests.map((r)=>r.action).join(', ');
251
699
  if (result instanceof FormattedViemError) {
252
700
  this.log.error(`Failed to publish bundled transactions (${actionsListStr})`, result);
701
+ this.backupFailedTx({
702
+ id: keccak256(txContext.multicallData),
703
+ failureType: 'send-error',
704
+ request: {
705
+ to: MULTI_CALL_3_ADDRESS,
706
+ data: txContext.multicallData
707
+ },
708
+ blobData: txContext.blobData,
709
+ l1BlockNumber: txContext.l1BlockNumber.toString(),
710
+ error: {
711
+ message: result.message,
712
+ name: result.name
713
+ },
714
+ context: {
715
+ actions: requests.map((r)=>r.action),
716
+ requests: requests.map((r)=>({
717
+ action: r.action,
718
+ to: r.request.to,
719
+ data: r.request.data
720
+ })),
721
+ sender: this.getSenderAddress().toString()
722
+ }
723
+ });
253
724
  return {
254
725
  failedActions: requests.map((r)=>r.action)
255
726
  };
@@ -267,6 +738,37 @@ export class SequencerPublisher {
267
738
  failedActions.push(request.action);
268
739
  }
269
740
  }
741
+ // Single backup for the whole reverted tx
742
+ if (failedActions.length > 0 && result?.receipt?.status === 'reverted') {
743
+ this.backupFailedTx({
744
+ id: result.receipt.transactionHash,
745
+ failureType: 'revert',
746
+ request: {
747
+ to: MULTI_CALL_3_ADDRESS,
748
+ data: txContext.multicallData
749
+ },
750
+ blobData: txContext.blobData,
751
+ l1BlockNumber: result.receipt.blockNumber.toString(),
752
+ receipt: {
753
+ transactionHash: result.receipt.transactionHash,
754
+ blockNumber: result.receipt.blockNumber.toString(),
755
+ gasUsed: (result.receipt.gasUsed ?? 0n).toString(),
756
+ status: 'reverted'
757
+ },
758
+ error: {
759
+ message: result.errorMsg ?? 'Transaction reverted'
760
+ },
761
+ context: {
762
+ actions: failedActions,
763
+ requests: requests.filter((r)=>failedActions.includes(r.action)).map((r)=>({
764
+ action: r.action,
765
+ to: r.request.to,
766
+ data: r.request.data
767
+ })),
768
+ sender: this.getSenderAddress().toString()
769
+ }
770
+ });
771
+ }
270
772
  return {
271
773
  successfulActions,
272
774
  failedActions
@@ -285,7 +787,7 @@ export class SequencerPublisher {
285
787
  'InvalidArchive'
286
788
  ];
287
789
  return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
288
- forcePendingCheckpointNumber: opts.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(opts.forcePendingBlockNumber) : undefined
790
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
289
791
  }).catch((err)=>{
290
792
  if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
291
793
  this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
@@ -313,12 +815,11 @@ export class SequencerPublisher {
313
815
  [],
314
816
  Signature.empty().toViemSignature(),
315
817
  `0x${'0'.repeat(64)}`,
316
- header.contentCommitment.blobsHash.toString(),
818
+ header.blobsHash.toString(),
317
819
  flags
318
820
  ];
319
821
  const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
320
- const optsForcePendingCheckpointNumber = opts?.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(opts.forcePendingBlockNumber) : undefined;
321
- const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(optsForcePendingCheckpointNumber);
822
+ const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
322
823
  let balance = 0n;
323
824
  if (this.config.fishermanMode) {
324
825
  // In fisherman mode, we can't know where the proposer is publishing from
@@ -345,34 +846,38 @@ export class SequencerPublisher {
345
846
  this.log.debug(`Simulated validateHeader`);
346
847
  }
347
848
  /**
348
- * Simulate making a call to invalidate a block with invalid attestations. Returns undefined if no need to invalidate.
349
- * @param block - The block to invalidate and the criteria for invalidation (as returned by the archiver)
350
- */ async simulateInvalidateBlock(validationResult) {
849
+ * Simulate making a call to invalidate a checkpoint with invalid attestations. Returns undefined if no need to invalidate.
850
+ * @param validationResult - The validation result indicating which checkpoint to invalidate (as returned by the archiver)
851
+ */ async simulateInvalidateCheckpoint(validationResult) {
351
852
  if (validationResult.valid) {
352
853
  return undefined;
353
854
  }
354
- const { reason, block } = validationResult;
355
- const blockNumber = block.blockNumber;
855
+ const { reason, checkpoint } = validationResult;
856
+ const checkpointNumber = checkpoint.checkpointNumber;
356
857
  const logData = {
357
- ...block,
858
+ ...checkpoint,
358
859
  reason
359
860
  };
360
- const currentBlockNumber = await this.rollupContract.getCheckpointNumber();
361
- if (currentBlockNumber < validationResult.block.blockNumber) {
362
- this.log.verbose(`Skipping block ${blockNumber} invalidation since it has already been removed from the pending chain`, {
363
- currentBlockNumber,
861
+ const currentCheckpointNumber = await this.rollupContract.getCheckpointNumber();
862
+ if (currentCheckpointNumber < checkpointNumber) {
863
+ this.log.verbose(`Skipping checkpoint ${checkpointNumber} invalidation since it has already been removed from the pending chain`, {
864
+ currentCheckpointNumber,
364
865
  ...logData
365
866
  });
366
867
  return undefined;
367
868
  }
368
- const request = this.buildInvalidateBlockRequest(validationResult);
369
- this.log.debug(`Simulating invalidate block ${blockNumber}`, {
869
+ const request = this.buildInvalidateCheckpointRequest(validationResult);
870
+ this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
370
871
  ...logData,
371
872
  request
372
873
  });
874
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
373
875
  try {
374
- const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, ErrorsAbi);
375
- this.log.verbose(`Simulation for invalidate block ${blockNumber} succeeded`, {
876
+ const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, mergeAbis([
877
+ request.abi ?? [],
878
+ ErrorsAbi
879
+ ]));
880
+ this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} succeeded`, {
376
881
  ...logData,
377
882
  request,
378
883
  gasUsed
@@ -380,55 +885,76 @@ export class SequencerPublisher {
380
885
  return {
381
886
  request,
382
887
  gasUsed,
383
- blockNumber,
384
- forcePendingBlockNumber: BlockNumber(blockNumber - 1),
888
+ checkpointNumber,
889
+ forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
385
890
  reason
386
891
  };
387
892
  } catch (err) {
388
893
  const viemError = formatViemError(err);
389
- // If the error is due to the block not being in the pending chain, and it was indeed removed by someone else,
390
- // we can safely ignore it and return undefined so we go ahead with block building.
391
- if (viemError.message?.includes('Rollup__BlockNotInPendingChain')) {
392
- this.log.verbose(`Simulation for invalidate block ${blockNumber} failed due to block not being in pending chain`, {
894
+ // If the error is due to the checkpoint not being in the pending chain, and it was indeed removed by someone else,
895
+ // we can safely ignore it and return undefined so we go ahead with checkpoint building.
896
+ if (viemError.message?.includes('Rollup__CheckpointNotInPendingChain')) {
897
+ this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} failed due to checkpoint not being in pending chain`, {
393
898
  ...logData,
394
899
  request,
395
900
  error: viemError.message
396
901
  });
397
- const latestPendingBlockNumber = await this.rollupContract.getCheckpointNumber();
398
- if (latestPendingBlockNumber < blockNumber) {
399
- this.log.verbose(`Block number ${blockNumber} has already been invalidated`, {
902
+ const latestPendingCheckpointNumber = await this.rollupContract.getCheckpointNumber();
903
+ if (latestPendingCheckpointNumber < checkpointNumber) {
904
+ this.log.verbose(`Checkpoint ${checkpointNumber} has already been invalidated`, {
400
905
  ...logData
401
906
  });
402
907
  return undefined;
403
908
  } else {
404
- this.log.error(`Simulation for invalidate ${blockNumber} failed and it is still in pending chain`, viemError, logData);
405
- throw new Error(`Failed to simulate invalidate block ${blockNumber} while it is still in pending chain`, {
909
+ this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed and it is still in pending chain`, viemError, logData);
910
+ throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber} while it is still in pending chain`, {
406
911
  cause: viemError
407
912
  });
408
913
  }
409
914
  }
410
- // Otherwise, throw. We cannot build the next block if we cannot invalidate the previous one.
411
- this.log.error(`Simulation for invalidate block ${blockNumber} failed`, viemError, logData);
412
- throw new Error(`Failed to simulate invalidate block ${blockNumber}`, {
915
+ // Otherwise, throw. We cannot build the next checkpoint if we cannot invalidate the previous one.
916
+ this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed`, viemError, logData);
917
+ this.backupFailedTx({
918
+ id: keccak256(request.data),
919
+ failureType: 'simulation',
920
+ request: {
921
+ to: request.to,
922
+ data: request.data,
923
+ value: request.value?.toString()
924
+ },
925
+ l1BlockNumber: l1BlockNumber.toString(),
926
+ error: {
927
+ message: viemError.message,
928
+ name: viemError.name
929
+ },
930
+ context: {
931
+ actions: [
932
+ `invalidate-${reason}`
933
+ ],
934
+ checkpointNumber,
935
+ sender: this.getSenderAddress().toString()
936
+ }
937
+ });
938
+ throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber}`, {
413
939
  cause: viemError
414
940
  });
415
941
  }
416
942
  }
417
- buildInvalidateBlockRequest(validationResult) {
943
+ buildInvalidateCheckpointRequest(validationResult) {
418
944
  if (validationResult.valid) {
419
- throw new Error('Cannot invalidate a valid block');
945
+ throw new Error('Cannot invalidate a valid checkpoint');
420
946
  }
421
- const { block, committee, reason } = validationResult;
947
+ const { checkpoint, committee, reason } = validationResult;
422
948
  const logData = {
423
- ...block,
949
+ ...checkpoint,
424
950
  reason
425
951
  };
426
- this.log.debug(`Simulating invalidate block ${block.blockNumber}`, logData);
952
+ this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
427
953
  const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
428
954
  if (reason === 'invalid-attestation') {
429
- return this.rollupContract.buildInvalidateBadAttestationRequest(CheckpointNumber.fromBlockNumber(block.blockNumber), attestationsAndSigners, committee, validationResult.invalidIndex);
955
+ return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
430
956
  } else if (reason === 'insufficient-attestations') {
431
- return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(CheckpointNumber.fromBlockNumber(block.blockNumber), attestationsAndSigners, committee);
957
+ return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
432
958
  } else {
433
959
  const _ = reason;
434
960
  throw new Error(`Unknown reason for invalidation`);
@@ -436,29 +962,15 @@ export class SequencerPublisher {
436
962
  }
437
963
  /** Simulates `propose` to make sure that the checkpoint is valid for submission */ async validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, options) {
438
964
  const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
439
- // TODO(palla/mbps): This should not be needed, there's no flow where we propose with zero attestations. Or is there?
440
- // If we have no attestations, we still need to provide the empty attestations
441
- // so that the committee is recalculated correctly
442
- // const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
443
- // if (ignoreSignatures) {
444
- // const { committee } = await this.epochCache.getCommittee(block.header.globalVariables.slotNumber);
445
- // if (!committee) {
446
- // this.log.warn(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
447
- // throw new Error(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
448
- // }
449
- // attestationsAndSigners.attestations = committee.map(committeeMember =>
450
- // CommitteeAttestation.fromAddress(committeeMember),
451
- // );
452
- // }
453
965
  const blobFields = checkpoint.toBlobFields();
454
- const blobs = getBlobsPerL1Block(blobFields);
966
+ const blobs = await getBlobsPerL1Block(blobFields);
455
967
  const blobInput = getPrefixedEthBlobCommitments(blobs);
456
968
  const args = [
457
969
  {
458
970
  header: checkpoint.header.toViem(),
459
971
  archive: toHex(checkpoint.archive.root.toBuffer()),
460
972
  oracleInput: {
461
- feeAssetPriceModifier: 0n
973
+ feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
462
974
  }
463
975
  },
464
976
  attestationsAndSigners.getPackedAttestations(),
@@ -493,6 +1005,28 @@ export class SequencerPublisher {
493
1005
  this.log.warn(`Skipping vote cast for payload with empty code`);
494
1006
  return false;
495
1007
  }
1008
+ // Check if payload was already submitted to governance
1009
+ const cacheKey = payload.toString();
1010
+ if (!this.payloadProposedCache.has(cacheKey)) {
1011
+ try {
1012
+ const l1StartBlock = await this.rollupContract.getL1StartBlock();
1013
+ const proposed = await retry(()=>base.hasPayloadBeenProposed(payload.toString(), l1StartBlock), 'Check if payload was proposed', makeBackoff([
1014
+ 0,
1015
+ 1,
1016
+ 2
1017
+ ]), this.log, true);
1018
+ if (proposed) {
1019
+ this.payloadProposedCache.add(cacheKey);
1020
+ }
1021
+ } catch (err) {
1022
+ this.log.warn(`Failed to check if payload ${payload} was proposed after retries, skipping signal`, err);
1023
+ return false;
1024
+ }
1025
+ }
1026
+ if (this.payloadProposedCache.has(cacheKey)) {
1027
+ this.log.info(`Payload ${payload} was already proposed to governance, stopping signals`);
1028
+ return false;
1029
+ }
496
1030
  const cachedLastVote = this.lastActions[signalType];
497
1031
  this.lastActions[signalType] = slotNumber;
498
1032
  const action = signalType;
@@ -503,15 +1037,41 @@ export class SequencerPublisher {
503
1037
  signer: this.l1TxUtils.client.account?.address,
504
1038
  lastValidL2Slot: slotNumber
505
1039
  });
1040
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
506
1041
  try {
507
1042
  await this.l1TxUtils.simulate(request, {
508
1043
  time: timestamp
509
- }, [], ErrorsAbi);
1044
+ }, [], mergeAbis([
1045
+ request.abi ?? [],
1046
+ ErrorsAbi
1047
+ ]));
510
1048
  this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
511
1049
  request
512
1050
  });
513
1051
  } catch (err) {
514
- this.log.error(`Failed simulation for ${action} at slot ${slotNumber} (enqueuing the action anyway)`, err);
1052
+ const viemError = formatViemError(err);
1053
+ this.log.error(`Failed simulation for ${action} at slot ${slotNumber} (enqueuing the action anyway)`, viemError);
1054
+ this.backupFailedTx({
1055
+ id: keccak256(request.data),
1056
+ failureType: 'simulation',
1057
+ request: {
1058
+ to: request.to,
1059
+ data: request.data,
1060
+ value: request.value?.toString()
1061
+ },
1062
+ l1BlockNumber: l1BlockNumber.toString(),
1063
+ error: {
1064
+ message: viemError.message,
1065
+ name: viemError.name
1066
+ },
1067
+ context: {
1068
+ actions: [
1069
+ action
1070
+ ],
1071
+ slot: slotNumber,
1072
+ sender: this.getSenderAddress().toString()
1073
+ }
1074
+ });
515
1075
  // Yes, we enqueue the request anyway, in case there was a bug with the simulation itself
516
1076
  }
517
1077
  // TODO(palla/slash): All votes (governance and slashing) should txTimeoutAt at the end of the slot.
@@ -650,13 +1210,14 @@ export class SequencerPublisher {
650
1210
  /** Simulates and enqueues a proposal for a checkpoint on L1 */ async enqueueProposeCheckpoint(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
651
1211
  const checkpointHeader = checkpoint.header;
652
1212
  const blobFields = checkpoint.toBlobFields();
653
- const blobs = getBlobsPerL1Block(blobFields);
1213
+ const blobs = await getBlobsPerL1Block(blobFields);
654
1214
  const proposeTxArgs = {
655
1215
  header: checkpointHeader,
656
1216
  archive: checkpoint.archive.root.toBuffer(),
657
1217
  blobs,
658
1218
  attestationsAndSigners,
659
- attestationsAndSignersSignature
1219
+ attestationsAndSignersSignature,
1220
+ feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
660
1221
  };
661
1222
  let ts;
662
1223
  try {
@@ -670,7 +1231,7 @@ export class SequencerPublisher {
670
1231
  this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
671
1232
  ...checkpoint.getStats(),
672
1233
  slotNumber: checkpoint.header.slotNumber,
673
- forcePendingBlockNumber: opts.forcePendingBlockNumber
1234
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
674
1235
  });
675
1236
  throw err;
676
1237
  }
@@ -680,20 +1241,20 @@ export class SequencerPublisher {
680
1241
  });
681
1242
  await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
682
1243
  }
683
- enqueueInvalidateBlock(request, opts = {}) {
1244
+ enqueueInvalidateCheckpoint(request, opts = {}) {
684
1245
  if (!request) {
685
1246
  return;
686
1247
  }
687
1248
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
688
1249
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
689
- const { gasUsed, blockNumber } = request;
1250
+ const { gasUsed, checkpointNumber } = request;
690
1251
  const logData = {
691
1252
  gasUsed,
692
- blockNumber,
1253
+ checkpointNumber,
693
1254
  gasLimit,
694
1255
  opts
695
1256
  };
696
- this.log.verbose(`Enqueuing invalidate block request`, logData);
1257
+ this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
697
1258
  this.addRequest({
698
1259
  action: `invalidate-by-${request.reason}`,
699
1260
  request: request.request,
@@ -705,12 +1266,12 @@ export class SequencerPublisher {
705
1266
  checkSuccess: (_req, result)=>{
706
1267
  const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
707
1268
  if (!success) {
708
- this.log.warn(`Invalidate block ${request.blockNumber} failed`, {
1269
+ this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
709
1270
  ...result,
710
1271
  ...logData
711
1272
  });
712
1273
  } else {
713
- this.log.info(`Invalidate block ${request.blockNumber} succeeded`, {
1274
+ this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
714
1275
  ...result,
715
1276
  ...logData
716
1277
  });
@@ -732,28 +1293,60 @@ export class SequencerPublisher {
732
1293
  const cachedLastActionSlot = this.lastActions[action];
733
1294
  this.lastActions[action] = slotNumber;
734
1295
  this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
1296
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
735
1297
  let gasUsed;
1298
+ const simulateAbi = mergeAbis([
1299
+ request.abi ?? [],
1300
+ ErrorsAbi
1301
+ ]);
736
1302
  try {
737
1303
  ({ gasUsed } = await this.l1TxUtils.simulate(request, {
738
1304
  time: timestamp
739
- }, [], ErrorsAbi)); // TODO(palla/slash): Check the timestamp logic
1305
+ }, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
740
1306
  this.log.verbose(`Simulation for ${action} succeeded`, {
741
1307
  ...logData,
742
1308
  request,
743
1309
  gasUsed
744
1310
  });
745
1311
  } catch (err) {
746
- const viemError = formatViemError(err);
1312
+ const viemError = formatViemError(err, simulateAbi);
747
1313
  this.log.error(`Simulation for ${action} at ${slotNumber} failed`, viemError, logData);
1314
+ this.backupFailedTx({
1315
+ id: keccak256(request.data),
1316
+ failureType: 'simulation',
1317
+ request: {
1318
+ to: request.to,
1319
+ data: request.data,
1320
+ value: request.value?.toString()
1321
+ },
1322
+ l1BlockNumber: l1BlockNumber.toString(),
1323
+ error: {
1324
+ message: viemError.message,
1325
+ name: viemError.name
1326
+ },
1327
+ context: {
1328
+ actions: [
1329
+ action
1330
+ ],
1331
+ slot: slotNumber,
1332
+ sender: this.getSenderAddress().toString()
1333
+ }
1334
+ });
748
1335
  return false;
749
1336
  }
750
1337
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
751
1338
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
752
1339
  logData.gasLimit = gasLimit;
1340
+ // Store the ABI used for simulation on the request so Multicall3.forward can decode errors
1341
+ // when the tx is sent and a revert is diagnosed via simulation.
1342
+ const requestWithAbi = {
1343
+ ...request,
1344
+ abi: simulateAbi
1345
+ };
753
1346
  this.log.debug(`Enqueuing ${action}`, logData);
754
1347
  this.addRequest({
755
1348
  action,
756
- request,
1349
+ request: requestWithAbi,
757
1350
  gasConfig: {
758
1351
  gasLimit
759
1352
  },
@@ -817,10 +1410,38 @@ export class SequencerPublisher {
817
1410
  }, {}, {
818
1411
  blobs: encodedData.blobs.map((b)=>b.data),
819
1412
  kzg
820
- }).catch((err)=>{
821
- const { message, metaMessages } = formatViemError(err);
822
- this.log.error(`Failed to validate blobs`, message, {
823
- metaMessages
1413
+ }).catch(async (err)=>{
1414
+ const viemError = formatViemError(err);
1415
+ this.log.error(`Failed to validate blobs`, viemError.message, {
1416
+ metaMessages: viemError.metaMessages
1417
+ });
1418
+ const validateBlobsData = encodeFunctionData({
1419
+ abi: RollupAbi,
1420
+ functionName: 'validateBlobs',
1421
+ args: [
1422
+ blobInput
1423
+ ]
1424
+ });
1425
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
1426
+ this.backupFailedTx({
1427
+ id: keccak256(validateBlobsData),
1428
+ failureType: 'simulation',
1429
+ request: {
1430
+ to: this.rollupContract.address,
1431
+ data: validateBlobsData
1432
+ },
1433
+ blobData: encodedData.blobs.map((b)=>toHex(b.data)),
1434
+ l1BlockNumber: l1BlockNumber.toString(),
1435
+ error: {
1436
+ message: viemError.message,
1437
+ name: viemError.name
1438
+ },
1439
+ context: {
1440
+ actions: [
1441
+ 'validate-blobs'
1442
+ ],
1443
+ sender: this.getSenderAddress().toString()
1444
+ }
824
1445
  });
825
1446
  throw new Error('Failed to validate blobs');
826
1447
  });
@@ -831,8 +1452,7 @@ export class SequencerPublisher {
831
1452
  header: encodedData.header.toViem(),
832
1453
  archive: toHex(encodedData.archive),
833
1454
  oracleInput: {
834
- // We are currently not modifying these. See #9963
835
- feeAssetPriceModifier: 0n
1455
+ feeAssetPriceModifier: encodedData.feeAssetPriceModifier
836
1456
  }
837
1457
  },
838
1458
  encodedData.attestationsAndSigners.getPackedAttestations(),
@@ -860,8 +1480,7 @@ export class SequencerPublisher {
860
1480
  args
861
1481
  });
862
1482
  // override the pending checkpoint number if requested
863
- const optsForcePendingCheckpointNumber = options.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(options.forcePendingBlockNumber) : undefined;
864
- const forcePendingCheckpointNumberStateDiff = (optsForcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(optsForcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
1483
+ const forcePendingCheckpointNumberStateDiff = (options.forcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
865
1484
  const stateOverrides = [
866
1485
  {
867
1486
  address: this.rollupContract.address,
@@ -882,10 +1501,11 @@ export class SequencerPublisher {
882
1501
  balance: 10n * WEI_CONST * WEI_CONST
883
1502
  });
884
1503
  }
1504
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
885
1505
  const simulationResult = await this.l1TxUtils.simulate({
886
1506
  to: this.rollupContract.address,
887
1507
  data: rollupData,
888
- gas: SequencerPublisher.PROPOSE_GAS_GUESS,
1508
+ gas: MAX_L1_TX_LIMIT,
889
1509
  ...this.proposerAddressForSimulation && {
890
1510
  from: this.proposerAddressForSimulation.toString()
891
1511
  }
@@ -893,10 +1513,10 @@ export class SequencerPublisher {
893
1513
  // @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
894
1514
  time: timestamp + 1n,
895
1515
  // @note reth should have a 30m gas limit per block but throws errors that this tx is beyond limit so we increase here
896
- gasLimit: SequencerPublisher.PROPOSE_GAS_GUESS * 2n
1516
+ gasLimit: MAX_L1_TX_LIMIT * 2n
897
1517
  }, stateOverrides, RollupAbi, {
898
1518
  // @note fallback gas estimate to use if the node doesn't support simulation API
899
- fallbackGasEstimate: SequencerPublisher.PROPOSE_GAS_GUESS
1519
+ fallbackGasEstimate: MAX_L1_TX_LIMIT
900
1520
  }).catch((err)=>{
901
1521
  // In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
902
1522
  const viemError = formatViemError(err);
@@ -904,11 +1524,31 @@ export class SequencerPublisher {
904
1524
  this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
905
1525
  // Return a minimal simulation result with the fallback gas estimate
906
1526
  return {
907
- gasUsed: SequencerPublisher.PROPOSE_GAS_GUESS,
1527
+ gasUsed: MAX_L1_TX_LIMIT,
908
1528
  logs: []
909
1529
  };
910
1530
  }
911
1531
  this.log.error(`Failed to simulate propose tx`, viemError);
1532
+ this.backupFailedTx({
1533
+ id: keccak256(rollupData),
1534
+ failureType: 'simulation',
1535
+ request: {
1536
+ to: this.rollupContract.address,
1537
+ data: rollupData
1538
+ },
1539
+ l1BlockNumber: l1BlockNumber.toString(),
1540
+ error: {
1541
+ message: viemError.message,
1542
+ name: viemError.name
1543
+ },
1544
+ context: {
1545
+ actions: [
1546
+ 'propose'
1547
+ ],
1548
+ slot: Number(args[0].header.slotNumber),
1549
+ sender: this.getSenderAddress().toString()
1550
+ }
1551
+ });
912
1552
  throw err;
913
1553
  });
914
1554
  return {