@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.
- package/dest/client/sequencer-client.d.ts +26 -11
- package/dest/client/sequencer-client.d.ts.map +1 -1
- package/dest/client/sequencer-client.js +99 -16
- package/dest/config.d.ts +24 -6
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +45 -28
- 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 +27 -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 +44 -25
- package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher.js +781 -101
- package/dest/sequencer/checkpoint_proposal_job.d.ts +39 -13
- package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -1
- package/dest/sequencer/checkpoint_proposal_job.js +683 -79
- 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 +46 -23
- package/dest/sequencer/sequencer.d.ts.map +1 -1
- package/dest/sequencer/sequencer.js +514 -67
- package/dest/sequencer/timetable.d.ts +4 -6
- package/dest/sequencer/timetable.d.ts.map +1 -1
- package/dest/sequencer/timetable.js +7 -11
- package/dest/sequencer/types.d.ts +5 -2
- package/dest/sequencer/types.d.ts.map +1 -1
- 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 +28 -16
- package/dest/test/mock_checkpoint_builder.d.ts.map +1 -1
- package/dest/test/mock_checkpoint_builder.js +86 -34
- 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 +139 -23
- package/src/config.ts +59 -38
- 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 +39 -7
- package/src/publisher/sequencer-publisher-metrics.ts +17 -69
- package/src/publisher/sequencer-publisher.ts +420 -137
- package/src/sequencer/checkpoint_proposal_job.ts +361 -104
- 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 +160 -69
- package/src/sequencer/timetable.ts +13 -12
- package/src/sequencer/types.ts +4 -1
- package/src/test/index.ts +3 -6
- package/src/test/mock_checkpoint_builder.ts +147 -71
- 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,398 @@
|
|
|
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';
|
|
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
|
-
|
|
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
|
|
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 :
|
|
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
|
-
|
|
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
|
|
229
|
-
|
|
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
|
-
|
|
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.
|
|
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.
|
|
858
|
+
header.blobsHash.toString(),
|
|
317
859
|
flags
|
|
318
860
|
];
|
|
319
861
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
320
|
-
const
|
|
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
|
|
349
|
-
* @param
|
|
350
|
-
*/ async
|
|
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,
|
|
355
|
-
const
|
|
895
|
+
const { reason, checkpoint } = validationResult;
|
|
896
|
+
const checkpointNumber = checkpoint.checkpointNumber;
|
|
356
897
|
const logData = {
|
|
357
|
-
...
|
|
898
|
+
...checkpoint,
|
|
358
899
|
reason
|
|
359
900
|
};
|
|
360
|
-
const
|
|
361
|
-
if (
|
|
362
|
-
this.log.verbose(`Skipping
|
|
363
|
-
|
|
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.
|
|
369
|
-
this.log.debug(`Simulating invalidate
|
|
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,
|
|
375
|
-
|
|
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
|
-
|
|
384
|
-
|
|
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
|
|
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
|
|
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
|
|
398
|
-
if (
|
|
399
|
-
this.log.verbose(`
|
|
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 ${
|
|
405
|
-
throw new Error(`Failed to simulate invalidate
|
|
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
|
|
411
|
-
this.log.error(`Simulation for invalidate
|
|
412
|
-
|
|
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
|
-
|
|
983
|
+
buildInvalidateCheckpointRequest(validationResult) {
|
|
418
984
|
if (validationResult.valid) {
|
|
419
|
-
throw new Error('Cannot invalidate a valid
|
|
985
|
+
throw new Error('Cannot invalidate a valid checkpoint');
|
|
420
986
|
}
|
|
421
|
-
const {
|
|
987
|
+
const { checkpoint, committee, reason } = validationResult;
|
|
422
988
|
const logData = {
|
|
423
|
-
...
|
|
989
|
+
...checkpoint,
|
|
424
990
|
reason
|
|
425
991
|
};
|
|
426
|
-
this.log.debug(`
|
|
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(
|
|
995
|
+
return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
|
|
430
996
|
} else if (reason === 'insufficient-attestations') {
|
|
431
|
-
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(
|
|
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:
|
|
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
|
-
}, [],
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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,
|
|
1290
|
+
const { gasUsed, checkpointNumber } = request;
|
|
690
1291
|
const logData = {
|
|
691
1292
|
gasUsed,
|
|
692
|
-
|
|
1293
|
+
checkpointNumber,
|
|
693
1294
|
gasLimit,
|
|
694
1295
|
opts
|
|
695
1296
|
};
|
|
696
|
-
this.log.verbose(`Enqueuing invalidate
|
|
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
|
|
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
|
|
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
|
-
}, [],
|
|
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
|
|
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
|
-
|
|
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
|
|
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:
|
|
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:
|
|
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:
|
|
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:
|
|
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 {
|