@aztec/sequencer-client 0.0.1-commit.03f7ef2 → 0.0.1-commit.04852196a

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 (106) hide show
  1. package/dest/client/sequencer-client.d.ts +26 -11
  2. package/dest/client/sequencer-client.d.ts.map +1 -1
  3. package/dest/client/sequencer-client.js +99 -16
  4. package/dest/config.d.ts +24 -6
  5. package/dest/config.d.ts.map +1 -1
  6. package/dest/config.js +45 -28
  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 +27 -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 +44 -25
  37. package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
  38. package/dest/publisher/sequencer-publisher.js +781 -101
  39. package/dest/sequencer/checkpoint_proposal_job.d.ts +39 -13
  40. package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -1
  41. package/dest/sequencer/checkpoint_proposal_job.js +683 -79
  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 +46 -23
  52. package/dest/sequencer/sequencer.d.ts.map +1 -1
  53. package/dest/sequencer/sequencer.js +514 -67
  54. package/dest/sequencer/timetable.d.ts +4 -6
  55. package/dest/sequencer/timetable.d.ts.map +1 -1
  56. package/dest/sequencer/timetable.js +7 -11
  57. package/dest/sequencer/types.d.ts +5 -2
  58. package/dest/sequencer/types.d.ts.map +1 -1
  59. package/dest/test/index.d.ts +4 -7
  60. package/dest/test/index.d.ts.map +1 -1
  61. package/dest/test/mock_checkpoint_builder.d.ts +28 -16
  62. package/dest/test/mock_checkpoint_builder.d.ts.map +1 -1
  63. package/dest/test/mock_checkpoint_builder.js +86 -34
  64. package/dest/test/utils.d.ts +13 -9
  65. package/dest/test/utils.d.ts.map +1 -1
  66. package/dest/test/utils.js +27 -17
  67. package/package.json +30 -28
  68. package/src/client/sequencer-client.ts +139 -23
  69. package/src/config.ts +59 -38
  70. package/src/global_variable_builder/global_builder.ts +14 -14
  71. package/src/index.ts +1 -9
  72. package/src/publisher/config.ts +121 -43
  73. package/src/publisher/index.ts +3 -0
  74. package/src/publisher/l1_tx_failed_store/factory.ts +32 -0
  75. package/src/publisher/l1_tx_failed_store/failed_tx_store.ts +55 -0
  76. package/src/publisher/l1_tx_failed_store/file_store_failed_tx_store.ts +46 -0
  77. package/src/publisher/l1_tx_failed_store/index.ts +3 -0
  78. package/src/publisher/sequencer-publisher-factory.ts +39 -7
  79. package/src/publisher/sequencer-publisher-metrics.ts +17 -69
  80. package/src/publisher/sequencer-publisher.ts +420 -137
  81. package/src/sequencer/checkpoint_proposal_job.ts +361 -104
  82. package/src/sequencer/checkpoint_voter.ts +32 -7
  83. package/src/sequencer/index.ts +0 -2
  84. package/src/sequencer/metrics.ts +132 -148
  85. package/src/sequencer/sequencer.ts +160 -69
  86. package/src/sequencer/timetable.ts +13 -12
  87. package/src/sequencer/types.ts +4 -1
  88. package/src/test/index.ts +3 -6
  89. package/src/test/mock_checkpoint_builder.ts +147 -71
  90. package/src/test/utils.ts +58 -28
  91. package/dest/sequencer/block_builder.d.ts +0 -26
  92. package/dest/sequencer/block_builder.d.ts.map +0 -1
  93. package/dest/sequencer/block_builder.js +0 -129
  94. package/dest/sequencer/checkpoint_builder.d.ts +0 -63
  95. package/dest/sequencer/checkpoint_builder.d.ts.map +0 -1
  96. package/dest/sequencer/checkpoint_builder.js +0 -131
  97. package/dest/tx_validator/nullifier_cache.d.ts +0 -14
  98. package/dest/tx_validator/nullifier_cache.d.ts.map +0 -1
  99. package/dest/tx_validator/nullifier_cache.js +0 -24
  100. package/dest/tx_validator/tx_validator_factory.d.ts +0 -18
  101. package/dest/tx_validator/tx_validator_factory.d.ts.map +0 -1
  102. package/dest/tx_validator/tx_validator_factory.js +0 -53
  103. package/src/sequencer/block_builder.ts +0 -217
  104. package/src/sequencer/checkpoint_builder.ts +0 -217
  105. package/src/tx_validator/nullifier_cache.ts +0 -30
  106. package/src/tx_validator/tx_validator_factory.ts +0 -133
@@ -1,23 +1,398 @@
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';
383
+ import { TimeoutError } from '@aztec/foundation/error';
11
384
  import { EthAddress } from '@aztec/foundation/eth-address';
12
385
  import { Signature } from '@aztec/foundation/eth-signature';
13
386
  import { createLogger } from '@aztec/foundation/log';
387
+ import { makeBackoff, retry } from '@aztec/foundation/retry';
14
388
  import { bufferToHex } from '@aztec/foundation/string';
15
389
  import { Timer } from '@aztec/foundation/timer';
16
390
  import { EmpireBaseAbi, ErrorsAbi, RollupAbi } from '@aztec/l1-artifacts';
17
391
  import { encodeSlashConsensusVotes } from '@aztec/slasher';
18
392
  import { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
19
- import { getTelemetryClient } from '@aztec/telemetry-client';
20
- import { encodeFunctionData, toHex } from 'viem';
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';
21
396
  import { SequencerPublisherMetrics } from './sequencer-publisher-metrics.js';
22
397
  export const Actions = [
23
398
  'invalidate-by-invalid-attestation',
@@ -32,24 +407,44 @@ export const Actions = [
32
407
  ];
33
408
  // Sorting for actions such that invalidations go before proposals, and proposals go before votes
34
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');
35
411
  export class SequencerPublisher {
36
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
+ }
37
432
  interrupted;
38
433
  metrics;
39
434
  epochCache;
435
+ failedTxStore;
40
436
  governanceLog;
41
437
  slashingLog;
42
438
  lastActions;
43
439
  isPayloadEmptyCache;
440
+ payloadProposedCache;
44
441
  log;
45
442
  ethereumSlotDuration;
46
443
  blobClient;
47
444
  /** Address to use for simulations in fisherman mode (actual proposer's address) */ proposerAddressForSimulation;
445
+ /** Optional callback to obtain a replacement publisher when the current one fails to send. */ getNextPublisher;
48
446
  /** 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;
447
+ /** Fee asset price oracle for computing price modifiers from Uniswap V4 */ feeAssetPriceOracle;
53
448
  // A CALL to a cold address is 2700 gas
54
449
  static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
55
450
  // Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
@@ -59,25 +454,27 @@ export class SequencerPublisher {
59
454
  govProposerContract;
60
455
  slashingProposerContract;
61
456
  slashFactoryContract;
457
+ tracer;
62
458
  requests;
63
459
  constructor(config, deps){
64
460
  this.config = config;
65
- this.interrupted = false;
461
+ this.interrupted = (_initProto(this), false);
66
462
  this.governanceLog = createLogger('sequencer:publisher:governance');
67
463
  this.slashingLog = createLogger('sequencer:publisher:slashing');
68
464
  this.lastActions = {};
69
465
  this.isPayloadEmptyCache = new Map();
466
+ this.payloadProposedCache = new Set();
70
467
  this.requests = [];
71
468
  this.log = deps.log ?? createLogger('sequencer:publisher');
72
469
  this.ethereumSlotDuration = BigInt(config.ethereumSlotDuration);
73
470
  this.epochCache = deps.epochCache;
74
471
  this.lastActions = deps.lastActions;
75
- this.blobClient = deps.blobClient ?? createBlobClient(config, {
76
- logger: createLogger('sequencer:blob-client:client')
77
- });
472
+ this.blobClient = deps.blobClient;
78
473
  const telemetry = deps.telemetry ?? getTelemetryClient();
79
474
  this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
475
+ this.tracer = telemetry.getTracer('SequencerPublisher');
80
476
  this.l1TxUtils = deps.l1TxUtils;
477
+ this.getNextPublisher = deps.getNextPublisher;
81
478
  this.rollupContract = deps.rollupContract;
82
479
  this.govProposerContract = deps.governanceProposerContract;
83
480
  this.slashingProposerContract = deps.slashingProposerContract;
@@ -91,10 +488,36 @@ export class SequencerPublisher {
91
488
  if (config.fishermanMode) {
92
489
  this.l1FeeAnalyzer = new L1FeeAnalyzer(this.l1TxUtils.client, deps.dateProvider, createLogger('sequencer:publisher:fee-analyzer'));
93
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
+ });
94
511
  }
95
512
  getRollupContract() {
96
513
  return this.rollupContract;
97
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
+ }
98
521
  getSenderAddress() {
99
522
  return this.l1TxUtils.getSenderAddress();
100
523
  }
@@ -152,7 +575,7 @@ export class SequencerPublisher {
152
575
  // Get the transaction requests
153
576
  const l1Requests = requestsToAnalyze.map((r)=>r.request);
154
577
  // Start the analysis
155
- const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : SequencerPublisher.PROPOSE_GAS_GUESS, l1Requests, blobConfig, onComplete);
578
+ const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
156
579
  this.log.info('Started L1 fee analysis', {
157
580
  analysisId,
158
581
  l2SlotNumber: l2SlotNumber.toString(),
@@ -210,7 +633,16 @@ export class SequencerPublisher {
210
633
  const blobConfig = blobConfigs[0];
211
634
  // Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
212
635
  const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
213
- 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
+ }
214
646
  const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
215
647
  const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
216
648
  const txConfig = {
@@ -221,12 +653,34 @@ export class SequencerPublisher {
221
653
  // This ensures the committee gets precomputed correctly
222
654
  validRequests.sort((a, b)=>compareActions(a.action, b.action));
223
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
+ };
224
675
  this.log.debug('Forwarding transactions', {
225
676
  validRequests: validRequests.map((request)=>request.action),
226
677
  txConfig
227
678
  });
228
- 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);
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);
230
684
  return {
231
685
  result,
232
686
  expiredActions,
@@ -246,10 +700,67 @@ export class SequencerPublisher {
246
700
  }
247
701
  }
248
702
  }
249
- 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) {
250
738
  const actionsListStr = requests.map((r)=>r.action).join(', ');
251
739
  if (result instanceof FormattedViemError) {
252
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
+ });
253
764
  return {
254
765
  failedActions: requests.map((r)=>r.action)
255
766
  };
@@ -267,6 +778,37 @@ export class SequencerPublisher {
267
778
  failedActions.push(request.action);
268
779
  }
269
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
+ }
270
812
  return {
271
813
  successfulActions,
272
814
  failedActions
@@ -285,7 +827,7 @@ export class SequencerPublisher {
285
827
  'InvalidArchive'
286
828
  ];
287
829
  return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
288
- forcePendingCheckpointNumber: opts.forcePendingBlockNumber !== undefined ? CheckpointNumber.fromBlockNumber(opts.forcePendingBlockNumber) : undefined
830
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
289
831
  }).catch((err)=>{
290
832
  if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
291
833
  this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
@@ -313,12 +855,11 @@ export class SequencerPublisher {
313
855
  [],
314
856
  Signature.empty().toViemSignature(),
315
857
  `0x${'0'.repeat(64)}`,
316
- header.contentCommitment.blobsHash.toString(),
858
+ header.blobsHash.toString(),
317
859
  flags
318
860
  ];
319
861
  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);
862
+ const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
322
863
  let balance = 0n;
323
864
  if (this.config.fishermanMode) {
324
865
  // In fisherman mode, we can't know where the proposer is publishing from
@@ -345,34 +886,38 @@ export class SequencerPublisher {
345
886
  this.log.debug(`Simulated validateHeader`);
346
887
  }
347
888
  /**
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) {
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) {
351
892
  if (validationResult.valid) {
352
893
  return undefined;
353
894
  }
354
- const { reason, block } = validationResult;
355
- const blockNumber = block.blockNumber;
895
+ const { reason, checkpoint } = validationResult;
896
+ const checkpointNumber = checkpoint.checkpointNumber;
356
897
  const logData = {
357
- ...block,
898
+ ...checkpoint,
358
899
  reason
359
900
  };
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,
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,
364
905
  ...logData
365
906
  });
366
907
  return undefined;
367
908
  }
368
- const request = this.buildInvalidateBlockRequest(validationResult);
369
- this.log.debug(`Simulating invalidate block ${blockNumber}`, {
909
+ const request = this.buildInvalidateCheckpointRequest(validationResult);
910
+ this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
370
911
  ...logData,
371
912
  request
372
913
  });
914
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
373
915
  try {
374
- const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, ErrorsAbi);
375
- 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`, {
376
921
  ...logData,
377
922
  request,
378
923
  gasUsed
@@ -380,55 +925,76 @@ export class SequencerPublisher {
380
925
  return {
381
926
  request,
382
927
  gasUsed,
383
- blockNumber,
384
- forcePendingBlockNumber: BlockNumber(blockNumber - 1),
928
+ checkpointNumber,
929
+ forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
385
930
  reason
386
931
  };
387
932
  } catch (err) {
388
933
  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`, {
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`, {
393
938
  ...logData,
394
939
  request,
395
940
  error: viemError.message
396
941
  });
397
- const latestPendingBlockNumber = await this.rollupContract.getCheckpointNumber();
398
- if (latestPendingBlockNumber < blockNumber) {
399
- 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`, {
400
945
  ...logData
401
946
  });
402
947
  return undefined;
403
948
  } 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`, {
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`, {
406
951
  cause: viemError
407
952
  });
408
953
  }
409
954
  }
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}`, {
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}`, {
413
979
  cause: viemError
414
980
  });
415
981
  }
416
982
  }
417
- buildInvalidateBlockRequest(validationResult) {
983
+ buildInvalidateCheckpointRequest(validationResult) {
418
984
  if (validationResult.valid) {
419
- throw new Error('Cannot invalidate a valid block');
985
+ throw new Error('Cannot invalidate a valid checkpoint');
420
986
  }
421
- const { block, committee, reason } = validationResult;
987
+ const { checkpoint, committee, reason } = validationResult;
422
988
  const logData = {
423
- ...block,
989
+ ...checkpoint,
424
990
  reason
425
991
  };
426
- this.log.debug(`Simulating invalidate block ${block.blockNumber}`, logData);
992
+ this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
427
993
  const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
428
994
  if (reason === 'invalid-attestation') {
429
- return this.rollupContract.buildInvalidateBadAttestationRequest(CheckpointNumber.fromBlockNumber(block.blockNumber), attestationsAndSigners, committee, validationResult.invalidIndex);
995
+ return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
430
996
  } else if (reason === 'insufficient-attestations') {
431
- return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(CheckpointNumber.fromBlockNumber(block.blockNumber), attestationsAndSigners, committee);
997
+ return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
432
998
  } else {
433
999
  const _ = reason;
434
1000
  throw new Error(`Unknown reason for invalidation`);
@@ -436,29 +1002,15 @@ export class SequencerPublisher {
436
1002
  }
437
1003
  /** Simulates `propose` to make sure that the checkpoint is valid for submission */ async validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, options) {
438
1004
  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
1005
  const blobFields = checkpoint.toBlobFields();
454
- const blobs = getBlobsPerL1Block(blobFields);
1006
+ const blobs = await getBlobsPerL1Block(blobFields);
455
1007
  const blobInput = getPrefixedEthBlobCommitments(blobs);
456
1008
  const args = [
457
1009
  {
458
1010
  header: checkpoint.header.toViem(),
459
1011
  archive: toHex(checkpoint.archive.root.toBuffer()),
460
1012
  oracleInput: {
461
- feeAssetPriceModifier: 0n
1013
+ feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
462
1014
  }
463
1015
  },
464
1016
  attestationsAndSigners.getPackedAttestations(),
@@ -493,6 +1045,28 @@ export class SequencerPublisher {
493
1045
  this.log.warn(`Skipping vote cast for payload with empty code`);
494
1046
  return false;
495
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
+ }
496
1070
  const cachedLastVote = this.lastActions[signalType];
497
1071
  this.lastActions[signalType] = slotNumber;
498
1072
  const action = signalType;
@@ -503,15 +1077,41 @@ export class SequencerPublisher {
503
1077
  signer: this.l1TxUtils.client.account?.address,
504
1078
  lastValidL2Slot: slotNumber
505
1079
  });
1080
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
506
1081
  try {
507
1082
  await this.l1TxUtils.simulate(request, {
508
1083
  time: timestamp
509
- }, [], ErrorsAbi);
1084
+ }, [], mergeAbis([
1085
+ request.abi ?? [],
1086
+ ErrorsAbi
1087
+ ]));
510
1088
  this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
511
1089
  request
512
1090
  });
513
1091
  } catch (err) {
514
- 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
+ });
515
1115
  // Yes, we enqueue the request anyway, in case there was a bug with the simulation itself
516
1116
  }
517
1117
  // TODO(palla/slash): All votes (governance and slashing) should txTimeoutAt at the end of the slot.
@@ -650,13 +1250,14 @@ export class SequencerPublisher {
650
1250
  /** Simulates and enqueues a proposal for a checkpoint on L1 */ async enqueueProposeCheckpoint(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
651
1251
  const checkpointHeader = checkpoint.header;
652
1252
  const blobFields = checkpoint.toBlobFields();
653
- const blobs = getBlobsPerL1Block(blobFields);
1253
+ const blobs = await getBlobsPerL1Block(blobFields);
654
1254
  const proposeTxArgs = {
655
1255
  header: checkpointHeader,
656
1256
  archive: checkpoint.archive.root.toBuffer(),
657
1257
  blobs,
658
1258
  attestationsAndSigners,
659
- attestationsAndSignersSignature
1259
+ attestationsAndSignersSignature,
1260
+ feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
660
1261
  };
661
1262
  let ts;
662
1263
  try {
@@ -670,7 +1271,7 @@ export class SequencerPublisher {
670
1271
  this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
671
1272
  ...checkpoint.getStats(),
672
1273
  slotNumber: checkpoint.header.slotNumber,
673
- forcePendingBlockNumber: opts.forcePendingBlockNumber
1274
+ forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
674
1275
  });
675
1276
  throw err;
676
1277
  }
@@ -680,20 +1281,20 @@ export class SequencerPublisher {
680
1281
  });
681
1282
  await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
682
1283
  }
683
- enqueueInvalidateBlock(request, opts = {}) {
1284
+ enqueueInvalidateCheckpoint(request, opts = {}) {
684
1285
  if (!request) {
685
1286
  return;
686
1287
  }
687
1288
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
688
1289
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
689
- const { gasUsed, blockNumber } = request;
1290
+ const { gasUsed, checkpointNumber } = request;
690
1291
  const logData = {
691
1292
  gasUsed,
692
- blockNumber,
1293
+ checkpointNumber,
693
1294
  gasLimit,
694
1295
  opts
695
1296
  };
696
- this.log.verbose(`Enqueuing invalidate block request`, logData);
1297
+ this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
697
1298
  this.addRequest({
698
1299
  action: `invalidate-by-${request.reason}`,
699
1300
  request: request.request,
@@ -705,12 +1306,12 @@ export class SequencerPublisher {
705
1306
  checkSuccess: (_req, result)=>{
706
1307
  const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
707
1308
  if (!success) {
708
- this.log.warn(`Invalidate block ${request.blockNumber} failed`, {
1309
+ this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
709
1310
  ...result,
710
1311
  ...logData
711
1312
  });
712
1313
  } else {
713
- this.log.info(`Invalidate block ${request.blockNumber} succeeded`, {
1314
+ this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
714
1315
  ...result,
715
1316
  ...logData
716
1317
  });
@@ -732,28 +1333,60 @@ export class SequencerPublisher {
732
1333
  const cachedLastActionSlot = this.lastActions[action];
733
1334
  this.lastActions[action] = slotNumber;
734
1335
  this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
1336
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
735
1337
  let gasUsed;
1338
+ const simulateAbi = mergeAbis([
1339
+ request.abi ?? [],
1340
+ ErrorsAbi
1341
+ ]);
736
1342
  try {
737
1343
  ({ gasUsed } = await this.l1TxUtils.simulate(request, {
738
1344
  time: timestamp
739
- }, [], ErrorsAbi)); // TODO(palla/slash): Check the timestamp logic
1345
+ }, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
740
1346
  this.log.verbose(`Simulation for ${action} succeeded`, {
741
1347
  ...logData,
742
1348
  request,
743
1349
  gasUsed
744
1350
  });
745
1351
  } catch (err) {
746
- const viemError = formatViemError(err);
1352
+ const viemError = formatViemError(err, simulateAbi);
747
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
+ });
748
1375
  return false;
749
1376
  }
750
1377
  // We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
751
1378
  const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
752
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
+ };
753
1386
  this.log.debug(`Enqueuing ${action}`, logData);
754
1387
  this.addRequest({
755
1388
  action,
756
- request,
1389
+ request: requestWithAbi,
757
1390
  gasConfig: {
758
1391
  gasLimit
759
1392
  },
@@ -817,10 +1450,38 @@ export class SequencerPublisher {
817
1450
  }, {}, {
818
1451
  blobs: encodedData.blobs.map((b)=>b.data),
819
1452
  kzg
820
- }).catch((err)=>{
821
- const { message, metaMessages } = formatViemError(err);
822
- this.log.error(`Failed to validate blobs`, message, {
823
- 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
+ }
824
1485
  });
825
1486
  throw new Error('Failed to validate blobs');
826
1487
  });
@@ -831,8 +1492,7 @@ export class SequencerPublisher {
831
1492
  header: encodedData.header.toViem(),
832
1493
  archive: toHex(encodedData.archive),
833
1494
  oracleInput: {
834
- // We are currently not modifying these. See #9963
835
- feeAssetPriceModifier: 0n
1495
+ feeAssetPriceModifier: encodedData.feeAssetPriceModifier
836
1496
  }
837
1497
  },
838
1498
  encodedData.attestationsAndSigners.getPackedAttestations(),
@@ -860,8 +1520,7 @@ export class SequencerPublisher {
860
1520
  args
861
1521
  });
862
1522
  // 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 ?? []);
1523
+ const forcePendingCheckpointNumberStateDiff = (options.forcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
865
1524
  const stateOverrides = [
866
1525
  {
867
1526
  address: this.rollupContract.address,
@@ -882,10 +1541,11 @@ export class SequencerPublisher {
882
1541
  balance: 10n * WEI_CONST * WEI_CONST
883
1542
  });
884
1543
  }
1544
+ const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
885
1545
  const simulationResult = await this.l1TxUtils.simulate({
886
1546
  to: this.rollupContract.address,
887
1547
  data: rollupData,
888
- gas: SequencerPublisher.PROPOSE_GAS_GUESS,
1548
+ gas: MAX_L1_TX_LIMIT,
889
1549
  ...this.proposerAddressForSimulation && {
890
1550
  from: this.proposerAddressForSimulation.toString()
891
1551
  }
@@ -893,10 +1553,10 @@ export class SequencerPublisher {
893
1553
  // @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
894
1554
  time: timestamp + 1n,
895
1555
  // @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
1556
+ gasLimit: MAX_L1_TX_LIMIT * 2n
897
1557
  }, stateOverrides, RollupAbi, {
898
1558
  // @note fallback gas estimate to use if the node doesn't support simulation API
899
- fallbackGasEstimate: SequencerPublisher.PROPOSE_GAS_GUESS
1559
+ fallbackGasEstimate: MAX_L1_TX_LIMIT
900
1560
  }).catch((err)=>{
901
1561
  // In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
902
1562
  const viemError = formatViemError(err);
@@ -904,11 +1564,31 @@ export class SequencerPublisher {
904
1564
  this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
905
1565
  // Return a minimal simulation result with the fallback gas estimate
906
1566
  return {
907
- gasUsed: SequencerPublisher.PROPOSE_GAS_GUESS,
1567
+ gasUsed: MAX_L1_TX_LIMIT,
908
1568
  logs: []
909
1569
  };
910
1570
  }
911
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
+ });
912
1592
  throw err;
913
1593
  });
914
1594
  return {