@aztec/sequencer-client 0.0.1-commit.21caa21 → 0.0.1-commit.21ecf947b
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 +12 -12
- package/dest/client/sequencer-client.d.ts.map +1 -1
- package/dest/client/sequencer-client.js +33 -26
- package/dest/config.d.ts +12 -6
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +71 -32
- package/dest/global_variable_builder/global_builder.d.ts +22 -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 +7 -4
- package/dest/publisher/config.d.ts.map +1 -1
- package/dest/publisher/config.js +9 -3
- package/dest/publisher/sequencer-publisher-factory.d.ts +5 -4
- package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher-factory.js +1 -1
- 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 +56 -42
- package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher.js +615 -133
- 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 +1193 -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 +27 -3
- package/dest/sequencer/metrics.d.ts.map +1 -1
- package/dest/sequencer/metrics.js +188 -66
- package/dest/sequencer/sequencer.d.ts +109 -131
- package/dest/sequencer/sequencer.d.ts.map +1 -1
- package/dest/sequencer/sequencer.js +700 -606
- 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 +4 -3
- 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 +31 -42
- package/src/config.ts +78 -36
- package/src/global_variable_builder/global_builder.ts +65 -61
- package/src/index.ts +1 -7
- package/src/publisher/config.ts +12 -9
- package/src/publisher/sequencer-publisher-factory.ts +5 -4
- package/src/publisher/sequencer-publisher-metrics.ts +19 -71
- package/src/publisher/sequencer-publisher.ts +327 -166
- package/src/sequencer/README.md +531 -0
- package/src/sequencer/checkpoint_proposal_job.ts +882 -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 +236 -76
- package/src/sequencer/sequencer.ts +445 -813
- 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 +3 -2
- 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,9 +1,385 @@
|
|
|
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';
|
|
@@ -11,8 +387,8 @@ import { bufferToHex } from '@aztec/foundation/string';
|
|
|
11
387
|
import { Timer } from '@aztec/foundation/timer';
|
|
12
388
|
import { EmpireBaseAbi, ErrorsAbi, RollupAbi } from '@aztec/l1-artifacts';
|
|
13
389
|
import { encodeSlashConsensusVotes } from '@aztec/slasher';
|
|
14
|
-
import {
|
|
15
|
-
import { getTelemetryClient } from '@aztec/telemetry-client';
|
|
390
|
+
import { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
|
|
391
|
+
import { getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
|
|
16
392
|
import { encodeFunctionData, toHex } from 'viem';
|
|
17
393
|
import { SequencerPublisherMetrics } from './sequencer-publisher-metrics.js';
|
|
18
394
|
export const Actions = [
|
|
@@ -28,22 +404,41 @@ export const Actions = [
|
|
|
28
404
|
];
|
|
29
405
|
// Sorting for actions such that invalidations go before proposals, and proposals go before votes
|
|
30
406
|
export const compareActions = (a, b)=>Actions.indexOf(a) - Actions.indexOf(b);
|
|
407
|
+
_dec = trackSpan('SequencerPublisher.sendRequests'), _dec1 = trackSpan('SequencerPublisher.validateBlockHeader'), _dec2 = trackSpan('SequencerPublisher.validateCheckpointForSubmission');
|
|
31
408
|
export class SequencerPublisher {
|
|
32
409
|
config;
|
|
410
|
+
static{
|
|
411
|
+
({ e: [_initProto] } = _apply_decs_2203_r(this, [
|
|
412
|
+
[
|
|
413
|
+
_dec,
|
|
414
|
+
2,
|
|
415
|
+
"sendRequests"
|
|
416
|
+
],
|
|
417
|
+
[
|
|
418
|
+
_dec1,
|
|
419
|
+
2,
|
|
420
|
+
"validateBlockHeader"
|
|
421
|
+
],
|
|
422
|
+
[
|
|
423
|
+
_dec2,
|
|
424
|
+
2,
|
|
425
|
+
"validateCheckpointForSubmission"
|
|
426
|
+
]
|
|
427
|
+
], []));
|
|
428
|
+
}
|
|
33
429
|
interrupted;
|
|
34
430
|
metrics;
|
|
35
431
|
epochCache;
|
|
36
432
|
governanceLog;
|
|
37
433
|
slashingLog;
|
|
38
434
|
lastActions;
|
|
435
|
+
isPayloadEmptyCache;
|
|
39
436
|
log;
|
|
40
437
|
ethereumSlotDuration;
|
|
41
|
-
|
|
438
|
+
blobClient;
|
|
42
439
|
/** 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;
|
|
440
|
+
/** L1 fee analyzer for fisherman mode */ l1FeeAnalyzer;
|
|
441
|
+
/** Fee asset price oracle for computing price modifiers from Uniswap V4 */ feeAssetPriceOracle;
|
|
47
442
|
// A CALL to a cold address is 2700 gas
|
|
48
443
|
static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
|
|
49
444
|
// Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
|
|
@@ -53,23 +448,24 @@ export class SequencerPublisher {
|
|
|
53
448
|
govProposerContract;
|
|
54
449
|
slashingProposerContract;
|
|
55
450
|
slashFactoryContract;
|
|
451
|
+
tracer;
|
|
56
452
|
requests;
|
|
57
453
|
constructor(config, deps){
|
|
58
454
|
this.config = config;
|
|
59
|
-
this.interrupted = false;
|
|
455
|
+
this.interrupted = (_initProto(this), false);
|
|
60
456
|
this.governanceLog = createLogger('sequencer:publisher:governance');
|
|
61
457
|
this.slashingLog = createLogger('sequencer:publisher:slashing');
|
|
62
458
|
this.lastActions = {};
|
|
459
|
+
this.isPayloadEmptyCache = new Map();
|
|
63
460
|
this.requests = [];
|
|
64
461
|
this.log = deps.log ?? createLogger('sequencer:publisher');
|
|
65
462
|
this.ethereumSlotDuration = BigInt(config.ethereumSlotDuration);
|
|
66
463
|
this.epochCache = deps.epochCache;
|
|
67
464
|
this.lastActions = deps.lastActions;
|
|
68
|
-
this.
|
|
69
|
-
logger: createLogger('sequencer:blob-sink:client')
|
|
70
|
-
});
|
|
465
|
+
this.blobClient = deps.blobClient;
|
|
71
466
|
const telemetry = deps.telemetry ?? getTelemetryClient();
|
|
72
467
|
this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
|
|
468
|
+
this.tracer = telemetry.getTracer('SequencerPublisher');
|
|
73
469
|
this.l1TxUtils = deps.l1TxUtils;
|
|
74
470
|
this.rollupContract = deps.rollupContract;
|
|
75
471
|
this.govProposerContract = deps.governanceProposerContract;
|
|
@@ -80,14 +476,31 @@ export class SequencerPublisher {
|
|
|
80
476
|
this.slashingProposerContract = newSlashingProposer;
|
|
81
477
|
});
|
|
82
478
|
this.slashFactoryContract = deps.slashFactoryContract;
|
|
479
|
+
// Initialize L1 fee analyzer for fisherman mode
|
|
480
|
+
if (config.fishermanMode) {
|
|
481
|
+
this.l1FeeAnalyzer = new L1FeeAnalyzer(this.l1TxUtils.client, deps.dateProvider, createLogger('sequencer:publisher:fee-analyzer'));
|
|
482
|
+
}
|
|
483
|
+
// Initialize fee asset price oracle
|
|
484
|
+
this.feeAssetPriceOracle = new FeeAssetPriceOracle(this.l1TxUtils.client, this.rollupContract, createLogger('sequencer:publisher:price-oracle'));
|
|
83
485
|
}
|
|
84
486
|
getRollupContract() {
|
|
85
487
|
return this.rollupContract;
|
|
86
488
|
}
|
|
489
|
+
/**
|
|
490
|
+
* Gets the fee asset price modifier from the oracle.
|
|
491
|
+
* Returns 0n if the oracle query fails.
|
|
492
|
+
*/ getFeeAssetPriceModifier() {
|
|
493
|
+
return this.feeAssetPriceOracle.computePriceModifier();
|
|
494
|
+
}
|
|
87
495
|
getSenderAddress() {
|
|
88
496
|
return this.l1TxUtils.getSenderAddress();
|
|
89
497
|
}
|
|
90
498
|
/**
|
|
499
|
+
* Gets the L1 fee analyzer instance (only available in fisherman mode)
|
|
500
|
+
*/ getL1FeeAnalyzer() {
|
|
501
|
+
return this.l1FeeAnalyzer;
|
|
502
|
+
}
|
|
503
|
+
/**
|
|
91
504
|
* Sets the proposer address to use for simulations in fisherman mode.
|
|
92
505
|
* @param proposerAddress - The actual proposer's address to use for balance lookups in simulations
|
|
93
506
|
*/ setProposerAddressForSimulation(proposerAddress) {
|
|
@@ -109,6 +522,46 @@ export class SequencerPublisher {
|
|
|
109
522
|
}
|
|
110
523
|
}
|
|
111
524
|
/**
|
|
525
|
+
* Analyzes L1 fees for the pending requests without sending them.
|
|
526
|
+
* This is used in fisherman mode to validate fee calculations.
|
|
527
|
+
* @param l2SlotNumber - The L2 slot number for this analysis
|
|
528
|
+
* @param onComplete - Optional callback to invoke when analysis completes (after block is mined)
|
|
529
|
+
* @returns The analysis result (incomplete until block mines), or undefined if no requests
|
|
530
|
+
*/ async analyzeL1Fees(l2SlotNumber, onComplete) {
|
|
531
|
+
if (!this.l1FeeAnalyzer) {
|
|
532
|
+
this.log.warn('L1 fee analyzer not available (not in fisherman mode)');
|
|
533
|
+
return undefined;
|
|
534
|
+
}
|
|
535
|
+
const requestsToAnalyze = [
|
|
536
|
+
...this.requests
|
|
537
|
+
];
|
|
538
|
+
if (requestsToAnalyze.length === 0) {
|
|
539
|
+
this.log.debug('No requests to analyze for L1 fees');
|
|
540
|
+
return undefined;
|
|
541
|
+
}
|
|
542
|
+
// Extract blob config from requests (if any)
|
|
543
|
+
const blobConfigs = requestsToAnalyze.filter((request)=>request.blobConfig).map((request)=>request.blobConfig);
|
|
544
|
+
const blobConfig = blobConfigs[0];
|
|
545
|
+
// Get gas configs
|
|
546
|
+
const gasConfigs = requestsToAnalyze.filter((request)=>request.gasConfig).map((request)=>request.gasConfig);
|
|
547
|
+
const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
|
|
548
|
+
const gasLimit = gasLimits.length > 0 ? gasLimits.reduce((sum, g)=>sum + g, 0n) : 0n;
|
|
549
|
+
// Get the transaction requests
|
|
550
|
+
const l1Requests = requestsToAnalyze.map((r)=>r.request);
|
|
551
|
+
// Start the analysis
|
|
552
|
+
const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
|
|
553
|
+
this.log.info('Started L1 fee analysis', {
|
|
554
|
+
analysisId,
|
|
555
|
+
l2SlotNumber: l2SlotNumber.toString(),
|
|
556
|
+
requestCount: requestsToAnalyze.length,
|
|
557
|
+
hasBlobConfig: !!blobConfig,
|
|
558
|
+
gasLimit: gasLimit.toString(),
|
|
559
|
+
actions: requestsToAnalyze.map((r)=>r.action)
|
|
560
|
+
});
|
|
561
|
+
// Return the analysis result (will be incomplete until block mines)
|
|
562
|
+
return this.l1FeeAnalyzer.getAnalysis(analysisId);
|
|
563
|
+
}
|
|
564
|
+
/**
|
|
112
565
|
* Sends all requests that are still valid.
|
|
113
566
|
* @returns one of:
|
|
114
567
|
* - A receipt and stats if the tx succeeded
|
|
@@ -119,7 +572,7 @@ export class SequencerPublisher {
|
|
|
119
572
|
...this.requests
|
|
120
573
|
];
|
|
121
574
|
this.requests = [];
|
|
122
|
-
if (this.interrupted) {
|
|
575
|
+
if (this.interrupted || requestsToProcess.length === 0) {
|
|
123
576
|
return undefined;
|
|
124
577
|
}
|
|
125
578
|
const currentL2Slot = this.getCurrentL2Slot();
|
|
@@ -154,7 +607,16 @@ export class SequencerPublisher {
|
|
|
154
607
|
const blobConfig = blobConfigs[0];
|
|
155
608
|
// Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
|
|
156
609
|
const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
|
|
157
|
-
|
|
610
|
+
let gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
|
|
611
|
+
// Cap at L1 block gas limit so the node accepts the tx ("gas limit too high" otherwise).
|
|
612
|
+
const maxGas = MAX_L1_TX_LIMIT;
|
|
613
|
+
if (gasLimit !== undefined && gasLimit > maxGas) {
|
|
614
|
+
this.log.debug('Capping bundled tx gas limit to L1 max', {
|
|
615
|
+
requested: gasLimit,
|
|
616
|
+
capped: maxGas
|
|
617
|
+
});
|
|
618
|
+
gasLimit = maxGas;
|
|
619
|
+
}
|
|
158
620
|
const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
|
|
159
621
|
const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
|
|
160
622
|
const txConfig = {
|
|
@@ -229,7 +691,7 @@ export class SequencerPublisher {
|
|
|
229
691
|
'InvalidArchive'
|
|
230
692
|
];
|
|
231
693
|
return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
|
|
232
|
-
forcePendingCheckpointNumber: opts.
|
|
694
|
+
forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
|
|
233
695
|
}).catch((err)=>{
|
|
234
696
|
if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
|
|
235
697
|
this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
|
|
@@ -257,11 +719,11 @@ export class SequencerPublisher {
|
|
|
257
719
|
[],
|
|
258
720
|
Signature.empty().toViemSignature(),
|
|
259
721
|
`0x${'0'.repeat(64)}`,
|
|
260
|
-
header.
|
|
722
|
+
header.blobsHash.toString(),
|
|
261
723
|
flags
|
|
262
724
|
];
|
|
263
725
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
264
|
-
const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.
|
|
726
|
+
const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
|
|
265
727
|
let balance = 0n;
|
|
266
728
|
if (this.config.fishermanMode) {
|
|
267
729
|
// In fisherman mode, we can't know where the proposer is publishing from
|
|
@@ -288,34 +750,37 @@ export class SequencerPublisher {
|
|
|
288
750
|
this.log.debug(`Simulated validateHeader`);
|
|
289
751
|
}
|
|
290
752
|
/**
|
|
291
|
-
* Simulate making a call to invalidate a
|
|
292
|
-
* @param
|
|
293
|
-
*/ async
|
|
753
|
+
* Simulate making a call to invalidate a checkpoint with invalid attestations. Returns undefined if no need to invalidate.
|
|
754
|
+
* @param validationResult - The validation result indicating which checkpoint to invalidate (as returned by the archiver)
|
|
755
|
+
*/ async simulateInvalidateCheckpoint(validationResult) {
|
|
294
756
|
if (validationResult.valid) {
|
|
295
757
|
return undefined;
|
|
296
758
|
}
|
|
297
|
-
const { reason,
|
|
298
|
-
const
|
|
759
|
+
const { reason, checkpoint } = validationResult;
|
|
760
|
+
const checkpointNumber = checkpoint.checkpointNumber;
|
|
299
761
|
const logData = {
|
|
300
|
-
...
|
|
762
|
+
...checkpoint,
|
|
301
763
|
reason
|
|
302
764
|
};
|
|
303
|
-
const
|
|
304
|
-
if (
|
|
305
|
-
this.log.verbose(`Skipping
|
|
306
|
-
|
|
765
|
+
const currentCheckpointNumber = await this.rollupContract.getCheckpointNumber();
|
|
766
|
+
if (currentCheckpointNumber < checkpointNumber) {
|
|
767
|
+
this.log.verbose(`Skipping checkpoint ${checkpointNumber} invalidation since it has already been removed from the pending chain`, {
|
|
768
|
+
currentCheckpointNumber,
|
|
307
769
|
...logData
|
|
308
770
|
});
|
|
309
771
|
return undefined;
|
|
310
772
|
}
|
|
311
|
-
const request = this.
|
|
312
|
-
this.log.debug(`Simulating invalidate
|
|
773
|
+
const request = this.buildInvalidateCheckpointRequest(validationResult);
|
|
774
|
+
this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
|
|
313
775
|
...logData,
|
|
314
776
|
request
|
|
315
777
|
});
|
|
316
778
|
try {
|
|
317
|
-
const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined,
|
|
318
|
-
|
|
779
|
+
const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, mergeAbis([
|
|
780
|
+
request.abi ?? [],
|
|
781
|
+
ErrorsAbi
|
|
782
|
+
]));
|
|
783
|
+
this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} succeeded`, {
|
|
319
784
|
...logData,
|
|
320
785
|
request,
|
|
321
786
|
gasUsed
|
|
@@ -323,90 +788,85 @@ export class SequencerPublisher {
|
|
|
323
788
|
return {
|
|
324
789
|
request,
|
|
325
790
|
gasUsed,
|
|
326
|
-
|
|
327
|
-
|
|
791
|
+
checkpointNumber,
|
|
792
|
+
forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
|
|
328
793
|
reason
|
|
329
794
|
};
|
|
330
795
|
} catch (err) {
|
|
331
796
|
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
|
|
797
|
+
// If the error is due to the checkpoint not being in the pending chain, and it was indeed removed by someone else,
|
|
798
|
+
// we can safely ignore it and return undefined so we go ahead with checkpoint building.
|
|
799
|
+
if (viemError.message?.includes('Rollup__CheckpointNotInPendingChain')) {
|
|
800
|
+
this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} failed due to checkpoint not being in pending chain`, {
|
|
336
801
|
...logData,
|
|
337
802
|
request,
|
|
338
803
|
error: viemError.message
|
|
339
804
|
});
|
|
340
|
-
const
|
|
341
|
-
if (
|
|
342
|
-
this.log.verbose(`
|
|
805
|
+
const latestPendingCheckpointNumber = await this.rollupContract.getCheckpointNumber();
|
|
806
|
+
if (latestPendingCheckpointNumber < checkpointNumber) {
|
|
807
|
+
this.log.verbose(`Checkpoint ${checkpointNumber} has already been invalidated`, {
|
|
343
808
|
...logData
|
|
344
809
|
});
|
|
345
810
|
return undefined;
|
|
346
811
|
} else {
|
|
347
|
-
this.log.error(`Simulation for invalidate ${
|
|
348
|
-
throw new Error(`Failed to simulate invalidate
|
|
812
|
+
this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed and it is still in pending chain`, viemError, logData);
|
|
813
|
+
throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber} while it is still in pending chain`, {
|
|
349
814
|
cause: viemError
|
|
350
815
|
});
|
|
351
816
|
}
|
|
352
817
|
}
|
|
353
|
-
// Otherwise, throw. We cannot build the next
|
|
354
|
-
this.log.error(`Simulation for invalidate
|
|
355
|
-
throw new Error(`Failed to simulate invalidate
|
|
818
|
+
// Otherwise, throw. We cannot build the next checkpoint if we cannot invalidate the previous one.
|
|
819
|
+
this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed`, viemError, logData);
|
|
820
|
+
throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber}`, {
|
|
356
821
|
cause: viemError
|
|
357
822
|
});
|
|
358
823
|
}
|
|
359
824
|
}
|
|
360
|
-
|
|
825
|
+
buildInvalidateCheckpointRequest(validationResult) {
|
|
361
826
|
if (validationResult.valid) {
|
|
362
|
-
throw new Error('Cannot invalidate a valid
|
|
827
|
+
throw new Error('Cannot invalidate a valid checkpoint');
|
|
363
828
|
}
|
|
364
|
-
const {
|
|
829
|
+
const { checkpoint, committee, reason } = validationResult;
|
|
365
830
|
const logData = {
|
|
366
|
-
...
|
|
831
|
+
...checkpoint,
|
|
367
832
|
reason
|
|
368
833
|
};
|
|
369
|
-
this.log.debug(`
|
|
834
|
+
this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
|
|
370
835
|
const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
|
|
371
836
|
if (reason === 'invalid-attestation') {
|
|
372
|
-
return this.rollupContract.buildInvalidateBadAttestationRequest(
|
|
837
|
+
return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
|
|
373
838
|
} else if (reason === 'insufficient-attestations') {
|
|
374
|
-
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(
|
|
839
|
+
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
|
|
375
840
|
} else {
|
|
376
841
|
const _ = reason;
|
|
377
842
|
throw new Error(`Unknown reason for invalidation`);
|
|
378
843
|
}
|
|
379
844
|
}
|
|
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) {
|
|
845
|
+
/** Simulates `propose` to make sure that the checkpoint is valid for submission */ async validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, options) {
|
|
389
846
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
847
|
+
// TODO(palla/mbps): This should not be needed, there's no flow where we propose with zero attestations. Or is there?
|
|
390
848
|
// If we have no attestations, we still need to provide the empty attestations
|
|
391
849
|
// so that the committee is recalculated correctly
|
|
392
|
-
const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
|
|
393
|
-
if (ignoreSignatures) {
|
|
394
|
-
|
|
395
|
-
|
|
396
|
-
|
|
397
|
-
|
|
398
|
-
|
|
399
|
-
|
|
400
|
-
|
|
401
|
-
|
|
850
|
+
// const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
|
|
851
|
+
// if (ignoreSignatures) {
|
|
852
|
+
// const { committee } = await this.epochCache.getCommittee(block.header.globalVariables.slotNumber);
|
|
853
|
+
// if (!committee) {
|
|
854
|
+
// this.log.warn(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
|
|
855
|
+
// throw new Error(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
|
|
856
|
+
// }
|
|
857
|
+
// attestationsAndSigners.attestations = committee.map(committeeMember =>
|
|
858
|
+
// CommitteeAttestation.fromAddress(committeeMember),
|
|
859
|
+
// );
|
|
860
|
+
// }
|
|
861
|
+
const blobFields = checkpoint.toBlobFields();
|
|
402
862
|
const blobs = getBlobsPerL1Block(blobFields);
|
|
403
863
|
const blobInput = getPrefixedEthBlobCommitments(blobs);
|
|
404
864
|
const args = [
|
|
405
865
|
{
|
|
406
|
-
header:
|
|
407
|
-
archive: toHex(
|
|
866
|
+
header: checkpoint.header.toViem(),
|
|
867
|
+
archive: toHex(checkpoint.archive.root.toBuffer()),
|
|
408
868
|
oracleInput: {
|
|
409
|
-
feeAssetPriceModifier:
|
|
869
|
+
feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
|
|
410
870
|
}
|
|
411
871
|
},
|
|
412
872
|
attestationsAndSigners.getPackedAttestations(),
|
|
@@ -431,9 +891,16 @@ export class SequencerPublisher {
|
|
|
431
891
|
}
|
|
432
892
|
const round = await base.computeRound(slotNumber);
|
|
433
893
|
const roundInfo = await base.getRoundInfo(this.rollupContract.address, round);
|
|
894
|
+
if (roundInfo.quorumReached) {
|
|
895
|
+
return false;
|
|
896
|
+
}
|
|
434
897
|
if (roundInfo.lastSignalSlot >= slotNumber) {
|
|
435
898
|
return false;
|
|
436
899
|
}
|
|
900
|
+
if (await this.isPayloadEmpty(payload)) {
|
|
901
|
+
this.log.warn(`Skipping vote cast for payload with empty code`);
|
|
902
|
+
return false;
|
|
903
|
+
}
|
|
437
904
|
const cachedLastVote = this.lastActions[signalType];
|
|
438
905
|
this.lastActions[signalType] = slotNumber;
|
|
439
906
|
const action = signalType;
|
|
@@ -447,7 +914,10 @@ export class SequencerPublisher {
|
|
|
447
914
|
try {
|
|
448
915
|
await this.l1TxUtils.simulate(request, {
|
|
449
916
|
time: timestamp
|
|
450
|
-
}, [],
|
|
917
|
+
}, [], mergeAbis([
|
|
918
|
+
request.abi ?? [],
|
|
919
|
+
ErrorsAbi
|
|
920
|
+
]));
|
|
451
921
|
this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
|
|
452
922
|
request
|
|
453
923
|
});
|
|
@@ -472,17 +942,27 @@ export class SequencerPublisher {
|
|
|
472
942
|
payload: payload.toString()
|
|
473
943
|
};
|
|
474
944
|
if (!success) {
|
|
475
|
-
this.log.error(`Signaling in
|
|
945
|
+
this.log.error(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
|
|
476
946
|
this.lastActions[signalType] = cachedLastVote;
|
|
477
947
|
return false;
|
|
478
948
|
} else {
|
|
479
|
-
this.log.info(`Signaling in
|
|
949
|
+
this.log.info(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
|
|
480
950
|
return true;
|
|
481
951
|
}
|
|
482
952
|
}
|
|
483
953
|
});
|
|
484
954
|
return true;
|
|
485
955
|
}
|
|
956
|
+
async isPayloadEmpty(payload) {
|
|
957
|
+
const key = payload.toString();
|
|
958
|
+
const cached = this.isPayloadEmptyCache.get(key);
|
|
959
|
+
if (cached) {
|
|
960
|
+
return cached;
|
|
961
|
+
}
|
|
962
|
+
const isEmpty = !await this.l1TxUtils.getCode(payload);
|
|
963
|
+
this.isPayloadEmptyCache.set(key, isEmpty);
|
|
964
|
+
return isEmpty;
|
|
965
|
+
}
|
|
486
966
|
/**
|
|
487
967
|
* Enqueues a governance castSignal transaction to cast a signal for a given slot number.
|
|
488
968
|
* @param slotNumber - The slot number to cast a signal for.
|
|
@@ -578,22 +1058,17 @@ export class SequencerPublisher {
|
|
|
578
1058
|
}
|
|
579
1059
|
return true;
|
|
580
1060
|
}
|
|
581
|
-
/**
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
* @param block - L2 block to propose.
|
|
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();
|
|
1061
|
+
/** Simulates and enqueues a proposal for a checkpoint on L1 */ async enqueueProposeCheckpoint(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
|
|
1062
|
+
const checkpointHeader = checkpoint.header;
|
|
1063
|
+
const blobFields = checkpoint.toBlobFields();
|
|
589
1064
|
const blobs = getBlobsPerL1Block(blobFields);
|
|
590
1065
|
const proposeTxArgs = {
|
|
591
1066
|
header: checkpointHeader,
|
|
592
|
-
archive:
|
|
593
|
-
body: block.body.toBuffer(),
|
|
1067
|
+
archive: checkpoint.archive.root.toBuffer(),
|
|
594
1068
|
blobs,
|
|
595
1069
|
attestationsAndSigners,
|
|
596
|
-
attestationsAndSignersSignature
|
|
1070
|
+
attestationsAndSignersSignature,
|
|
1071
|
+
feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
|
|
597
1072
|
};
|
|
598
1073
|
let ts;
|
|
599
1074
|
try {
|
|
@@ -602,36 +1077,35 @@ export class SequencerPublisher {
|
|
|
602
1077
|
// By simulation issue, I mean the fact that the block.timestamp is equal to the last block, not the next, which
|
|
603
1078
|
// make time consistency checks break.
|
|
604
1079
|
// 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.
|
|
1080
|
+
ts = await this.validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts);
|
|
606
1081
|
} catch (err) {
|
|
607
|
-
this.log.error(`
|
|
608
|
-
...
|
|
609
|
-
slotNumber:
|
|
610
|
-
|
|
1082
|
+
this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
|
|
1083
|
+
...checkpoint.getStats(),
|
|
1084
|
+
slotNumber: checkpoint.header.slotNumber,
|
|
1085
|
+
forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
|
|
611
1086
|
});
|
|
612
1087
|
throw err;
|
|
613
1088
|
}
|
|
614
|
-
this.log.verbose(`Enqueuing
|
|
615
|
-
...
|
|
1089
|
+
this.log.verbose(`Enqueuing checkpoint propose transaction`, {
|
|
1090
|
+
...checkpoint.toCheckpointInfo(),
|
|
616
1091
|
...opts
|
|
617
1092
|
});
|
|
618
|
-
await this.addProposeTx(
|
|
619
|
-
return true;
|
|
1093
|
+
await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
|
|
620
1094
|
}
|
|
621
|
-
|
|
1095
|
+
enqueueInvalidateCheckpoint(request, opts = {}) {
|
|
622
1096
|
if (!request) {
|
|
623
1097
|
return;
|
|
624
1098
|
}
|
|
625
1099
|
// We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
|
|
626
1100
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
|
|
627
|
-
const { gasUsed,
|
|
1101
|
+
const { gasUsed, checkpointNumber } = request;
|
|
628
1102
|
const logData = {
|
|
629
1103
|
gasUsed,
|
|
630
|
-
|
|
1104
|
+
checkpointNumber,
|
|
631
1105
|
gasLimit,
|
|
632
1106
|
opts
|
|
633
1107
|
};
|
|
634
|
-
this.log.verbose(`Enqueuing invalidate
|
|
1108
|
+
this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
|
|
635
1109
|
this.addRequest({
|
|
636
1110
|
action: `invalidate-by-${request.reason}`,
|
|
637
1111
|
request: request.request,
|
|
@@ -643,12 +1117,12 @@ export class SequencerPublisher {
|
|
|
643
1117
|
checkSuccess: (_req, result)=>{
|
|
644
1118
|
const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
|
|
645
1119
|
if (!success) {
|
|
646
|
-
this.log.warn(`Invalidate
|
|
1120
|
+
this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
|
|
647
1121
|
...result,
|
|
648
1122
|
...logData
|
|
649
1123
|
});
|
|
650
1124
|
} else {
|
|
651
|
-
this.log.info(`Invalidate
|
|
1125
|
+
this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
|
|
652
1126
|
...result,
|
|
653
1127
|
...logData
|
|
654
1128
|
});
|
|
@@ -671,27 +1145,37 @@ export class SequencerPublisher {
|
|
|
671
1145
|
this.lastActions[action] = slotNumber;
|
|
672
1146
|
this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
|
|
673
1147
|
let gasUsed;
|
|
1148
|
+
const simulateAbi = mergeAbis([
|
|
1149
|
+
request.abi ?? [],
|
|
1150
|
+
ErrorsAbi
|
|
1151
|
+
]);
|
|
674
1152
|
try {
|
|
675
1153
|
({ gasUsed } = await this.l1TxUtils.simulate(request, {
|
|
676
1154
|
time: timestamp
|
|
677
|
-
}, [],
|
|
1155
|
+
}, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
|
|
678
1156
|
this.log.verbose(`Simulation for ${action} succeeded`, {
|
|
679
1157
|
...logData,
|
|
680
1158
|
request,
|
|
681
1159
|
gasUsed
|
|
682
1160
|
});
|
|
683
1161
|
} catch (err) {
|
|
684
|
-
const viemError = formatViemError(err);
|
|
1162
|
+
const viemError = formatViemError(err, simulateAbi);
|
|
685
1163
|
this.log.error(`Simulation for ${action} at ${slotNumber} failed`, viemError, logData);
|
|
686
1164
|
return false;
|
|
687
1165
|
}
|
|
688
1166
|
// We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
|
|
689
1167
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
|
|
690
1168
|
logData.gasLimit = gasLimit;
|
|
1169
|
+
// Store the ABI used for simulation on the request so Multicall3.forward can decode errors
|
|
1170
|
+
// when the tx is sent and a revert is diagnosed via simulation.
|
|
1171
|
+
const requestWithAbi = {
|
|
1172
|
+
...request,
|
|
1173
|
+
abi: simulateAbi
|
|
1174
|
+
};
|
|
691
1175
|
this.log.debug(`Enqueuing ${action}`, logData);
|
|
692
1176
|
this.addRequest({
|
|
693
1177
|
action,
|
|
694
|
-
request,
|
|
1178
|
+
request: requestWithAbi,
|
|
695
1179
|
gasConfig: {
|
|
696
1180
|
gasLimit
|
|
697
1181
|
},
|
|
@@ -769,8 +1253,7 @@ export class SequencerPublisher {
|
|
|
769
1253
|
header: encodedData.header.toViem(),
|
|
770
1254
|
archive: toHex(encodedData.archive),
|
|
771
1255
|
oracleInput: {
|
|
772
|
-
|
|
773
|
-
feeAssetPriceModifier: 0n
|
|
1256
|
+
feeAssetPriceModifier: encodedData.feeAssetPriceModifier
|
|
774
1257
|
}
|
|
775
1258
|
},
|
|
776
1259
|
encodedData.attestationsAndSigners.getPackedAttestations(),
|
|
@@ -797,8 +1280,8 @@ export class SequencerPublisher {
|
|
|
797
1280
|
functionName: 'propose',
|
|
798
1281
|
args
|
|
799
1282
|
});
|
|
800
|
-
// override the pending
|
|
801
|
-
const
|
|
1283
|
+
// override the pending checkpoint number if requested
|
|
1284
|
+
const forcePendingCheckpointNumberStateDiff = (options.forcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
|
|
802
1285
|
const stateOverrides = [
|
|
803
1286
|
{
|
|
804
1287
|
address: this.rollupContract.address,
|
|
@@ -808,7 +1291,7 @@ export class SequencerPublisher {
|
|
|
808
1291
|
slot: toPaddedHex(RollupContract.checkBlobStorageSlot, true),
|
|
809
1292
|
value: toPaddedHex(0n, true)
|
|
810
1293
|
},
|
|
811
|
-
...
|
|
1294
|
+
...forcePendingCheckpointNumberStateDiff
|
|
812
1295
|
]
|
|
813
1296
|
}
|
|
814
1297
|
];
|
|
@@ -822,7 +1305,7 @@ export class SequencerPublisher {
|
|
|
822
1305
|
const simulationResult = await this.l1TxUtils.simulate({
|
|
823
1306
|
to: this.rollupContract.address,
|
|
824
1307
|
data: rollupData,
|
|
825
|
-
gas:
|
|
1308
|
+
gas: MAX_L1_TX_LIMIT,
|
|
826
1309
|
...this.proposerAddressForSimulation && {
|
|
827
1310
|
from: this.proposerAddressForSimulation.toString()
|
|
828
1311
|
}
|
|
@@ -830,10 +1313,10 @@ export class SequencerPublisher {
|
|
|
830
1313
|
// @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
|
|
831
1314
|
time: timestamp + 1n,
|
|
832
1315
|
// @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:
|
|
1316
|
+
gasLimit: MAX_L1_TX_LIMIT * 2n
|
|
834
1317
|
}, stateOverrides, RollupAbi, {
|
|
835
1318
|
// @note fallback gas estimate to use if the node doesn't support simulation API
|
|
836
|
-
fallbackGasEstimate:
|
|
1319
|
+
fallbackGasEstimate: MAX_L1_TX_LIMIT
|
|
837
1320
|
}).catch((err)=>{
|
|
838
1321
|
// In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
|
|
839
1322
|
const viemError = formatViemError(err);
|
|
@@ -841,7 +1324,7 @@ export class SequencerPublisher {
|
|
|
841
1324
|
this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
|
|
842
1325
|
// Return a minimal simulation result with the fallback gas estimate
|
|
843
1326
|
return {
|
|
844
|
-
gasUsed:
|
|
1327
|
+
gasUsed: MAX_L1_TX_LIMIT,
|
|
845
1328
|
logs: []
|
|
846
1329
|
};
|
|
847
1330
|
}
|
|
@@ -853,24 +1336,25 @@ export class SequencerPublisher {
|
|
|
853
1336
|
simulationResult
|
|
854
1337
|
};
|
|
855
1338
|
}
|
|
856
|
-
async addProposeTx(
|
|
1339
|
+
async addProposeTx(checkpoint, encodedData, opts = {}, timestamp) {
|
|
1340
|
+
const slot = checkpoint.header.slotNumber;
|
|
857
1341
|
const timer = new Timer();
|
|
858
1342
|
const kzg = Blob.getViemKzgInstance();
|
|
859
1343
|
const { rollupData, simulationResult, blobEvaluationGas } = await this.prepareProposeTx(encodedData, timestamp, opts);
|
|
860
1344
|
const startBlock = await this.l1TxUtils.getBlockNumber();
|
|
861
1345
|
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
|
-
|
|
1346
|
+
// Send the blobs to the blob client preemptively. This helps in tests where the sequencer mistakingly thinks that the propose
|
|
1347
|
+
// tx fails but it does get mined. We make sure that the blobs are sent to the blob client regardless of the tx outcome.
|
|
1348
|
+
void Promise.resolve().then(()=>this.blobClient.sendBlobsToFilestore(encodedData.blobs).catch((_err)=>{
|
|
1349
|
+
this.log.error('Failed to send blobs to blob client');
|
|
1350
|
+
}));
|
|
867
1351
|
return this.addRequest({
|
|
868
1352
|
action: 'propose',
|
|
869
1353
|
request: {
|
|
870
1354
|
to: this.rollupContract.address,
|
|
871
1355
|
data: rollupData
|
|
872
1356
|
},
|
|
873
|
-
lastValidL2Slot:
|
|
1357
|
+
lastValidL2Slot: checkpoint.header.slotNumber,
|
|
874
1358
|
gasConfig: {
|
|
875
1359
|
...opts,
|
|
876
1360
|
gasLimit
|
|
@@ -898,25 +1382,23 @@ export class SequencerPublisher {
|
|
|
898
1382
|
calldataGas,
|
|
899
1383
|
calldataSize,
|
|
900
1384
|
sender,
|
|
901
|
-
...
|
|
1385
|
+
...checkpoint.getStats(),
|
|
902
1386
|
eventName: 'rollup-published-to-l1',
|
|
903
1387
|
blobCount: encodedData.blobs.length,
|
|
904
1388
|
inclusionBlocks
|
|
905
1389
|
};
|
|
906
|
-
this.log.info(`Published
|
|
1390
|
+
this.log.info(`Published checkpoint ${checkpoint.number} at slot ${slot} to rollup contract`, {
|
|
907
1391
|
...stats,
|
|
908
|
-
...
|
|
909
|
-
...receipt
|
|
1392
|
+
...checkpoint.getStats(),
|
|
1393
|
+
...pick(receipt, 'transactionHash', 'blockHash')
|
|
910
1394
|
});
|
|
911
1395
|
this.metrics.recordProcessBlockTx(timer.ms(), publishStats);
|
|
912
1396
|
return true;
|
|
913
1397
|
} else {
|
|
914
1398
|
this.metrics.recordFailedTx('process');
|
|
915
|
-
this.log.error(`
|
|
916
|
-
...
|
|
917
|
-
receipt
|
|
918
|
-
txHash: receipt.transactionHash,
|
|
919
|
-
slotNumber: block.header.globalVariables.slotNumber
|
|
1399
|
+
this.log.error(`Publishing checkpoint at slot ${slot} failed with ${errorMsg ?? 'no error message'}`, undefined, {
|
|
1400
|
+
...checkpoint.getStats(),
|
|
1401
|
+
...receipt
|
|
920
1402
|
});
|
|
921
1403
|
return false;
|
|
922
1404
|
}
|