@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.
- package/dest/client/sequencer-client.d.ts +15 -11
- package/dest/client/sequencer-client.d.ts.map +1 -1
- package/dest/client/sequencer-client.js +15 -4
- package/dest/config.d.ts +3 -4
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +22 -12
- package/dest/global_variable_builder/global_builder.d.ts +5 -7
- package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
- package/dest/global_variable_builder/global_builder.js +13 -13
- package/dest/index.d.ts +2 -3
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +1 -2
- package/dest/publisher/config.d.ts +35 -17
- package/dest/publisher/config.d.ts.map +1 -1
- package/dest/publisher/config.js +106 -42
- package/dest/publisher/index.d.ts +2 -1
- package/dest/publisher/index.d.ts.map +1 -1
- package/dest/publisher/l1_tx_failed_store/factory.d.ts +11 -0
- package/dest/publisher/l1_tx_failed_store/factory.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/factory.js +22 -0
- package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts +59 -0
- package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/failed_tx_store.js +1 -0
- package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts +15 -0
- package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.js +34 -0
- package/dest/publisher/l1_tx_failed_store/index.d.ts +4 -0
- package/dest/publisher/l1_tx_failed_store/index.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/index.js +2 -0
- package/dest/publisher/sequencer-publisher-factory.d.ts +12 -4
- package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher-factory.js +13 -2
- package/dest/publisher/sequencer-publisher-metrics.d.ts +1 -1
- package/dest/publisher/sequencer-publisher-metrics.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher-metrics.js +23 -86
- package/dest/publisher/sequencer-publisher.d.ts +40 -25
- package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher.js +740 -100
- package/dest/sequencer/checkpoint_proposal_job.d.ts +40 -12
- package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -1
- package/dest/sequencer/checkpoint_proposal_job.js +643 -64
- package/dest/sequencer/checkpoint_voter.d.ts +3 -2
- package/dest/sequencer/checkpoint_voter.d.ts.map +1 -1
- package/dest/sequencer/checkpoint_voter.js +34 -10
- package/dest/sequencer/index.d.ts +1 -3
- package/dest/sequencer/index.d.ts.map +1 -1
- package/dest/sequencer/index.js +0 -2
- package/dest/sequencer/metrics.d.ts +19 -7
- package/dest/sequencer/metrics.d.ts.map +1 -1
- package/dest/sequencer/metrics.js +131 -141
- package/dest/sequencer/sequencer.d.ts +38 -18
- package/dest/sequencer/sequencer.d.ts.map +1 -1
- package/dest/sequencer/sequencer.js +513 -66
- package/dest/sequencer/timetable.d.ts +1 -4
- package/dest/sequencer/timetable.d.ts.map +1 -1
- package/dest/sequencer/timetable.js +1 -4
- package/dest/test/index.d.ts +4 -7
- package/dest/test/index.d.ts.map +1 -1
- package/dest/test/mock_checkpoint_builder.d.ts +25 -11
- package/dest/test/mock_checkpoint_builder.d.ts.map +1 -1
- package/dest/test/mock_checkpoint_builder.js +52 -9
- package/dest/test/utils.d.ts +13 -9
- package/dest/test/utils.d.ts.map +1 -1
- package/dest/test/utils.js +27 -17
- package/package.json +30 -28
- package/src/client/sequencer-client.ts +29 -12
- package/src/config.ts +31 -19
- package/src/global_variable_builder/global_builder.ts +14 -14
- package/src/index.ts +1 -9
- package/src/publisher/config.ts +121 -43
- package/src/publisher/index.ts +3 -0
- package/src/publisher/l1_tx_failed_store/factory.ts +32 -0
- package/src/publisher/l1_tx_failed_store/failed_tx_store.ts +55 -0
- package/src/publisher/l1_tx_failed_store/file_store_failed_tx_store.ts +46 -0
- package/src/publisher/l1_tx_failed_store/index.ts +3 -0
- package/src/publisher/sequencer-publisher-factory.ts +24 -7
- package/src/publisher/sequencer-publisher-metrics.ts +17 -69
- package/src/publisher/sequencer-publisher.ts +361 -130
- package/src/sequencer/checkpoint_proposal_job.ts +316 -93
- package/src/sequencer/checkpoint_voter.ts +32 -7
- package/src/sequencer/index.ts +0 -2
- package/src/sequencer/metrics.ts +132 -148
- package/src/sequencer/sequencer.ts +159 -68
- package/src/sequencer/timetable.ts +6 -5
- package/src/test/index.ts +3 -6
- package/src/test/mock_checkpoint_builder.ts +102 -29
- package/src/test/utils.ts +58 -28
- package/dest/sequencer/block_builder.d.ts +0 -26
- package/dest/sequencer/block_builder.d.ts.map +0 -1
- package/dest/sequencer/block_builder.js +0 -129
- package/dest/sequencer/checkpoint_builder.d.ts +0 -63
- package/dest/sequencer/checkpoint_builder.d.ts.map +0 -1
- package/dest/sequencer/checkpoint_builder.js +0 -131
- package/dest/tx_validator/nullifier_cache.d.ts +0 -14
- package/dest/tx_validator/nullifier_cache.d.ts.map +0 -1
- package/dest/tx_validator/nullifier_cache.js +0 -24
- package/dest/tx_validator/tx_validator_factory.d.ts +0 -18
- package/dest/tx_validator/tx_validator_factory.d.ts.map +0 -1
- package/dest/tx_validator/tx_validator_factory.js +0 -53
- package/src/sequencer/block_builder.ts +0 -217
- package/src/sequencer/checkpoint_builder.ts +0 -217
- package/src/tx_validator/nullifier_cache.ts +0 -30
- package/src/tx_validator/tx_validator_factory.ts +0 -133
|
@@ -1,23 +1,397 @@
|
|
|
1
|
-
|
|
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 {
|
|
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
|
-
|
|
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
|
|
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 :
|
|
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
|
-
|
|
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
|
|
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.
|
|
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.
|
|
818
|
+
header.blobsHash.toString(),
|
|
317
819
|
flags
|
|
318
820
|
];
|
|
319
821
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
320
|
-
const
|
|
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
|
|
349
|
-
* @param
|
|
350
|
-
*/ async
|
|
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,
|
|
355
|
-
const
|
|
855
|
+
const { reason, checkpoint } = validationResult;
|
|
856
|
+
const checkpointNumber = checkpoint.checkpointNumber;
|
|
356
857
|
const logData = {
|
|
357
|
-
...
|
|
858
|
+
...checkpoint,
|
|
358
859
|
reason
|
|
359
860
|
};
|
|
360
|
-
const
|
|
361
|
-
if (
|
|
362
|
-
this.log.verbose(`Skipping
|
|
363
|
-
|
|
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.
|
|
369
|
-
this.log.debug(`Simulating invalidate
|
|
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,
|
|
375
|
-
|
|
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
|
-
|
|
384
|
-
|
|
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
|
|
390
|
-
// we can safely ignore it and return undefined so we go ahead with
|
|
391
|
-
if (viemError.message?.includes('
|
|
392
|
-
this.log.verbose(`Simulation for invalidate
|
|
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
|
|
398
|
-
if (
|
|
399
|
-
this.log.verbose(`
|
|
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 ${
|
|
405
|
-
throw new Error(`Failed to simulate invalidate
|
|
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
|
|
411
|
-
this.log.error(`Simulation for invalidate
|
|
412
|
-
|
|
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
|
-
|
|
943
|
+
buildInvalidateCheckpointRequest(validationResult) {
|
|
418
944
|
if (validationResult.valid) {
|
|
419
|
-
throw new Error('Cannot invalidate a valid
|
|
945
|
+
throw new Error('Cannot invalidate a valid checkpoint');
|
|
420
946
|
}
|
|
421
|
-
const {
|
|
947
|
+
const { checkpoint, committee, reason } = validationResult;
|
|
422
948
|
const logData = {
|
|
423
|
-
...
|
|
949
|
+
...checkpoint,
|
|
424
950
|
reason
|
|
425
951
|
};
|
|
426
|
-
this.log.debug(`
|
|
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(
|
|
955
|
+
return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
|
|
430
956
|
} else if (reason === 'insufficient-attestations') {
|
|
431
|
-
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(
|
|
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:
|
|
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
|
-
}, [],
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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,
|
|
1250
|
+
const { gasUsed, checkpointNumber } = request;
|
|
690
1251
|
const logData = {
|
|
691
1252
|
gasUsed,
|
|
692
|
-
|
|
1253
|
+
checkpointNumber,
|
|
693
1254
|
gasLimit,
|
|
694
1255
|
opts
|
|
695
1256
|
};
|
|
696
|
-
this.log.verbose(`Enqueuing invalidate
|
|
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
|
|
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
|
|
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
|
-
}, [],
|
|
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
|
|
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
|
-
|
|
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
|
|
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:
|
|
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:
|
|
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:
|
|
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:
|
|
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 {
|