@aztec/sequencer-client 0.0.1-commit.fce3e4f → 0.0.1-commit.ff7989d6c
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 +21 -16
- package/dest/client/sequencer-client.d.ts.map +1 -1
- package/dest/client/sequencer-client.js +45 -27
- package/dest/config.d.ts +14 -8
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +83 -35
- package/dest/global_variable_builder/global_builder.d.ts +20 -13
- package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
- package/dest/global_variable_builder/global_builder.js +51 -41
- 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 +37 -20
- package/dest/publisher/config.d.ts.map +1 -1
- package/dest/publisher/config.js +104 -39
- package/dest/publisher/sequencer-publisher-factory.d.ts +15 -6
- package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher-factory.js +14 -3
- package/dest/publisher/sequencer-publisher-metrics.d.ts +3 -3
- 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 +63 -47
- package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher.js +630 -137
- package/dest/sequencer/checkpoint_proposal_job.d.ts +102 -0
- package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -0
- package/dest/sequencer/checkpoint_proposal_job.js +1213 -0
- package/dest/sequencer/checkpoint_voter.d.ts +35 -0
- package/dest/sequencer/checkpoint_voter.d.ts.map +1 -0
- package/dest/sequencer/checkpoint_voter.js +109 -0
- package/dest/sequencer/config.d.ts +3 -2
- package/dest/sequencer/config.d.ts.map +1 -1
- package/dest/sequencer/events.d.ts +46 -0
- package/dest/sequencer/events.d.ts.map +1 -0
- package/dest/sequencer/events.js +1 -0
- package/dest/sequencer/index.d.ts +4 -2
- package/dest/sequencer/index.d.ts.map +1 -1
- package/dest/sequencer/index.js +3 -1
- package/dest/sequencer/metrics.d.ts +38 -6
- package/dest/sequencer/metrics.d.ts.map +1 -1
- package/dest/sequencer/metrics.js +216 -72
- package/dest/sequencer/sequencer.d.ts +119 -133
- package/dest/sequencer/sequencer.d.ts.map +1 -1
- package/dest/sequencer/sequencer.js +717 -625
- package/dest/sequencer/timetable.d.ts +51 -14
- package/dest/sequencer/timetable.d.ts.map +1 -1
- package/dest/sequencer/timetable.js +145 -59
- package/dest/sequencer/types.d.ts +3 -0
- package/dest/sequencer/types.d.ts.map +1 -0
- package/dest/sequencer/types.js +1 -0
- package/dest/sequencer/utils.d.ts +14 -8
- package/dest/sequencer/utils.d.ts.map +1 -1
- package/dest/sequencer/utils.js +7 -4
- package/dest/test/index.d.ts +6 -7
- package/dest/test/index.d.ts.map +1 -1
- package/dest/test/mock_checkpoint_builder.d.ts +97 -0
- package/dest/test/mock_checkpoint_builder.d.ts.map +1 -0
- package/dest/test/mock_checkpoint_builder.js +222 -0
- package/dest/test/utils.d.ts +53 -0
- package/dest/test/utils.d.ts.map +1 -0
- package/dest/test/utils.js +104 -0
- package/package.json +32 -30
- package/src/client/sequencer-client.ts +54 -47
- package/src/config.ts +95 -44
- package/src/global_variable_builder/global_builder.ts +65 -61
- package/src/index.ts +1 -7
- package/src/publisher/config.ts +122 -50
- package/src/publisher/sequencer-publisher-factory.ts +28 -10
- package/src/publisher/sequencer-publisher-metrics.ts +19 -71
- package/src/publisher/sequencer-publisher.ts +350 -176
- package/src/sequencer/README.md +531 -0
- package/src/sequencer/checkpoint_proposal_job.ts +914 -0
- package/src/sequencer/checkpoint_voter.ts +130 -0
- package/src/sequencer/config.ts +2 -1
- package/src/sequencer/events.ts +27 -0
- package/src/sequencer/index.ts +3 -1
- package/src/sequencer/metrics.ts +268 -82
- package/src/sequencer/sequencer.ts +464 -831
- package/src/sequencer/timetable.ts +175 -80
- package/src/sequencer/types.ts +6 -0
- package/src/sequencer/utils.ts +18 -9
- package/src/test/index.ts +5 -6
- package/src/test/mock_checkpoint_builder.ts +320 -0
- package/src/test/utils.ts +167 -0
- package/dest/sequencer/block_builder.d.ts +0 -27
- package/dest/sequencer/block_builder.d.ts.map +0 -1
- package/dest/sequencer/block_builder.js +0 -134
- 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 -17
- 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 -222
- package/src/tx_validator/nullifier_cache.ts +0 -30
- package/src/tx_validator/tx_validator_factory.ts +0 -132
|
@@ -1,18 +1,395 @@
|
|
|
1
|
+
function applyDecs2203RFactory() {
|
|
2
|
+
function createAddInitializerMethod(initializers, decoratorFinishedRef) {
|
|
3
|
+
return function addInitializer(initializer) {
|
|
4
|
+
assertNotFinished(decoratorFinishedRef, "addInitializer");
|
|
5
|
+
assertCallable(initializer, "An initializer");
|
|
6
|
+
initializers.push(initializer);
|
|
7
|
+
};
|
|
8
|
+
}
|
|
9
|
+
function memberDec(dec, name, desc, initializers, kind, isStatic, isPrivate, metadata, value) {
|
|
10
|
+
var kindStr;
|
|
11
|
+
switch(kind){
|
|
12
|
+
case 1:
|
|
13
|
+
kindStr = "accessor";
|
|
14
|
+
break;
|
|
15
|
+
case 2:
|
|
16
|
+
kindStr = "method";
|
|
17
|
+
break;
|
|
18
|
+
case 3:
|
|
19
|
+
kindStr = "getter";
|
|
20
|
+
break;
|
|
21
|
+
case 4:
|
|
22
|
+
kindStr = "setter";
|
|
23
|
+
break;
|
|
24
|
+
default:
|
|
25
|
+
kindStr = "field";
|
|
26
|
+
}
|
|
27
|
+
var ctx = {
|
|
28
|
+
kind: kindStr,
|
|
29
|
+
name: isPrivate ? "#" + name : name,
|
|
30
|
+
static: isStatic,
|
|
31
|
+
private: isPrivate,
|
|
32
|
+
metadata: metadata
|
|
33
|
+
};
|
|
34
|
+
var decoratorFinishedRef = {
|
|
35
|
+
v: false
|
|
36
|
+
};
|
|
37
|
+
ctx.addInitializer = createAddInitializerMethod(initializers, decoratorFinishedRef);
|
|
38
|
+
var get, set;
|
|
39
|
+
if (kind === 0) {
|
|
40
|
+
if (isPrivate) {
|
|
41
|
+
get = desc.get;
|
|
42
|
+
set = desc.set;
|
|
43
|
+
} else {
|
|
44
|
+
get = function() {
|
|
45
|
+
return this[name];
|
|
46
|
+
};
|
|
47
|
+
set = function(v) {
|
|
48
|
+
this[name] = v;
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
} else if (kind === 2) {
|
|
52
|
+
get = function() {
|
|
53
|
+
return desc.value;
|
|
54
|
+
};
|
|
55
|
+
} else {
|
|
56
|
+
if (kind === 1 || kind === 3) {
|
|
57
|
+
get = function() {
|
|
58
|
+
return desc.get.call(this);
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
if (kind === 1 || kind === 4) {
|
|
62
|
+
set = function(v) {
|
|
63
|
+
desc.set.call(this, v);
|
|
64
|
+
};
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
ctx.access = get && set ? {
|
|
68
|
+
get: get,
|
|
69
|
+
set: set
|
|
70
|
+
} : get ? {
|
|
71
|
+
get: get
|
|
72
|
+
} : {
|
|
73
|
+
set: set
|
|
74
|
+
};
|
|
75
|
+
try {
|
|
76
|
+
return dec(value, ctx);
|
|
77
|
+
} finally{
|
|
78
|
+
decoratorFinishedRef.v = true;
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
function assertNotFinished(decoratorFinishedRef, fnName) {
|
|
82
|
+
if (decoratorFinishedRef.v) {
|
|
83
|
+
throw new Error("attempted to call " + fnName + " after decoration was finished");
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
function assertCallable(fn, hint) {
|
|
87
|
+
if (typeof fn !== "function") {
|
|
88
|
+
throw new TypeError(hint + " must be a function");
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
function assertValidReturnValue(kind, value) {
|
|
92
|
+
var type = typeof value;
|
|
93
|
+
if (kind === 1) {
|
|
94
|
+
if (type !== "object" || value === null) {
|
|
95
|
+
throw new TypeError("accessor decorators must return an object with get, set, or init properties or void 0");
|
|
96
|
+
}
|
|
97
|
+
if (value.get !== undefined) {
|
|
98
|
+
assertCallable(value.get, "accessor.get");
|
|
99
|
+
}
|
|
100
|
+
if (value.set !== undefined) {
|
|
101
|
+
assertCallable(value.set, "accessor.set");
|
|
102
|
+
}
|
|
103
|
+
if (value.init !== undefined) {
|
|
104
|
+
assertCallable(value.init, "accessor.init");
|
|
105
|
+
}
|
|
106
|
+
} else if (type !== "function") {
|
|
107
|
+
var hint;
|
|
108
|
+
if (kind === 0) {
|
|
109
|
+
hint = "field";
|
|
110
|
+
} else if (kind === 10) {
|
|
111
|
+
hint = "class";
|
|
112
|
+
} else {
|
|
113
|
+
hint = "method";
|
|
114
|
+
}
|
|
115
|
+
throw new TypeError(hint + " decorators must return a function or void 0");
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
function applyMemberDec(ret, base, decInfo, name, kind, isStatic, isPrivate, initializers, metadata) {
|
|
119
|
+
var decs = decInfo[0];
|
|
120
|
+
var desc, init, value;
|
|
121
|
+
if (isPrivate) {
|
|
122
|
+
if (kind === 0 || kind === 1) {
|
|
123
|
+
desc = {
|
|
124
|
+
get: decInfo[3],
|
|
125
|
+
set: decInfo[4]
|
|
126
|
+
};
|
|
127
|
+
} else if (kind === 3) {
|
|
128
|
+
desc = {
|
|
129
|
+
get: decInfo[3]
|
|
130
|
+
};
|
|
131
|
+
} else if (kind === 4) {
|
|
132
|
+
desc = {
|
|
133
|
+
set: decInfo[3]
|
|
134
|
+
};
|
|
135
|
+
} else {
|
|
136
|
+
desc = {
|
|
137
|
+
value: decInfo[3]
|
|
138
|
+
};
|
|
139
|
+
}
|
|
140
|
+
} else if (kind !== 0) {
|
|
141
|
+
desc = Object.getOwnPropertyDescriptor(base, name);
|
|
142
|
+
}
|
|
143
|
+
if (kind === 1) {
|
|
144
|
+
value = {
|
|
145
|
+
get: desc.get,
|
|
146
|
+
set: desc.set
|
|
147
|
+
};
|
|
148
|
+
} else if (kind === 2) {
|
|
149
|
+
value = desc.value;
|
|
150
|
+
} else if (kind === 3) {
|
|
151
|
+
value = desc.get;
|
|
152
|
+
} else if (kind === 4) {
|
|
153
|
+
value = desc.set;
|
|
154
|
+
}
|
|
155
|
+
var newValue, get, set;
|
|
156
|
+
if (typeof decs === "function") {
|
|
157
|
+
newValue = memberDec(decs, name, desc, initializers, kind, isStatic, isPrivate, metadata, value);
|
|
158
|
+
if (newValue !== void 0) {
|
|
159
|
+
assertValidReturnValue(kind, newValue);
|
|
160
|
+
if (kind === 0) {
|
|
161
|
+
init = newValue;
|
|
162
|
+
} else if (kind === 1) {
|
|
163
|
+
init = newValue.init;
|
|
164
|
+
get = newValue.get || value.get;
|
|
165
|
+
set = newValue.set || value.set;
|
|
166
|
+
value = {
|
|
167
|
+
get: get,
|
|
168
|
+
set: set
|
|
169
|
+
};
|
|
170
|
+
} else {
|
|
171
|
+
value = newValue;
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
} else {
|
|
175
|
+
for(var i = decs.length - 1; i >= 0; i--){
|
|
176
|
+
var dec = decs[i];
|
|
177
|
+
newValue = memberDec(dec, name, desc, initializers, kind, isStatic, isPrivate, metadata, value);
|
|
178
|
+
if (newValue !== void 0) {
|
|
179
|
+
assertValidReturnValue(kind, newValue);
|
|
180
|
+
var newInit;
|
|
181
|
+
if (kind === 0) {
|
|
182
|
+
newInit = newValue;
|
|
183
|
+
} else if (kind === 1) {
|
|
184
|
+
newInit = newValue.init;
|
|
185
|
+
get = newValue.get || value.get;
|
|
186
|
+
set = newValue.set || value.set;
|
|
187
|
+
value = {
|
|
188
|
+
get: get,
|
|
189
|
+
set: set
|
|
190
|
+
};
|
|
191
|
+
} else {
|
|
192
|
+
value = newValue;
|
|
193
|
+
}
|
|
194
|
+
if (newInit !== void 0) {
|
|
195
|
+
if (init === void 0) {
|
|
196
|
+
init = newInit;
|
|
197
|
+
} else if (typeof init === "function") {
|
|
198
|
+
init = [
|
|
199
|
+
init,
|
|
200
|
+
newInit
|
|
201
|
+
];
|
|
202
|
+
} else {
|
|
203
|
+
init.push(newInit);
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
if (kind === 0 || kind === 1) {
|
|
210
|
+
if (init === void 0) {
|
|
211
|
+
init = function(instance, init) {
|
|
212
|
+
return init;
|
|
213
|
+
};
|
|
214
|
+
} else if (typeof init !== "function") {
|
|
215
|
+
var ownInitializers = init;
|
|
216
|
+
init = function(instance, init) {
|
|
217
|
+
var value = init;
|
|
218
|
+
for(var i = 0; i < ownInitializers.length; i++){
|
|
219
|
+
value = ownInitializers[i].call(instance, value);
|
|
220
|
+
}
|
|
221
|
+
return value;
|
|
222
|
+
};
|
|
223
|
+
} else {
|
|
224
|
+
var originalInitializer = init;
|
|
225
|
+
init = function(instance, init) {
|
|
226
|
+
return originalInitializer.call(instance, init);
|
|
227
|
+
};
|
|
228
|
+
}
|
|
229
|
+
ret.push(init);
|
|
230
|
+
}
|
|
231
|
+
if (kind !== 0) {
|
|
232
|
+
if (kind === 1) {
|
|
233
|
+
desc.get = value.get;
|
|
234
|
+
desc.set = value.set;
|
|
235
|
+
} else if (kind === 2) {
|
|
236
|
+
desc.value = value;
|
|
237
|
+
} else if (kind === 3) {
|
|
238
|
+
desc.get = value;
|
|
239
|
+
} else if (kind === 4) {
|
|
240
|
+
desc.set = value;
|
|
241
|
+
}
|
|
242
|
+
if (isPrivate) {
|
|
243
|
+
if (kind === 1) {
|
|
244
|
+
ret.push(function(instance, args) {
|
|
245
|
+
return value.get.call(instance, args);
|
|
246
|
+
});
|
|
247
|
+
ret.push(function(instance, args) {
|
|
248
|
+
return value.set.call(instance, args);
|
|
249
|
+
});
|
|
250
|
+
} else if (kind === 2) {
|
|
251
|
+
ret.push(value);
|
|
252
|
+
} else {
|
|
253
|
+
ret.push(function(instance, args) {
|
|
254
|
+
return value.call(instance, args);
|
|
255
|
+
});
|
|
256
|
+
}
|
|
257
|
+
} else {
|
|
258
|
+
Object.defineProperty(base, name, desc);
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
function applyMemberDecs(Class, decInfos, metadata) {
|
|
263
|
+
var ret = [];
|
|
264
|
+
var protoInitializers;
|
|
265
|
+
var staticInitializers;
|
|
266
|
+
var existingProtoNonFields = new Map();
|
|
267
|
+
var existingStaticNonFields = new Map();
|
|
268
|
+
for(var i = 0; i < decInfos.length; i++){
|
|
269
|
+
var decInfo = decInfos[i];
|
|
270
|
+
if (!Array.isArray(decInfo)) continue;
|
|
271
|
+
var kind = decInfo[1];
|
|
272
|
+
var name = decInfo[2];
|
|
273
|
+
var isPrivate = decInfo.length > 3;
|
|
274
|
+
var isStatic = kind >= 5;
|
|
275
|
+
var base;
|
|
276
|
+
var initializers;
|
|
277
|
+
if (isStatic) {
|
|
278
|
+
base = Class;
|
|
279
|
+
kind = kind - 5;
|
|
280
|
+
staticInitializers = staticInitializers || [];
|
|
281
|
+
initializers = staticInitializers;
|
|
282
|
+
} else {
|
|
283
|
+
base = Class.prototype;
|
|
284
|
+
protoInitializers = protoInitializers || [];
|
|
285
|
+
initializers = protoInitializers;
|
|
286
|
+
}
|
|
287
|
+
if (kind !== 0 && !isPrivate) {
|
|
288
|
+
var existingNonFields = isStatic ? existingStaticNonFields : existingProtoNonFields;
|
|
289
|
+
var existingKind = existingNonFields.get(name) || 0;
|
|
290
|
+
if (existingKind === true || existingKind === 3 && kind !== 4 || existingKind === 4 && kind !== 3) {
|
|
291
|
+
throw new Error("Attempted to decorate a public method/accessor that has the same name as a previously decorated public method/accessor. This is not currently supported by the decorators plugin. Property name was: " + name);
|
|
292
|
+
} else if (!existingKind && kind > 2) {
|
|
293
|
+
existingNonFields.set(name, kind);
|
|
294
|
+
} else {
|
|
295
|
+
existingNonFields.set(name, true);
|
|
296
|
+
}
|
|
297
|
+
}
|
|
298
|
+
applyMemberDec(ret, base, decInfo, name, kind, isStatic, isPrivate, initializers, metadata);
|
|
299
|
+
}
|
|
300
|
+
pushInitializers(ret, protoInitializers);
|
|
301
|
+
pushInitializers(ret, staticInitializers);
|
|
302
|
+
return ret;
|
|
303
|
+
}
|
|
304
|
+
function pushInitializers(ret, initializers) {
|
|
305
|
+
if (initializers) {
|
|
306
|
+
ret.push(function(instance) {
|
|
307
|
+
for(var i = 0; i < initializers.length; i++){
|
|
308
|
+
initializers[i].call(instance);
|
|
309
|
+
}
|
|
310
|
+
return instance;
|
|
311
|
+
});
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
function applyClassDecs(targetClass, classDecs, metadata) {
|
|
315
|
+
if (classDecs.length > 0) {
|
|
316
|
+
var initializers = [];
|
|
317
|
+
var newClass = targetClass;
|
|
318
|
+
var name = targetClass.name;
|
|
319
|
+
for(var i = classDecs.length - 1; i >= 0; i--){
|
|
320
|
+
var decoratorFinishedRef = {
|
|
321
|
+
v: false
|
|
322
|
+
};
|
|
323
|
+
try {
|
|
324
|
+
var nextNewClass = classDecs[i](newClass, {
|
|
325
|
+
kind: "class",
|
|
326
|
+
name: name,
|
|
327
|
+
addInitializer: createAddInitializerMethod(initializers, decoratorFinishedRef),
|
|
328
|
+
metadata
|
|
329
|
+
});
|
|
330
|
+
} finally{
|
|
331
|
+
decoratorFinishedRef.v = true;
|
|
332
|
+
}
|
|
333
|
+
if (nextNewClass !== undefined) {
|
|
334
|
+
assertValidReturnValue(10, nextNewClass);
|
|
335
|
+
newClass = nextNewClass;
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
return [
|
|
339
|
+
defineMetadata(newClass, metadata),
|
|
340
|
+
function() {
|
|
341
|
+
for(var i = 0; i < initializers.length; i++){
|
|
342
|
+
initializers[i].call(newClass);
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
];
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
function defineMetadata(Class, metadata) {
|
|
349
|
+
return Object.defineProperty(Class, Symbol.metadata || Symbol.for("Symbol.metadata"), {
|
|
350
|
+
configurable: true,
|
|
351
|
+
enumerable: true,
|
|
352
|
+
value: metadata
|
|
353
|
+
});
|
|
354
|
+
}
|
|
355
|
+
return function applyDecs2203R(targetClass, memberDecs, classDecs, parentClass) {
|
|
356
|
+
if (parentClass !== void 0) {
|
|
357
|
+
var parentMetadata = parentClass[Symbol.metadata || Symbol.for("Symbol.metadata")];
|
|
358
|
+
}
|
|
359
|
+
var metadata = Object.create(parentMetadata === void 0 ? null : parentMetadata);
|
|
360
|
+
var e = applyMemberDecs(targetClass, memberDecs, metadata);
|
|
361
|
+
if (!classDecs.length) defineMetadata(targetClass, metadata);
|
|
362
|
+
return {
|
|
363
|
+
e: e,
|
|
364
|
+
get c () {
|
|
365
|
+
return applyClassDecs(targetClass, classDecs, metadata);
|
|
366
|
+
}
|
|
367
|
+
};
|
|
368
|
+
};
|
|
369
|
+
}
|
|
370
|
+
function _apply_decs_2203_r(targetClass, memberDecs, classDecs, parentClass) {
|
|
371
|
+
return (_apply_decs_2203_r = applyDecs2203RFactory())(targetClass, memberDecs, classDecs, parentClass);
|
|
372
|
+
}
|
|
373
|
+
var _dec, _dec1, _dec2, _initProto;
|
|
1
374
|
import { Blob, getBlobsPerL1Block, getPrefixedEthBlobCommitments } from '@aztec/blob-lib';
|
|
2
|
-
import {
|
|
3
|
-
import {
|
|
375
|
+
import { FeeAssetPriceOracle, MULTI_CALL_3_ADDRESS, Multicall3, RollupContract } from '@aztec/ethereum/contracts';
|
|
376
|
+
import { L1FeeAnalyzer } from '@aztec/ethereum/l1-fee-analysis';
|
|
377
|
+
import { MAX_L1_TX_LIMIT, WEI_CONST } from '@aztec/ethereum/l1-tx-utils';
|
|
378
|
+
import { FormattedViemError, formatViemError, mergeAbis, tryExtractEvent } from '@aztec/ethereum/utils';
|
|
4
379
|
import { sumBigint } from '@aztec/foundation/bigint';
|
|
5
380
|
import { toHex as toPaddedHex } from '@aztec/foundation/bigint-buffer';
|
|
6
|
-
import { SlotNumber } from '@aztec/foundation/branded-types';
|
|
381
|
+
import { CheckpointNumber, SlotNumber } from '@aztec/foundation/branded-types';
|
|
382
|
+
import { pick } from '@aztec/foundation/collection';
|
|
7
383
|
import { EthAddress } from '@aztec/foundation/eth-address';
|
|
8
384
|
import { Signature } from '@aztec/foundation/eth-signature';
|
|
9
385
|
import { createLogger } from '@aztec/foundation/log';
|
|
386
|
+
import { makeBackoff, retry } from '@aztec/foundation/retry';
|
|
10
387
|
import { bufferToHex } from '@aztec/foundation/string';
|
|
11
388
|
import { Timer } from '@aztec/foundation/timer';
|
|
12
389
|
import { EmpireBaseAbi, ErrorsAbi, RollupAbi } from '@aztec/l1-artifacts';
|
|
13
390
|
import { encodeSlashConsensusVotes } from '@aztec/slasher';
|
|
14
|
-
import {
|
|
15
|
-
import { getTelemetryClient } from '@aztec/telemetry-client';
|
|
391
|
+
import { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
|
|
392
|
+
import { getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
|
|
16
393
|
import { encodeFunctionData, toHex } from 'viem';
|
|
17
394
|
import { SequencerPublisherMetrics } from './sequencer-publisher-metrics.js';
|
|
18
395
|
export const Actions = [
|
|
@@ -28,22 +405,42 @@ export const Actions = [
|
|
|
28
405
|
];
|
|
29
406
|
// Sorting for actions such that invalidations go before proposals, and proposals go before votes
|
|
30
407
|
export const compareActions = (a, b)=>Actions.indexOf(a) - Actions.indexOf(b);
|
|
408
|
+
_dec = trackSpan('SequencerPublisher.sendRequests'), _dec1 = trackSpan('SequencerPublisher.validateBlockHeader'), _dec2 = trackSpan('SequencerPublisher.validateCheckpointForSubmission');
|
|
31
409
|
export class SequencerPublisher {
|
|
32
410
|
config;
|
|
411
|
+
static{
|
|
412
|
+
({ e: [_initProto] } = _apply_decs_2203_r(this, [
|
|
413
|
+
[
|
|
414
|
+
_dec,
|
|
415
|
+
2,
|
|
416
|
+
"sendRequests"
|
|
417
|
+
],
|
|
418
|
+
[
|
|
419
|
+
_dec1,
|
|
420
|
+
2,
|
|
421
|
+
"validateBlockHeader"
|
|
422
|
+
],
|
|
423
|
+
[
|
|
424
|
+
_dec2,
|
|
425
|
+
2,
|
|
426
|
+
"validateCheckpointForSubmission"
|
|
427
|
+
]
|
|
428
|
+
], []));
|
|
429
|
+
}
|
|
33
430
|
interrupted;
|
|
34
431
|
metrics;
|
|
35
432
|
epochCache;
|
|
36
433
|
governanceLog;
|
|
37
434
|
slashingLog;
|
|
38
435
|
lastActions;
|
|
436
|
+
isPayloadEmptyCache;
|
|
437
|
+
payloadProposedCache;
|
|
39
438
|
log;
|
|
40
439
|
ethereumSlotDuration;
|
|
41
|
-
|
|
440
|
+
blobClient;
|
|
42
441
|
/** Address to use for simulations in fisherman mode (actual proposer's address) */ proposerAddressForSimulation;
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
// Total used for emptier block from above test: 429k (of which 84k is 1x blob)
|
|
46
|
-
static PROPOSE_GAS_GUESS = 12_000_000n;
|
|
442
|
+
/** L1 fee analyzer for fisherman mode */ l1FeeAnalyzer;
|
|
443
|
+
/** Fee asset price oracle for computing price modifiers from Uniswap V4 */ feeAssetPriceOracle;
|
|
47
444
|
// A CALL to a cold address is 2700 gas
|
|
48
445
|
static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
|
|
49
446
|
// Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
|
|
@@ -53,23 +450,25 @@ export class SequencerPublisher {
|
|
|
53
450
|
govProposerContract;
|
|
54
451
|
slashingProposerContract;
|
|
55
452
|
slashFactoryContract;
|
|
453
|
+
tracer;
|
|
56
454
|
requests;
|
|
57
455
|
constructor(config, deps){
|
|
58
456
|
this.config = config;
|
|
59
|
-
this.interrupted = false;
|
|
457
|
+
this.interrupted = (_initProto(this), false);
|
|
60
458
|
this.governanceLog = createLogger('sequencer:publisher:governance');
|
|
61
459
|
this.slashingLog = createLogger('sequencer:publisher:slashing');
|
|
62
460
|
this.lastActions = {};
|
|
461
|
+
this.isPayloadEmptyCache = new Map();
|
|
462
|
+
this.payloadProposedCache = new Set();
|
|
63
463
|
this.requests = [];
|
|
64
464
|
this.log = deps.log ?? createLogger('sequencer:publisher');
|
|
65
465
|
this.ethereumSlotDuration = BigInt(config.ethereumSlotDuration);
|
|
66
466
|
this.epochCache = deps.epochCache;
|
|
67
467
|
this.lastActions = deps.lastActions;
|
|
68
|
-
this.
|
|
69
|
-
logger: createLogger('sequencer:blob-sink:client')
|
|
70
|
-
});
|
|
468
|
+
this.blobClient = deps.blobClient;
|
|
71
469
|
const telemetry = deps.telemetry ?? getTelemetryClient();
|
|
72
470
|
this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
|
|
471
|
+
this.tracer = telemetry.getTracer('SequencerPublisher');
|
|
73
472
|
this.l1TxUtils = deps.l1TxUtils;
|
|
74
473
|
this.rollupContract = deps.rollupContract;
|
|
75
474
|
this.govProposerContract = deps.governanceProposerContract;
|
|
@@ -80,14 +479,31 @@ export class SequencerPublisher {
|
|
|
80
479
|
this.slashingProposerContract = newSlashingProposer;
|
|
81
480
|
});
|
|
82
481
|
this.slashFactoryContract = deps.slashFactoryContract;
|
|
482
|
+
// Initialize L1 fee analyzer for fisherman mode
|
|
483
|
+
if (config.fishermanMode) {
|
|
484
|
+
this.l1FeeAnalyzer = new L1FeeAnalyzer(this.l1TxUtils.client, deps.dateProvider, createLogger('sequencer:publisher:fee-analyzer'));
|
|
485
|
+
}
|
|
486
|
+
// Initialize fee asset price oracle
|
|
487
|
+
this.feeAssetPriceOracle = new FeeAssetPriceOracle(this.l1TxUtils.client, this.rollupContract, createLogger('sequencer:publisher:price-oracle'));
|
|
83
488
|
}
|
|
84
489
|
getRollupContract() {
|
|
85
490
|
return this.rollupContract;
|
|
86
491
|
}
|
|
492
|
+
/**
|
|
493
|
+
* Gets the fee asset price modifier from the oracle.
|
|
494
|
+
* Returns 0n if the oracle query fails.
|
|
495
|
+
*/ getFeeAssetPriceModifier() {
|
|
496
|
+
return this.feeAssetPriceOracle.computePriceModifier();
|
|
497
|
+
}
|
|
87
498
|
getSenderAddress() {
|
|
88
499
|
return this.l1TxUtils.getSenderAddress();
|
|
89
500
|
}
|
|
90
501
|
/**
|
|
502
|
+
* Gets the L1 fee analyzer instance (only available in fisherman mode)
|
|
503
|
+
*/ getL1FeeAnalyzer() {
|
|
504
|
+
return this.l1FeeAnalyzer;
|
|
505
|
+
}
|
|
506
|
+
/**
|
|
91
507
|
* Sets the proposer address to use for simulations in fisherman mode.
|
|
92
508
|
* @param proposerAddress - The actual proposer's address to use for balance lookups in simulations
|
|
93
509
|
*/ setProposerAddressForSimulation(proposerAddress) {
|
|
@@ -109,6 +525,46 @@ export class SequencerPublisher {
|
|
|
109
525
|
}
|
|
110
526
|
}
|
|
111
527
|
/**
|
|
528
|
+
* Analyzes L1 fees for the pending requests without sending them.
|
|
529
|
+
* This is used in fisherman mode to validate fee calculations.
|
|
530
|
+
* @param l2SlotNumber - The L2 slot number for this analysis
|
|
531
|
+
* @param onComplete - Optional callback to invoke when analysis completes (after block is mined)
|
|
532
|
+
* @returns The analysis result (incomplete until block mines), or undefined if no requests
|
|
533
|
+
*/ async analyzeL1Fees(l2SlotNumber, onComplete) {
|
|
534
|
+
if (!this.l1FeeAnalyzer) {
|
|
535
|
+
this.log.warn('L1 fee analyzer not available (not in fisherman mode)');
|
|
536
|
+
return undefined;
|
|
537
|
+
}
|
|
538
|
+
const requestsToAnalyze = [
|
|
539
|
+
...this.requests
|
|
540
|
+
];
|
|
541
|
+
if (requestsToAnalyze.length === 0) {
|
|
542
|
+
this.log.debug('No requests to analyze for L1 fees');
|
|
543
|
+
return undefined;
|
|
544
|
+
}
|
|
545
|
+
// Extract blob config from requests (if any)
|
|
546
|
+
const blobConfigs = requestsToAnalyze.filter((request)=>request.blobConfig).map((request)=>request.blobConfig);
|
|
547
|
+
const blobConfig = blobConfigs[0];
|
|
548
|
+
// Get gas configs
|
|
549
|
+
const gasConfigs = requestsToAnalyze.filter((request)=>request.gasConfig).map((request)=>request.gasConfig);
|
|
550
|
+
const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
|
|
551
|
+
const gasLimit = gasLimits.length > 0 ? gasLimits.reduce((sum, g)=>sum + g, 0n) : 0n;
|
|
552
|
+
// Get the transaction requests
|
|
553
|
+
const l1Requests = requestsToAnalyze.map((r)=>r.request);
|
|
554
|
+
// Start the analysis
|
|
555
|
+
const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
|
|
556
|
+
this.log.info('Started L1 fee analysis', {
|
|
557
|
+
analysisId,
|
|
558
|
+
l2SlotNumber: l2SlotNumber.toString(),
|
|
559
|
+
requestCount: requestsToAnalyze.length,
|
|
560
|
+
hasBlobConfig: !!blobConfig,
|
|
561
|
+
gasLimit: gasLimit.toString(),
|
|
562
|
+
actions: requestsToAnalyze.map((r)=>r.action)
|
|
563
|
+
});
|
|
564
|
+
// Return the analysis result (will be incomplete until block mines)
|
|
565
|
+
return this.l1FeeAnalyzer.getAnalysis(analysisId);
|
|
566
|
+
}
|
|
567
|
+
/**
|
|
112
568
|
* Sends all requests that are still valid.
|
|
113
569
|
* @returns one of:
|
|
114
570
|
* - A receipt and stats if the tx succeeded
|
|
@@ -119,7 +575,7 @@ export class SequencerPublisher {
|
|
|
119
575
|
...this.requests
|
|
120
576
|
];
|
|
121
577
|
this.requests = [];
|
|
122
|
-
if (this.interrupted) {
|
|
578
|
+
if (this.interrupted || requestsToProcess.length === 0) {
|
|
123
579
|
return undefined;
|
|
124
580
|
}
|
|
125
581
|
const currentL2Slot = this.getCurrentL2Slot();
|
|
@@ -154,7 +610,16 @@ export class SequencerPublisher {
|
|
|
154
610
|
const blobConfig = blobConfigs[0];
|
|
155
611
|
// Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
|
|
156
612
|
const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
|
|
157
|
-
|
|
613
|
+
let gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
|
|
614
|
+
// Cap at L1 block gas limit so the node accepts the tx ("gas limit too high" otherwise).
|
|
615
|
+
const maxGas = MAX_L1_TX_LIMIT;
|
|
616
|
+
if (gasLimit !== undefined && gasLimit > maxGas) {
|
|
617
|
+
this.log.debug('Capping bundled tx gas limit to L1 max', {
|
|
618
|
+
requested: gasLimit,
|
|
619
|
+
capped: maxGas
|
|
620
|
+
});
|
|
621
|
+
gasLimit = maxGas;
|
|
622
|
+
}
|
|
158
623
|
const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
|
|
159
624
|
const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
|
|
160
625
|
const txConfig = {
|
|
@@ -229,7 +694,7 @@ export class SequencerPublisher {
|
|
|
229
694
|
'InvalidArchive'
|
|
230
695
|
];
|
|
231
696
|
return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
|
|
232
|
-
forcePendingCheckpointNumber: opts.
|
|
697
|
+
forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
|
|
233
698
|
}).catch((err)=>{
|
|
234
699
|
if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
|
|
235
700
|
this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
|
|
@@ -257,11 +722,11 @@ export class SequencerPublisher {
|
|
|
257
722
|
[],
|
|
258
723
|
Signature.empty().toViemSignature(),
|
|
259
724
|
`0x${'0'.repeat(64)}`,
|
|
260
|
-
header.
|
|
725
|
+
header.blobsHash.toString(),
|
|
261
726
|
flags
|
|
262
727
|
];
|
|
263
728
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
264
|
-
const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.
|
|
729
|
+
const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
|
|
265
730
|
let balance = 0n;
|
|
266
731
|
if (this.config.fishermanMode) {
|
|
267
732
|
// In fisherman mode, we can't know where the proposer is publishing from
|
|
@@ -288,34 +753,37 @@ export class SequencerPublisher {
|
|
|
288
753
|
this.log.debug(`Simulated validateHeader`);
|
|
289
754
|
}
|
|
290
755
|
/**
|
|
291
|
-
* Simulate making a call to invalidate a
|
|
292
|
-
* @param
|
|
293
|
-
*/ async
|
|
756
|
+
* Simulate making a call to invalidate a checkpoint with invalid attestations. Returns undefined if no need to invalidate.
|
|
757
|
+
* @param validationResult - The validation result indicating which checkpoint to invalidate (as returned by the archiver)
|
|
758
|
+
*/ async simulateInvalidateCheckpoint(validationResult) {
|
|
294
759
|
if (validationResult.valid) {
|
|
295
760
|
return undefined;
|
|
296
761
|
}
|
|
297
|
-
const { reason,
|
|
298
|
-
const
|
|
762
|
+
const { reason, checkpoint } = validationResult;
|
|
763
|
+
const checkpointNumber = checkpoint.checkpointNumber;
|
|
299
764
|
const logData = {
|
|
300
|
-
...
|
|
765
|
+
...checkpoint,
|
|
301
766
|
reason
|
|
302
767
|
};
|
|
303
|
-
const
|
|
304
|
-
if (
|
|
305
|
-
this.log.verbose(`Skipping
|
|
306
|
-
|
|
768
|
+
const currentCheckpointNumber = await this.rollupContract.getCheckpointNumber();
|
|
769
|
+
if (currentCheckpointNumber < checkpointNumber) {
|
|
770
|
+
this.log.verbose(`Skipping checkpoint ${checkpointNumber} invalidation since it has already been removed from the pending chain`, {
|
|
771
|
+
currentCheckpointNumber,
|
|
307
772
|
...logData
|
|
308
773
|
});
|
|
309
774
|
return undefined;
|
|
310
775
|
}
|
|
311
|
-
const request = this.
|
|
312
|
-
this.log.debug(`Simulating invalidate
|
|
776
|
+
const request = this.buildInvalidateCheckpointRequest(validationResult);
|
|
777
|
+
this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
|
|
313
778
|
...logData,
|
|
314
779
|
request
|
|
315
780
|
});
|
|
316
781
|
try {
|
|
317
|
-
const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined,
|
|
318
|
-
|
|
782
|
+
const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, mergeAbis([
|
|
783
|
+
request.abi ?? [],
|
|
784
|
+
ErrorsAbi
|
|
785
|
+
]));
|
|
786
|
+
this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} succeeded`, {
|
|
319
787
|
...logData,
|
|
320
788
|
request,
|
|
321
789
|
gasUsed
|
|
@@ -323,90 +791,71 @@ export class SequencerPublisher {
|
|
|
323
791
|
return {
|
|
324
792
|
request,
|
|
325
793
|
gasUsed,
|
|
326
|
-
|
|
327
|
-
|
|
794
|
+
checkpointNumber,
|
|
795
|
+
forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
|
|
328
796
|
reason
|
|
329
797
|
};
|
|
330
798
|
} catch (err) {
|
|
331
799
|
const viemError = formatViemError(err);
|
|
332
|
-
// If the error is due to the
|
|
333
|
-
// we can safely ignore it and return undefined so we go ahead with
|
|
334
|
-
if (viemError.message?.includes('
|
|
335
|
-
this.log.verbose(`Simulation for invalidate
|
|
800
|
+
// If the error is due to the checkpoint not being in the pending chain, and it was indeed removed by someone else,
|
|
801
|
+
// we can safely ignore it and return undefined so we go ahead with checkpoint building.
|
|
802
|
+
if (viemError.message?.includes('Rollup__CheckpointNotInPendingChain')) {
|
|
803
|
+
this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} failed due to checkpoint not being in pending chain`, {
|
|
336
804
|
...logData,
|
|
337
805
|
request,
|
|
338
806
|
error: viemError.message
|
|
339
807
|
});
|
|
340
|
-
const
|
|
341
|
-
if (
|
|
342
|
-
this.log.verbose(`
|
|
808
|
+
const latestPendingCheckpointNumber = await this.rollupContract.getCheckpointNumber();
|
|
809
|
+
if (latestPendingCheckpointNumber < checkpointNumber) {
|
|
810
|
+
this.log.verbose(`Checkpoint ${checkpointNumber} has already been invalidated`, {
|
|
343
811
|
...logData
|
|
344
812
|
});
|
|
345
813
|
return undefined;
|
|
346
814
|
} else {
|
|
347
|
-
this.log.error(`Simulation for invalidate ${
|
|
348
|
-
throw new Error(`Failed to simulate invalidate
|
|
815
|
+
this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed and it is still in pending chain`, viemError, logData);
|
|
816
|
+
throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber} while it is still in pending chain`, {
|
|
349
817
|
cause: viemError
|
|
350
818
|
});
|
|
351
819
|
}
|
|
352
820
|
}
|
|
353
|
-
// Otherwise, throw. We cannot build the next
|
|
354
|
-
this.log.error(`Simulation for invalidate
|
|
355
|
-
throw new Error(`Failed to simulate invalidate
|
|
821
|
+
// Otherwise, throw. We cannot build the next checkpoint if we cannot invalidate the previous one.
|
|
822
|
+
this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed`, viemError, logData);
|
|
823
|
+
throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber}`, {
|
|
356
824
|
cause: viemError
|
|
357
825
|
});
|
|
358
826
|
}
|
|
359
827
|
}
|
|
360
|
-
|
|
828
|
+
buildInvalidateCheckpointRequest(validationResult) {
|
|
361
829
|
if (validationResult.valid) {
|
|
362
|
-
throw new Error('Cannot invalidate a valid
|
|
830
|
+
throw new Error('Cannot invalidate a valid checkpoint');
|
|
363
831
|
}
|
|
364
|
-
const {
|
|
832
|
+
const { checkpoint, committee, reason } = validationResult;
|
|
365
833
|
const logData = {
|
|
366
|
-
...
|
|
834
|
+
...checkpoint,
|
|
367
835
|
reason
|
|
368
836
|
};
|
|
369
|
-
this.log.debug(`
|
|
837
|
+
this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
|
|
370
838
|
const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
|
|
371
839
|
if (reason === 'invalid-attestation') {
|
|
372
|
-
return this.rollupContract.buildInvalidateBadAttestationRequest(
|
|
840
|
+
return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
|
|
373
841
|
} else if (reason === 'insufficient-attestations') {
|
|
374
|
-
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(
|
|
842
|
+
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
|
|
375
843
|
} else {
|
|
376
844
|
const _ = reason;
|
|
377
845
|
throw new Error(`Unknown reason for invalidation`);
|
|
378
846
|
}
|
|
379
847
|
}
|
|
380
|
-
/**
|
|
381
|
-
* @notice Will simulate `propose` to make sure that the block is valid for submission
|
|
382
|
-
*
|
|
383
|
-
* @dev Throws if unable to propose
|
|
384
|
-
*
|
|
385
|
-
* @param block - The block to propose
|
|
386
|
-
* @param attestationData - The block's attestation data
|
|
387
|
-
*
|
|
388
|
-
*/ async validateBlockForSubmission(block, attestationsAndSigners, attestationsAndSignersSignature, options) {
|
|
848
|
+
/** Simulates `propose` to make sure that the checkpoint is valid for submission */ async validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, options) {
|
|
389
849
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
|
|
393
|
-
if (ignoreSignatures) {
|
|
394
|
-
const { committee } = await this.epochCache.getCommittee(block.header.globalVariables.slotNumber);
|
|
395
|
-
if (!committee) {
|
|
396
|
-
this.log.warn(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
|
|
397
|
-
throw new Error(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
|
|
398
|
-
}
|
|
399
|
-
attestationsAndSigners.attestations = committee.map((committeeMember)=>CommitteeAttestation.fromAddress(committeeMember));
|
|
400
|
-
}
|
|
401
|
-
const blobFields = block.getCheckpointBlobFields();
|
|
402
|
-
const blobs = getBlobsPerL1Block(blobFields);
|
|
850
|
+
const blobFields = checkpoint.toBlobFields();
|
|
851
|
+
const blobs = await getBlobsPerL1Block(blobFields);
|
|
403
852
|
const blobInput = getPrefixedEthBlobCommitments(blobs);
|
|
404
853
|
const args = [
|
|
405
854
|
{
|
|
406
|
-
header:
|
|
407
|
-
archive: toHex(
|
|
855
|
+
header: checkpoint.header.toViem(),
|
|
856
|
+
archive: toHex(checkpoint.archive.root.toBuffer()),
|
|
408
857
|
oracleInput: {
|
|
409
|
-
feeAssetPriceModifier:
|
|
858
|
+
feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
|
|
410
859
|
}
|
|
411
860
|
},
|
|
412
861
|
attestationsAndSigners.getPackedAttestations(),
|
|
@@ -431,9 +880,38 @@ export class SequencerPublisher {
|
|
|
431
880
|
}
|
|
432
881
|
const round = await base.computeRound(slotNumber);
|
|
433
882
|
const roundInfo = await base.getRoundInfo(this.rollupContract.address, round);
|
|
883
|
+
if (roundInfo.quorumReached) {
|
|
884
|
+
return false;
|
|
885
|
+
}
|
|
434
886
|
if (roundInfo.lastSignalSlot >= slotNumber) {
|
|
435
887
|
return false;
|
|
436
888
|
}
|
|
889
|
+
if (await this.isPayloadEmpty(payload)) {
|
|
890
|
+
this.log.warn(`Skipping vote cast for payload with empty code`);
|
|
891
|
+
return false;
|
|
892
|
+
}
|
|
893
|
+
// Check if payload was already submitted to governance
|
|
894
|
+
const cacheKey = payload.toString();
|
|
895
|
+
if (!this.payloadProposedCache.has(cacheKey)) {
|
|
896
|
+
try {
|
|
897
|
+
const l1StartBlock = await this.rollupContract.getL1StartBlock();
|
|
898
|
+
const proposed = await retry(()=>base.hasPayloadBeenProposed(payload.toString(), l1StartBlock), 'Check if payload was proposed', makeBackoff([
|
|
899
|
+
0,
|
|
900
|
+
1,
|
|
901
|
+
2
|
|
902
|
+
]), this.log, true);
|
|
903
|
+
if (proposed) {
|
|
904
|
+
this.payloadProposedCache.add(cacheKey);
|
|
905
|
+
}
|
|
906
|
+
} catch (err) {
|
|
907
|
+
this.log.warn(`Failed to check if payload ${payload} was proposed after retries, skipping signal`, err);
|
|
908
|
+
return false;
|
|
909
|
+
}
|
|
910
|
+
}
|
|
911
|
+
if (this.payloadProposedCache.has(cacheKey)) {
|
|
912
|
+
this.log.info(`Payload ${payload} was already proposed to governance, stopping signals`);
|
|
913
|
+
return false;
|
|
914
|
+
}
|
|
437
915
|
const cachedLastVote = this.lastActions[signalType];
|
|
438
916
|
this.lastActions[signalType] = slotNumber;
|
|
439
917
|
const action = signalType;
|
|
@@ -447,7 +925,10 @@ export class SequencerPublisher {
|
|
|
447
925
|
try {
|
|
448
926
|
await this.l1TxUtils.simulate(request, {
|
|
449
927
|
time: timestamp
|
|
450
|
-
}, [],
|
|
928
|
+
}, [], mergeAbis([
|
|
929
|
+
request.abi ?? [],
|
|
930
|
+
ErrorsAbi
|
|
931
|
+
]));
|
|
451
932
|
this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
|
|
452
933
|
request
|
|
453
934
|
});
|
|
@@ -472,17 +953,27 @@ export class SequencerPublisher {
|
|
|
472
953
|
payload: payload.toString()
|
|
473
954
|
};
|
|
474
955
|
if (!success) {
|
|
475
|
-
this.log.error(`Signaling in
|
|
956
|
+
this.log.error(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
|
|
476
957
|
this.lastActions[signalType] = cachedLastVote;
|
|
477
958
|
return false;
|
|
478
959
|
} else {
|
|
479
|
-
this.log.info(`Signaling in
|
|
960
|
+
this.log.info(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
|
|
480
961
|
return true;
|
|
481
962
|
}
|
|
482
963
|
}
|
|
483
964
|
});
|
|
484
965
|
return true;
|
|
485
966
|
}
|
|
967
|
+
async isPayloadEmpty(payload) {
|
|
968
|
+
const key = payload.toString();
|
|
969
|
+
const cached = this.isPayloadEmptyCache.get(key);
|
|
970
|
+
if (cached) {
|
|
971
|
+
return cached;
|
|
972
|
+
}
|
|
973
|
+
const isEmpty = !await this.l1TxUtils.getCode(payload);
|
|
974
|
+
this.isPayloadEmptyCache.set(key, isEmpty);
|
|
975
|
+
return isEmpty;
|
|
976
|
+
}
|
|
486
977
|
/**
|
|
487
978
|
* Enqueues a governance castSignal transaction to cast a signal for a given slot number.
|
|
488
979
|
* @param slotNumber - The slot number to cast a signal for.
|
|
@@ -578,22 +1069,17 @@ export class SequencerPublisher {
|
|
|
578
1069
|
}
|
|
579
1070
|
return true;
|
|
580
1071
|
}
|
|
581
|
-
/**
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
* @returns True if the tx has been enqueued, throws otherwise. See #9315
|
|
586
|
-
*/ async enqueueProposeL2Block(block, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
|
|
587
|
-
const checkpointHeader = block.getCheckpointHeader();
|
|
588
|
-
const blobFields = block.getCheckpointBlobFields();
|
|
589
|
-
const blobs = getBlobsPerL1Block(blobFields);
|
|
1072
|
+
/** Simulates and enqueues a proposal for a checkpoint on L1 */ async enqueueProposeCheckpoint(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
|
|
1073
|
+
const checkpointHeader = checkpoint.header;
|
|
1074
|
+
const blobFields = checkpoint.toBlobFields();
|
|
1075
|
+
const blobs = await getBlobsPerL1Block(blobFields);
|
|
590
1076
|
const proposeTxArgs = {
|
|
591
1077
|
header: checkpointHeader,
|
|
592
|
-
archive:
|
|
593
|
-
body: block.body.toBuffer(),
|
|
1078
|
+
archive: checkpoint.archive.root.toBuffer(),
|
|
594
1079
|
blobs,
|
|
595
1080
|
attestationsAndSigners,
|
|
596
|
-
attestationsAndSignersSignature
|
|
1081
|
+
attestationsAndSignersSignature,
|
|
1082
|
+
feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
|
|
597
1083
|
};
|
|
598
1084
|
let ts;
|
|
599
1085
|
try {
|
|
@@ -602,36 +1088,35 @@ export class SequencerPublisher {
|
|
|
602
1088
|
// By simulation issue, I mean the fact that the block.timestamp is equal to the last block, not the next, which
|
|
603
1089
|
// make time consistency checks break.
|
|
604
1090
|
// TODO(palla): Check whether we're validating twice, once here and once within addProposeTx, since we call simulateProposeTx in both places.
|
|
605
|
-
ts = await this.
|
|
1091
|
+
ts = await this.validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts);
|
|
606
1092
|
} catch (err) {
|
|
607
|
-
this.log.error(`
|
|
608
|
-
...
|
|
609
|
-
slotNumber:
|
|
610
|
-
|
|
1093
|
+
this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
|
|
1094
|
+
...checkpoint.getStats(),
|
|
1095
|
+
slotNumber: checkpoint.header.slotNumber,
|
|
1096
|
+
forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
|
|
611
1097
|
});
|
|
612
1098
|
throw err;
|
|
613
1099
|
}
|
|
614
|
-
this.log.verbose(`Enqueuing
|
|
615
|
-
...
|
|
1100
|
+
this.log.verbose(`Enqueuing checkpoint propose transaction`, {
|
|
1101
|
+
...checkpoint.toCheckpointInfo(),
|
|
616
1102
|
...opts
|
|
617
1103
|
});
|
|
618
|
-
await this.addProposeTx(
|
|
619
|
-
return true;
|
|
1104
|
+
await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
|
|
620
1105
|
}
|
|
621
|
-
|
|
1106
|
+
enqueueInvalidateCheckpoint(request, opts = {}) {
|
|
622
1107
|
if (!request) {
|
|
623
1108
|
return;
|
|
624
1109
|
}
|
|
625
1110
|
// We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
|
|
626
1111
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
|
|
627
|
-
const { gasUsed,
|
|
1112
|
+
const { gasUsed, checkpointNumber } = request;
|
|
628
1113
|
const logData = {
|
|
629
1114
|
gasUsed,
|
|
630
|
-
|
|
1115
|
+
checkpointNumber,
|
|
631
1116
|
gasLimit,
|
|
632
1117
|
opts
|
|
633
1118
|
};
|
|
634
|
-
this.log.verbose(`Enqueuing invalidate
|
|
1119
|
+
this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
|
|
635
1120
|
this.addRequest({
|
|
636
1121
|
action: `invalidate-by-${request.reason}`,
|
|
637
1122
|
request: request.request,
|
|
@@ -643,12 +1128,12 @@ export class SequencerPublisher {
|
|
|
643
1128
|
checkSuccess: (_req, result)=>{
|
|
644
1129
|
const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
|
|
645
1130
|
if (!success) {
|
|
646
|
-
this.log.warn(`Invalidate
|
|
1131
|
+
this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
|
|
647
1132
|
...result,
|
|
648
1133
|
...logData
|
|
649
1134
|
});
|
|
650
1135
|
} else {
|
|
651
|
-
this.log.info(`Invalidate
|
|
1136
|
+
this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
|
|
652
1137
|
...result,
|
|
653
1138
|
...logData
|
|
654
1139
|
});
|
|
@@ -671,27 +1156,37 @@ export class SequencerPublisher {
|
|
|
671
1156
|
this.lastActions[action] = slotNumber;
|
|
672
1157
|
this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
|
|
673
1158
|
let gasUsed;
|
|
1159
|
+
const simulateAbi = mergeAbis([
|
|
1160
|
+
request.abi ?? [],
|
|
1161
|
+
ErrorsAbi
|
|
1162
|
+
]);
|
|
674
1163
|
try {
|
|
675
1164
|
({ gasUsed } = await this.l1TxUtils.simulate(request, {
|
|
676
1165
|
time: timestamp
|
|
677
|
-
}, [],
|
|
1166
|
+
}, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
|
|
678
1167
|
this.log.verbose(`Simulation for ${action} succeeded`, {
|
|
679
1168
|
...logData,
|
|
680
1169
|
request,
|
|
681
1170
|
gasUsed
|
|
682
1171
|
});
|
|
683
1172
|
} catch (err) {
|
|
684
|
-
const viemError = formatViemError(err);
|
|
1173
|
+
const viemError = formatViemError(err, simulateAbi);
|
|
685
1174
|
this.log.error(`Simulation for ${action} at ${slotNumber} failed`, viemError, logData);
|
|
686
1175
|
return false;
|
|
687
1176
|
}
|
|
688
1177
|
// We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
|
|
689
1178
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
|
|
690
1179
|
logData.gasLimit = gasLimit;
|
|
1180
|
+
// Store the ABI used for simulation on the request so Multicall3.forward can decode errors
|
|
1181
|
+
// when the tx is sent and a revert is diagnosed via simulation.
|
|
1182
|
+
const requestWithAbi = {
|
|
1183
|
+
...request,
|
|
1184
|
+
abi: simulateAbi
|
|
1185
|
+
};
|
|
691
1186
|
this.log.debug(`Enqueuing ${action}`, logData);
|
|
692
1187
|
this.addRequest({
|
|
693
1188
|
action,
|
|
694
|
-
request,
|
|
1189
|
+
request: requestWithAbi,
|
|
695
1190
|
gasConfig: {
|
|
696
1191
|
gasLimit
|
|
697
1192
|
},
|
|
@@ -769,8 +1264,7 @@ export class SequencerPublisher {
|
|
|
769
1264
|
header: encodedData.header.toViem(),
|
|
770
1265
|
archive: toHex(encodedData.archive),
|
|
771
1266
|
oracleInput: {
|
|
772
|
-
|
|
773
|
-
feeAssetPriceModifier: 0n
|
|
1267
|
+
feeAssetPriceModifier: encodedData.feeAssetPriceModifier
|
|
774
1268
|
}
|
|
775
1269
|
},
|
|
776
1270
|
encodedData.attestationsAndSigners.getPackedAttestations(),
|
|
@@ -797,8 +1291,8 @@ export class SequencerPublisher {
|
|
|
797
1291
|
functionName: 'propose',
|
|
798
1292
|
args
|
|
799
1293
|
});
|
|
800
|
-
// override the pending
|
|
801
|
-
const
|
|
1294
|
+
// override the pending checkpoint number if requested
|
|
1295
|
+
const forcePendingCheckpointNumberStateDiff = (options.forcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
|
|
802
1296
|
const stateOverrides = [
|
|
803
1297
|
{
|
|
804
1298
|
address: this.rollupContract.address,
|
|
@@ -808,7 +1302,7 @@ export class SequencerPublisher {
|
|
|
808
1302
|
slot: toPaddedHex(RollupContract.checkBlobStorageSlot, true),
|
|
809
1303
|
value: toPaddedHex(0n, true)
|
|
810
1304
|
},
|
|
811
|
-
...
|
|
1305
|
+
...forcePendingCheckpointNumberStateDiff
|
|
812
1306
|
]
|
|
813
1307
|
}
|
|
814
1308
|
];
|
|
@@ -822,7 +1316,7 @@ export class SequencerPublisher {
|
|
|
822
1316
|
const simulationResult = await this.l1TxUtils.simulate({
|
|
823
1317
|
to: this.rollupContract.address,
|
|
824
1318
|
data: rollupData,
|
|
825
|
-
gas:
|
|
1319
|
+
gas: MAX_L1_TX_LIMIT,
|
|
826
1320
|
...this.proposerAddressForSimulation && {
|
|
827
1321
|
from: this.proposerAddressForSimulation.toString()
|
|
828
1322
|
}
|
|
@@ -830,10 +1324,10 @@ export class SequencerPublisher {
|
|
|
830
1324
|
// @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
|
|
831
1325
|
time: timestamp + 1n,
|
|
832
1326
|
// @note reth should have a 30m gas limit per block but throws errors that this tx is beyond limit so we increase here
|
|
833
|
-
gasLimit:
|
|
1327
|
+
gasLimit: MAX_L1_TX_LIMIT * 2n
|
|
834
1328
|
}, stateOverrides, RollupAbi, {
|
|
835
1329
|
// @note fallback gas estimate to use if the node doesn't support simulation API
|
|
836
|
-
fallbackGasEstimate:
|
|
1330
|
+
fallbackGasEstimate: MAX_L1_TX_LIMIT
|
|
837
1331
|
}).catch((err)=>{
|
|
838
1332
|
// In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
|
|
839
1333
|
const viemError = formatViemError(err);
|
|
@@ -841,7 +1335,7 @@ export class SequencerPublisher {
|
|
|
841
1335
|
this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
|
|
842
1336
|
// Return a minimal simulation result with the fallback gas estimate
|
|
843
1337
|
return {
|
|
844
|
-
gasUsed:
|
|
1338
|
+
gasUsed: MAX_L1_TX_LIMIT,
|
|
845
1339
|
logs: []
|
|
846
1340
|
};
|
|
847
1341
|
}
|
|
@@ -853,24 +1347,25 @@ export class SequencerPublisher {
|
|
|
853
1347
|
simulationResult
|
|
854
1348
|
};
|
|
855
1349
|
}
|
|
856
|
-
async addProposeTx(
|
|
1350
|
+
async addProposeTx(checkpoint, encodedData, opts = {}, timestamp) {
|
|
1351
|
+
const slot = checkpoint.header.slotNumber;
|
|
857
1352
|
const timer = new Timer();
|
|
858
1353
|
const kzg = Blob.getViemKzgInstance();
|
|
859
1354
|
const { rollupData, simulationResult, blobEvaluationGas } = await this.prepareProposeTx(encodedData, timestamp, opts);
|
|
860
1355
|
const startBlock = await this.l1TxUtils.getBlockNumber();
|
|
861
1356
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(simulationResult.gasUsed) * 64 / 63)) + blobEvaluationGas + SequencerPublisher.MULTICALL_OVERHEAD_GAS_GUESS);
|
|
862
|
-
// Send the blobs to the blob
|
|
863
|
-
// tx fails but it does get mined. We make sure that the blobs are sent to the blob
|
|
864
|
-
void this.
|
|
865
|
-
|
|
866
|
-
|
|
1357
|
+
// Send the blobs to the blob client preemptively. This helps in tests where the sequencer mistakingly thinks that the propose
|
|
1358
|
+
// tx fails but it does get mined. We make sure that the blobs are sent to the blob client regardless of the tx outcome.
|
|
1359
|
+
void Promise.resolve().then(()=>this.blobClient.sendBlobsToFilestore(encodedData.blobs).catch((_err)=>{
|
|
1360
|
+
this.log.error('Failed to send blobs to blob client');
|
|
1361
|
+
}));
|
|
867
1362
|
return this.addRequest({
|
|
868
1363
|
action: 'propose',
|
|
869
1364
|
request: {
|
|
870
1365
|
to: this.rollupContract.address,
|
|
871
1366
|
data: rollupData
|
|
872
1367
|
},
|
|
873
|
-
lastValidL2Slot:
|
|
1368
|
+
lastValidL2Slot: checkpoint.header.slotNumber,
|
|
874
1369
|
gasConfig: {
|
|
875
1370
|
...opts,
|
|
876
1371
|
gasLimit
|
|
@@ -898,25 +1393,23 @@ export class SequencerPublisher {
|
|
|
898
1393
|
calldataGas,
|
|
899
1394
|
calldataSize,
|
|
900
1395
|
sender,
|
|
901
|
-
...
|
|
1396
|
+
...checkpoint.getStats(),
|
|
902
1397
|
eventName: 'rollup-published-to-l1',
|
|
903
1398
|
blobCount: encodedData.blobs.length,
|
|
904
1399
|
inclusionBlocks
|
|
905
1400
|
};
|
|
906
|
-
this.log.info(`Published
|
|
1401
|
+
this.log.info(`Published checkpoint ${checkpoint.number} at slot ${slot} to rollup contract`, {
|
|
907
1402
|
...stats,
|
|
908
|
-
...
|
|
909
|
-
...receipt
|
|
1403
|
+
...checkpoint.getStats(),
|
|
1404
|
+
...pick(receipt, 'transactionHash', 'blockHash')
|
|
910
1405
|
});
|
|
911
1406
|
this.metrics.recordProcessBlockTx(timer.ms(), publishStats);
|
|
912
1407
|
return true;
|
|
913
1408
|
} else {
|
|
914
1409
|
this.metrics.recordFailedTx('process');
|
|
915
|
-
this.log.error(`
|
|
916
|
-
...
|
|
917
|
-
receipt
|
|
918
|
-
txHash: receipt.transactionHash,
|
|
919
|
-
slotNumber: block.header.globalVariables.slotNumber
|
|
1410
|
+
this.log.error(`Publishing checkpoint at slot ${slot} failed with ${errorMsg ?? 'no error message'}`, undefined, {
|
|
1411
|
+
...checkpoint.getStats(),
|
|
1412
|
+
...receipt
|
|
920
1413
|
});
|
|
921
1414
|
return false;
|
|
922
1415
|
}
|