@aztec/sequencer-client 0.0.1-commit.b655e406 → 0.0.1-commit.c0b82b2
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/index.d.ts +1 -1
- 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 -26
- package/dest/config.d.ts +14 -8
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +90 -33
- package/dest/global_variable_builder/global_builder.d.ts +20 -16
- package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
- package/dest/global_variable_builder/global_builder.js +52 -39
- package/dest/global_variable_builder/index.d.ts +1 -1
- 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 +43 -20
- package/dest/publisher/config.d.ts.map +1 -1
- package/dest/publisher/config.js +109 -34
- package/dest/publisher/index.d.ts +2 -1
- package/dest/publisher/index.d.ts.map +1 -1
- package/dest/publisher/l1_tx_failed_store/factory.d.ts +11 -0
- package/dest/publisher/l1_tx_failed_store/factory.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/factory.js +22 -0
- package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts +59 -0
- package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/failed_tx_store.js +1 -0
- package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts +15 -0
- package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.js +34 -0
- package/dest/publisher/l1_tx_failed_store/index.d.ts +4 -0
- package/dest/publisher/l1_tx_failed_store/index.d.ts.map +1 -0
- package/dest/publisher/l1_tx_failed_store/index.js +2 -0
- package/dest/publisher/sequencer-publisher-factory.d.ts +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 +95 -67
- package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher.js +935 -182
- 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 +1219 -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/errors.d.ts +1 -1
- package/dest/sequencer/errors.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 +44 -3
- package/dest/sequencer/metrics.d.ts.map +1 -1
- package/dest/sequencer/metrics.js +232 -50
- package/dest/sequencer/sequencer.d.ts +122 -144
- package/dest/sequencer/sequencer.d.ts.map +1 -1
- package/dest/sequencer/sequencer.js +736 -521
- 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 +33 -30
- package/src/client/sequencer-client.ts +54 -47
- package/src/config.ts +103 -42
- package/src/global_variable_builder/global_builder.ts +67 -59
- package/src/index.ts +1 -7
- package/src/publisher/config.ts +139 -50
- package/src/publisher/index.ts +3 -0
- package/src/publisher/l1_tx_failed_store/factory.ts +32 -0
- package/src/publisher/l1_tx_failed_store/failed_tx_store.ts +55 -0
- package/src/publisher/l1_tx_failed_store/file_store_failed_tx_store.ts +46 -0
- package/src/publisher/l1_tx_failed_store/index.ts +3 -0
- package/src/publisher/sequencer-publisher-factory.ts +30 -11
- package/src/publisher/sequencer-publisher-metrics.ts +19 -71
- package/src/publisher/sequencer-publisher.ts +633 -234
- package/src/sequencer/README.md +531 -0
- package/src/sequencer/checkpoint_proposal_job.ts +926 -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 +296 -61
- package/src/sequencer/sequencer.ts +488 -711
- 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 -130
- 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 -218
- package/src/tx_validator/nullifier_cache.ts +0 -30
- package/src/tx_validator/tx_validator_factory.ts +0 -132
|
@@ -1,18 +1,397 @@
|
|
|
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';
|
|
381
|
+
import { CheckpointNumber, SlotNumber } from '@aztec/foundation/branded-types';
|
|
382
|
+
import { pick } from '@aztec/foundation/collection';
|
|
6
383
|
import { EthAddress } from '@aztec/foundation/eth-address';
|
|
7
384
|
import { Signature } from '@aztec/foundation/eth-signature';
|
|
8
385
|
import { createLogger } from '@aztec/foundation/log';
|
|
386
|
+
import { makeBackoff, retry } from '@aztec/foundation/retry';
|
|
9
387
|
import { bufferToHex } from '@aztec/foundation/string';
|
|
10
388
|
import { Timer } from '@aztec/foundation/timer';
|
|
11
389
|
import { EmpireBaseAbi, ErrorsAbi, RollupAbi } from '@aztec/l1-artifacts';
|
|
12
390
|
import { encodeSlashConsensusVotes } from '@aztec/slasher';
|
|
13
|
-
import {
|
|
14
|
-
import { getTelemetryClient } from '@aztec/telemetry-client';
|
|
15
|
-
import { encodeFunctionData, toHex } from 'viem';
|
|
391
|
+
import { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
|
|
392
|
+
import { getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
|
|
393
|
+
import { encodeFunctionData, keccak256, multicall3Abi, toHex } from 'viem';
|
|
394
|
+
import { createL1TxFailedStore } from './l1_tx_failed_store/index.js';
|
|
16
395
|
import { SequencerPublisherMetrics } from './sequencer-publisher-metrics.js';
|
|
17
396
|
export const Actions = [
|
|
18
397
|
'invalidate-by-invalid-attestation',
|
|
@@ -27,21 +406,43 @@ export const Actions = [
|
|
|
27
406
|
];
|
|
28
407
|
// Sorting for actions such that invalidations go before proposals, and proposals go before votes
|
|
29
408
|
export const compareActions = (a, b)=>Actions.indexOf(a) - Actions.indexOf(b);
|
|
409
|
+
_dec = trackSpan('SequencerPublisher.sendRequests'), _dec1 = trackSpan('SequencerPublisher.validateBlockHeader'), _dec2 = trackSpan('SequencerPublisher.validateCheckpointForSubmission');
|
|
30
410
|
export class SequencerPublisher {
|
|
31
411
|
config;
|
|
412
|
+
static{
|
|
413
|
+
({ e: [_initProto] } = _apply_decs_2203_r(this, [
|
|
414
|
+
[
|
|
415
|
+
_dec,
|
|
416
|
+
2,
|
|
417
|
+
"sendRequests"
|
|
418
|
+
],
|
|
419
|
+
[
|
|
420
|
+
_dec1,
|
|
421
|
+
2,
|
|
422
|
+
"validateBlockHeader"
|
|
423
|
+
],
|
|
424
|
+
[
|
|
425
|
+
_dec2,
|
|
426
|
+
2,
|
|
427
|
+
"validateCheckpointForSubmission"
|
|
428
|
+
]
|
|
429
|
+
], []));
|
|
430
|
+
}
|
|
32
431
|
interrupted;
|
|
33
432
|
metrics;
|
|
34
433
|
epochCache;
|
|
434
|
+
failedTxStore;
|
|
35
435
|
governanceLog;
|
|
36
436
|
slashingLog;
|
|
37
437
|
lastActions;
|
|
438
|
+
isPayloadEmptyCache;
|
|
439
|
+
payloadProposedCache;
|
|
38
440
|
log;
|
|
39
441
|
ethereumSlotDuration;
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
static PROPOSE_GAS_GUESS = 12_000_000n;
|
|
442
|
+
blobClient;
|
|
443
|
+
/** Address to use for simulations in fisherman mode (actual proposer's address) */ proposerAddressForSimulation;
|
|
444
|
+
/** L1 fee analyzer for fisherman mode */ l1FeeAnalyzer;
|
|
445
|
+
/** Fee asset price oracle for computing price modifiers from Uniswap V4 */ feeAssetPriceOracle;
|
|
45
446
|
// A CALL to a cold address is 2700 gas
|
|
46
447
|
static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
|
|
47
448
|
// Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
|
|
@@ -51,23 +452,25 @@ export class SequencerPublisher {
|
|
|
51
452
|
govProposerContract;
|
|
52
453
|
slashingProposerContract;
|
|
53
454
|
slashFactoryContract;
|
|
455
|
+
tracer;
|
|
54
456
|
requests;
|
|
55
457
|
constructor(config, deps){
|
|
56
458
|
this.config = config;
|
|
57
|
-
this.interrupted = false;
|
|
459
|
+
this.interrupted = (_initProto(this), false);
|
|
58
460
|
this.governanceLog = createLogger('sequencer:publisher:governance');
|
|
59
461
|
this.slashingLog = createLogger('sequencer:publisher:slashing');
|
|
60
462
|
this.lastActions = {};
|
|
463
|
+
this.isPayloadEmptyCache = new Map();
|
|
464
|
+
this.payloadProposedCache = new Set();
|
|
61
465
|
this.requests = [];
|
|
62
466
|
this.log = deps.log ?? createLogger('sequencer:publisher');
|
|
63
467
|
this.ethereumSlotDuration = BigInt(config.ethereumSlotDuration);
|
|
64
468
|
this.epochCache = deps.epochCache;
|
|
65
469
|
this.lastActions = deps.lastActions;
|
|
66
|
-
this.
|
|
67
|
-
logger: createLogger('sequencer:blob-sink:client')
|
|
68
|
-
});
|
|
470
|
+
this.blobClient = deps.blobClient;
|
|
69
471
|
const telemetry = deps.telemetry ?? getTelemetryClient();
|
|
70
472
|
this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
|
|
473
|
+
this.tracer = telemetry.getTracer('SequencerPublisher');
|
|
71
474
|
this.l1TxUtils = deps.l1TxUtils;
|
|
72
475
|
this.rollupContract = deps.rollupContract;
|
|
73
476
|
this.govProposerContract = deps.governanceProposerContract;
|
|
@@ -78,13 +481,54 @@ export class SequencerPublisher {
|
|
|
78
481
|
this.slashingProposerContract = newSlashingProposer;
|
|
79
482
|
});
|
|
80
483
|
this.slashFactoryContract = deps.slashFactoryContract;
|
|
484
|
+
// Initialize L1 fee analyzer for fisherman mode
|
|
485
|
+
if (config.fishermanMode) {
|
|
486
|
+
this.l1FeeAnalyzer = new L1FeeAnalyzer(this.l1TxUtils.client, deps.dateProvider, createLogger('sequencer:publisher:fee-analyzer'));
|
|
487
|
+
}
|
|
488
|
+
// Initialize fee asset price oracle
|
|
489
|
+
this.feeAssetPriceOracle = new FeeAssetPriceOracle(this.l1TxUtils.client, this.rollupContract, createLogger('sequencer:publisher:price-oracle'));
|
|
490
|
+
// Initialize failed L1 tx store (optional, for test networks)
|
|
491
|
+
this.failedTxStore = createL1TxFailedStore(config.l1TxFailedStore, this.log);
|
|
492
|
+
}
|
|
493
|
+
/**
|
|
494
|
+
* Backs up a failed L1 transaction to the configured store for debugging.
|
|
495
|
+
* Does nothing if no store is configured.
|
|
496
|
+
*/ backupFailedTx(failedTx) {
|
|
497
|
+
if (!this.failedTxStore) {
|
|
498
|
+
return;
|
|
499
|
+
}
|
|
500
|
+
const tx = {
|
|
501
|
+
...failedTx,
|
|
502
|
+
timestamp: Date.now()
|
|
503
|
+
};
|
|
504
|
+
// Fire and forget - don't block on backup
|
|
505
|
+
void this.failedTxStore.then((store)=>store?.saveFailedTx(tx)).catch((err)=>{
|
|
506
|
+
this.log.warn(`Failed to backup failed L1 tx to store`, err);
|
|
507
|
+
});
|
|
81
508
|
}
|
|
82
509
|
getRollupContract() {
|
|
83
510
|
return this.rollupContract;
|
|
84
511
|
}
|
|
512
|
+
/**
|
|
513
|
+
* Gets the fee asset price modifier from the oracle.
|
|
514
|
+
* Returns 0n if the oracle query fails.
|
|
515
|
+
*/ getFeeAssetPriceModifier() {
|
|
516
|
+
return this.feeAssetPriceOracle.computePriceModifier();
|
|
517
|
+
}
|
|
85
518
|
getSenderAddress() {
|
|
86
519
|
return this.l1TxUtils.getSenderAddress();
|
|
87
520
|
}
|
|
521
|
+
/**
|
|
522
|
+
* Gets the L1 fee analyzer instance (only available in fisherman mode)
|
|
523
|
+
*/ getL1FeeAnalyzer() {
|
|
524
|
+
return this.l1FeeAnalyzer;
|
|
525
|
+
}
|
|
526
|
+
/**
|
|
527
|
+
* Sets the proposer address to use for simulations in fisherman mode.
|
|
528
|
+
* @param proposerAddress - The actual proposer's address to use for balance lookups in simulations
|
|
529
|
+
*/ setProposerAddressForSimulation(proposerAddress) {
|
|
530
|
+
this.proposerAddressForSimulation = proposerAddress;
|
|
531
|
+
}
|
|
88
532
|
addRequest(request) {
|
|
89
533
|
this.requests.push(request);
|
|
90
534
|
}
|
|
@@ -92,6 +536,55 @@ export class SequencerPublisher {
|
|
|
92
536
|
return this.epochCache.getEpochAndSlotNow().slot;
|
|
93
537
|
}
|
|
94
538
|
/**
|
|
539
|
+
* Clears all pending requests without sending them.
|
|
540
|
+
*/ clearPendingRequests() {
|
|
541
|
+
const count = this.requests.length;
|
|
542
|
+
this.requests = [];
|
|
543
|
+
if (count > 0) {
|
|
544
|
+
this.log.debug(`Cleared ${count} pending request(s)`);
|
|
545
|
+
}
|
|
546
|
+
}
|
|
547
|
+
/**
|
|
548
|
+
* Analyzes L1 fees for the pending requests without sending them.
|
|
549
|
+
* This is used in fisherman mode to validate fee calculations.
|
|
550
|
+
* @param l2SlotNumber - The L2 slot number for this analysis
|
|
551
|
+
* @param onComplete - Optional callback to invoke when analysis completes (after block is mined)
|
|
552
|
+
* @returns The analysis result (incomplete until block mines), or undefined if no requests
|
|
553
|
+
*/ async analyzeL1Fees(l2SlotNumber, onComplete) {
|
|
554
|
+
if (!this.l1FeeAnalyzer) {
|
|
555
|
+
this.log.warn('L1 fee analyzer not available (not in fisherman mode)');
|
|
556
|
+
return undefined;
|
|
557
|
+
}
|
|
558
|
+
const requestsToAnalyze = [
|
|
559
|
+
...this.requests
|
|
560
|
+
];
|
|
561
|
+
if (requestsToAnalyze.length === 0) {
|
|
562
|
+
this.log.debug('No requests to analyze for L1 fees');
|
|
563
|
+
return undefined;
|
|
564
|
+
}
|
|
565
|
+
// Extract blob config from requests (if any)
|
|
566
|
+
const blobConfigs = requestsToAnalyze.filter((request)=>request.blobConfig).map((request)=>request.blobConfig);
|
|
567
|
+
const blobConfig = blobConfigs[0];
|
|
568
|
+
// Get gas configs
|
|
569
|
+
const gasConfigs = requestsToAnalyze.filter((request)=>request.gasConfig).map((request)=>request.gasConfig);
|
|
570
|
+
const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
|
|
571
|
+
const gasLimit = gasLimits.length > 0 ? gasLimits.reduce((sum, g)=>sum + g, 0n) : 0n;
|
|
572
|
+
// Get the transaction requests
|
|
573
|
+
const l1Requests = requestsToAnalyze.map((r)=>r.request);
|
|
574
|
+
// Start the analysis
|
|
575
|
+
const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
|
|
576
|
+
this.log.info('Started L1 fee analysis', {
|
|
577
|
+
analysisId,
|
|
578
|
+
l2SlotNumber: l2SlotNumber.toString(),
|
|
579
|
+
requestCount: requestsToAnalyze.length,
|
|
580
|
+
hasBlobConfig: !!blobConfig,
|
|
581
|
+
gasLimit: gasLimit.toString(),
|
|
582
|
+
actions: requestsToAnalyze.map((r)=>r.action)
|
|
583
|
+
});
|
|
584
|
+
// Return the analysis result (will be incomplete until block mines)
|
|
585
|
+
return this.l1FeeAnalyzer.getAnalysis(analysisId);
|
|
586
|
+
}
|
|
587
|
+
/**
|
|
95
588
|
* Sends all requests that are still valid.
|
|
96
589
|
* @returns one of:
|
|
97
590
|
* - A receipt and stats if the tx succeeded
|
|
@@ -102,7 +595,7 @@ export class SequencerPublisher {
|
|
|
102
595
|
...this.requests
|
|
103
596
|
];
|
|
104
597
|
this.requests = [];
|
|
105
|
-
if (this.interrupted) {
|
|
598
|
+
if (this.interrupted || requestsToProcess.length === 0) {
|
|
106
599
|
return undefined;
|
|
107
600
|
}
|
|
108
601
|
const currentL2Slot = this.getCurrentL2Slot();
|
|
@@ -137,7 +630,16 @@ export class SequencerPublisher {
|
|
|
137
630
|
const blobConfig = blobConfigs[0];
|
|
138
631
|
// Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
|
|
139
632
|
const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
|
|
140
|
-
|
|
633
|
+
let gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
|
|
634
|
+
// Cap at L1 block gas limit so the node accepts the tx ("gas limit too high" otherwise).
|
|
635
|
+
const maxGas = MAX_L1_TX_LIMIT;
|
|
636
|
+
if (gasLimit !== undefined && gasLimit > maxGas) {
|
|
637
|
+
this.log.debug('Capping bundled tx gas limit to L1 max', {
|
|
638
|
+
requested: gasLimit,
|
|
639
|
+
capped: maxGas
|
|
640
|
+
});
|
|
641
|
+
gasLimit = maxGas;
|
|
642
|
+
}
|
|
141
643
|
const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
|
|
142
644
|
const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
|
|
143
645
|
const txConfig = {
|
|
@@ -148,12 +650,31 @@ export class SequencerPublisher {
|
|
|
148
650
|
// This ensures the committee gets precomputed correctly
|
|
149
651
|
validRequests.sort((a, b)=>compareActions(a.action, b.action));
|
|
150
652
|
try {
|
|
653
|
+
// Capture context for failed tx backup before sending
|
|
654
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
655
|
+
const multicallData = encodeFunctionData({
|
|
656
|
+
abi: multicall3Abi,
|
|
657
|
+
functionName: 'aggregate3',
|
|
658
|
+
args: [
|
|
659
|
+
validRequests.map((r)=>({
|
|
660
|
+
target: r.request.to,
|
|
661
|
+
callData: r.request.data,
|
|
662
|
+
allowFailure: true
|
|
663
|
+
}))
|
|
664
|
+
]
|
|
665
|
+
});
|
|
666
|
+
const blobDataHex = blobConfig?.blobs?.map((b)=>toHex(b));
|
|
151
667
|
this.log.debug('Forwarding transactions', {
|
|
152
668
|
validRequests: validRequests.map((request)=>request.action),
|
|
153
669
|
txConfig
|
|
154
670
|
});
|
|
155
671
|
const result = await Multicall3.forward(validRequests.map((request)=>request.request), this.l1TxUtils, txConfig, blobConfig, this.rollupContract.address, this.log);
|
|
156
|
-
const
|
|
672
|
+
const txContext = {
|
|
673
|
+
multicallData,
|
|
674
|
+
blobData: blobDataHex,
|
|
675
|
+
l1BlockNumber
|
|
676
|
+
};
|
|
677
|
+
const { successfulActions = [], failedActions = [] } = this.callbackBundledTransactions(validRequests, result, txContext);
|
|
157
678
|
return {
|
|
158
679
|
result,
|
|
159
680
|
expiredActions,
|
|
@@ -173,10 +694,33 @@ export class SequencerPublisher {
|
|
|
173
694
|
}
|
|
174
695
|
}
|
|
175
696
|
}
|
|
176
|
-
callbackBundledTransactions(requests, result) {
|
|
697
|
+
callbackBundledTransactions(requests, result, txContext) {
|
|
177
698
|
const actionsListStr = requests.map((r)=>r.action).join(', ');
|
|
178
699
|
if (result instanceof FormattedViemError) {
|
|
179
700
|
this.log.error(`Failed to publish bundled transactions (${actionsListStr})`, result);
|
|
701
|
+
this.backupFailedTx({
|
|
702
|
+
id: keccak256(txContext.multicallData),
|
|
703
|
+
failureType: 'send-error',
|
|
704
|
+
request: {
|
|
705
|
+
to: MULTI_CALL_3_ADDRESS,
|
|
706
|
+
data: txContext.multicallData
|
|
707
|
+
},
|
|
708
|
+
blobData: txContext.blobData,
|
|
709
|
+
l1BlockNumber: txContext.l1BlockNumber.toString(),
|
|
710
|
+
error: {
|
|
711
|
+
message: result.message,
|
|
712
|
+
name: result.name
|
|
713
|
+
},
|
|
714
|
+
context: {
|
|
715
|
+
actions: requests.map((r)=>r.action),
|
|
716
|
+
requests: requests.map((r)=>({
|
|
717
|
+
action: r.action,
|
|
718
|
+
to: r.request.to,
|
|
719
|
+
data: r.request.data
|
|
720
|
+
})),
|
|
721
|
+
sender: this.getSenderAddress().toString()
|
|
722
|
+
}
|
|
723
|
+
});
|
|
180
724
|
return {
|
|
181
725
|
failedActions: requests.map((r)=>r.action)
|
|
182
726
|
};
|
|
@@ -194,6 +738,37 @@ export class SequencerPublisher {
|
|
|
194
738
|
failedActions.push(request.action);
|
|
195
739
|
}
|
|
196
740
|
}
|
|
741
|
+
// Single backup for the whole reverted tx
|
|
742
|
+
if (failedActions.length > 0 && result?.receipt?.status === 'reverted') {
|
|
743
|
+
this.backupFailedTx({
|
|
744
|
+
id: result.receipt.transactionHash,
|
|
745
|
+
failureType: 'revert',
|
|
746
|
+
request: {
|
|
747
|
+
to: MULTI_CALL_3_ADDRESS,
|
|
748
|
+
data: txContext.multicallData
|
|
749
|
+
},
|
|
750
|
+
blobData: txContext.blobData,
|
|
751
|
+
l1BlockNumber: result.receipt.blockNumber.toString(),
|
|
752
|
+
receipt: {
|
|
753
|
+
transactionHash: result.receipt.transactionHash,
|
|
754
|
+
blockNumber: result.receipt.blockNumber.toString(),
|
|
755
|
+
gasUsed: (result.receipt.gasUsed ?? 0n).toString(),
|
|
756
|
+
status: 'reverted'
|
|
757
|
+
},
|
|
758
|
+
error: {
|
|
759
|
+
message: result.errorMsg ?? 'Transaction reverted'
|
|
760
|
+
},
|
|
761
|
+
context: {
|
|
762
|
+
actions: failedActions,
|
|
763
|
+
requests: requests.filter((r)=>failedActions.includes(r.action)).map((r)=>({
|
|
764
|
+
action: r.action,
|
|
765
|
+
to: r.request.to,
|
|
766
|
+
data: r.request.data
|
|
767
|
+
})),
|
|
768
|
+
sender: this.getSenderAddress().toString()
|
|
769
|
+
}
|
|
770
|
+
});
|
|
771
|
+
}
|
|
197
772
|
return {
|
|
198
773
|
successfulActions,
|
|
199
774
|
failedActions
|
|
@@ -211,7 +786,9 @@ export class SequencerPublisher {
|
|
|
211
786
|
'InvalidProposer',
|
|
212
787
|
'InvalidArchive'
|
|
213
788
|
];
|
|
214
|
-
return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), this.ethereumSlotDuration,
|
|
789
|
+
return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
|
|
790
|
+
forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
|
|
791
|
+
}).catch((err)=>{
|
|
215
792
|
if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
|
|
216
793
|
this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
|
|
217
794
|
error: err.message
|
|
@@ -238,13 +815,23 @@ export class SequencerPublisher {
|
|
|
238
815
|
[],
|
|
239
816
|
Signature.empty().toViemSignature(),
|
|
240
817
|
`0x${'0'.repeat(64)}`,
|
|
241
|
-
header.
|
|
818
|
+
header.blobsHash.toString(),
|
|
242
819
|
flags
|
|
243
820
|
];
|
|
244
821
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
this.
|
|
822
|
+
const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
|
|
823
|
+
let balance = 0n;
|
|
824
|
+
if (this.config.fishermanMode) {
|
|
825
|
+
// In fisherman mode, we can't know where the proposer is publishing from
|
|
826
|
+
// so we just add sufficient balance to the multicall3 address
|
|
827
|
+
balance = 10n * WEI_CONST * WEI_CONST; // 10 ETH
|
|
828
|
+
} else {
|
|
829
|
+
balance = await this.l1TxUtils.getSenderBalance();
|
|
830
|
+
}
|
|
831
|
+
stateOverrides.push({
|
|
832
|
+
address: MULTI_CALL_3_ADDRESS,
|
|
833
|
+
balance
|
|
834
|
+
});
|
|
248
835
|
await this.l1TxUtils.simulate({
|
|
249
836
|
to: this.rollupContract.address,
|
|
250
837
|
data: encodeFunctionData({
|
|
@@ -255,44 +842,42 @@ export class SequencerPublisher {
|
|
|
255
842
|
from: MULTI_CALL_3_ADDRESS
|
|
256
843
|
}, {
|
|
257
844
|
time: ts + 1n
|
|
258
|
-
},
|
|
259
|
-
{
|
|
260
|
-
address: MULTI_CALL_3_ADDRESS,
|
|
261
|
-
balance
|
|
262
|
-
},
|
|
263
|
-
...await this.rollupContract.makePendingBlockNumberOverride(opts?.forcePendingBlockNumber)
|
|
264
|
-
]);
|
|
845
|
+
}, stateOverrides);
|
|
265
846
|
this.log.debug(`Simulated validateHeader`);
|
|
266
847
|
}
|
|
267
848
|
/**
|
|
268
|
-
* Simulate making a call to invalidate a
|
|
269
|
-
* @param
|
|
270
|
-
*/ async
|
|
849
|
+
* Simulate making a call to invalidate a checkpoint with invalid attestations. Returns undefined if no need to invalidate.
|
|
850
|
+
* @param validationResult - The validation result indicating which checkpoint to invalidate (as returned by the archiver)
|
|
851
|
+
*/ async simulateInvalidateCheckpoint(validationResult) {
|
|
271
852
|
if (validationResult.valid) {
|
|
272
853
|
return undefined;
|
|
273
854
|
}
|
|
274
|
-
const { reason,
|
|
275
|
-
const
|
|
855
|
+
const { reason, checkpoint } = validationResult;
|
|
856
|
+
const checkpointNumber = checkpoint.checkpointNumber;
|
|
276
857
|
const logData = {
|
|
277
|
-
...
|
|
858
|
+
...checkpoint,
|
|
278
859
|
reason
|
|
279
860
|
};
|
|
280
|
-
const
|
|
281
|
-
if (
|
|
282
|
-
this.log.verbose(`Skipping
|
|
283
|
-
|
|
861
|
+
const currentCheckpointNumber = await this.rollupContract.getCheckpointNumber();
|
|
862
|
+
if (currentCheckpointNumber < checkpointNumber) {
|
|
863
|
+
this.log.verbose(`Skipping checkpoint ${checkpointNumber} invalidation since it has already been removed from the pending chain`, {
|
|
864
|
+
currentCheckpointNumber,
|
|
284
865
|
...logData
|
|
285
866
|
});
|
|
286
867
|
return undefined;
|
|
287
868
|
}
|
|
288
|
-
const request = this.
|
|
289
|
-
this.log.debug(`Simulating invalidate
|
|
869
|
+
const request = this.buildInvalidateCheckpointRequest(validationResult);
|
|
870
|
+
this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
|
|
290
871
|
...logData,
|
|
291
872
|
request
|
|
292
873
|
});
|
|
874
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
293
875
|
try {
|
|
294
|
-
const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined,
|
|
295
|
-
|
|
876
|
+
const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, mergeAbis([
|
|
877
|
+
request.abi ?? [],
|
|
878
|
+
ErrorsAbi
|
|
879
|
+
]));
|
|
880
|
+
this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} succeeded`, {
|
|
296
881
|
...logData,
|
|
297
882
|
request,
|
|
298
883
|
gasUsed
|
|
@@ -300,91 +885,92 @@ export class SequencerPublisher {
|
|
|
300
885
|
return {
|
|
301
886
|
request,
|
|
302
887
|
gasUsed,
|
|
303
|
-
|
|
304
|
-
|
|
888
|
+
checkpointNumber,
|
|
889
|
+
forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
|
|
305
890
|
reason
|
|
306
891
|
};
|
|
307
892
|
} catch (err) {
|
|
308
893
|
const viemError = formatViemError(err);
|
|
309
|
-
// If the error is due to the
|
|
310
|
-
// we can safely ignore it and return undefined so we go ahead with
|
|
311
|
-
if (viemError.message?.includes('
|
|
312
|
-
this.log.verbose(`Simulation for invalidate
|
|
894
|
+
// If the error is due to the checkpoint not being in the pending chain, and it was indeed removed by someone else,
|
|
895
|
+
// we can safely ignore it and return undefined so we go ahead with checkpoint building.
|
|
896
|
+
if (viemError.message?.includes('Rollup__CheckpointNotInPendingChain')) {
|
|
897
|
+
this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} failed due to checkpoint not being in pending chain`, {
|
|
313
898
|
...logData,
|
|
314
899
|
request,
|
|
315
900
|
error: viemError.message
|
|
316
901
|
});
|
|
317
|
-
const
|
|
318
|
-
if (
|
|
319
|
-
this.log.verbose(`
|
|
902
|
+
const latestPendingCheckpointNumber = await this.rollupContract.getCheckpointNumber();
|
|
903
|
+
if (latestPendingCheckpointNumber < checkpointNumber) {
|
|
904
|
+
this.log.verbose(`Checkpoint ${checkpointNumber} has already been invalidated`, {
|
|
320
905
|
...logData
|
|
321
906
|
});
|
|
322
907
|
return undefined;
|
|
323
908
|
} else {
|
|
324
|
-
this.log.error(`Simulation for invalidate ${
|
|
325
|
-
throw new Error(`Failed to simulate invalidate
|
|
909
|
+
this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed and it is still in pending chain`, viemError, logData);
|
|
910
|
+
throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber} while it is still in pending chain`, {
|
|
326
911
|
cause: viemError
|
|
327
912
|
});
|
|
328
913
|
}
|
|
329
914
|
}
|
|
330
|
-
// Otherwise, throw. We cannot build the next
|
|
331
|
-
this.log.error(`Simulation for invalidate
|
|
332
|
-
|
|
915
|
+
// Otherwise, throw. We cannot build the next checkpoint if we cannot invalidate the previous one.
|
|
916
|
+
this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed`, viemError, logData);
|
|
917
|
+
this.backupFailedTx({
|
|
918
|
+
id: keccak256(request.data),
|
|
919
|
+
failureType: 'simulation',
|
|
920
|
+
request: {
|
|
921
|
+
to: request.to,
|
|
922
|
+
data: request.data,
|
|
923
|
+
value: request.value?.toString()
|
|
924
|
+
},
|
|
925
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
926
|
+
error: {
|
|
927
|
+
message: viemError.message,
|
|
928
|
+
name: viemError.name
|
|
929
|
+
},
|
|
930
|
+
context: {
|
|
931
|
+
actions: [
|
|
932
|
+
`invalidate-${reason}`
|
|
933
|
+
],
|
|
934
|
+
checkpointNumber,
|
|
935
|
+
sender: this.getSenderAddress().toString()
|
|
936
|
+
}
|
|
937
|
+
});
|
|
938
|
+
throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber}`, {
|
|
333
939
|
cause: viemError
|
|
334
940
|
});
|
|
335
941
|
}
|
|
336
942
|
}
|
|
337
|
-
|
|
943
|
+
buildInvalidateCheckpointRequest(validationResult) {
|
|
338
944
|
if (validationResult.valid) {
|
|
339
|
-
throw new Error('Cannot invalidate a valid
|
|
945
|
+
throw new Error('Cannot invalidate a valid checkpoint');
|
|
340
946
|
}
|
|
341
|
-
const {
|
|
947
|
+
const { checkpoint, committee, reason } = validationResult;
|
|
342
948
|
const logData = {
|
|
343
|
-
...
|
|
949
|
+
...checkpoint,
|
|
344
950
|
reason
|
|
345
951
|
};
|
|
346
|
-
this.log.debug(`
|
|
952
|
+
this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
|
|
347
953
|
const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
|
|
348
954
|
if (reason === 'invalid-attestation') {
|
|
349
|
-
return this.rollupContract.buildInvalidateBadAttestationRequest(
|
|
955
|
+
return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
|
|
350
956
|
} else if (reason === 'insufficient-attestations') {
|
|
351
|
-
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(
|
|
957
|
+
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
|
|
352
958
|
} else {
|
|
353
959
|
const _ = reason;
|
|
354
960
|
throw new Error(`Unknown reason for invalidation`);
|
|
355
961
|
}
|
|
356
962
|
}
|
|
357
|
-
/**
|
|
358
|
-
* @notice Will simulate `propose` to make sure that the block is valid for submission
|
|
359
|
-
*
|
|
360
|
-
* @dev Throws if unable to propose
|
|
361
|
-
*
|
|
362
|
-
* @param block - The block to propose
|
|
363
|
-
* @param attestationData - The block's attestation data
|
|
364
|
-
*
|
|
365
|
-
*/ async validateBlockForSubmission(block, attestationsAndSigners, attestationsAndSignersSignature, options) {
|
|
963
|
+
/** Simulates `propose` to make sure that the checkpoint is valid for submission */ async validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, options) {
|
|
366
964
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
|
|
370
|
-
if (ignoreSignatures) {
|
|
371
|
-
const { committee } = await this.epochCache.getCommittee(block.header.globalVariables.slotNumber.toBigInt());
|
|
372
|
-
if (!committee) {
|
|
373
|
-
this.log.warn(`No committee found for slot ${block.header.globalVariables.slotNumber.toBigInt()}`);
|
|
374
|
-
throw new Error(`No committee found for slot ${block.header.globalVariables.slotNumber.toBigInt()}`);
|
|
375
|
-
}
|
|
376
|
-
attestationsAndSigners.attestations = committee.map((committeeMember)=>CommitteeAttestation.fromAddress(committeeMember));
|
|
377
|
-
}
|
|
378
|
-
const blobFields = block.getCheckpointBlobFields();
|
|
379
|
-
const blobs = getBlobsPerL1Block(blobFields);
|
|
965
|
+
const blobFields = checkpoint.toBlobFields();
|
|
966
|
+
const blobs = await getBlobsPerL1Block(blobFields);
|
|
380
967
|
const blobInput = getPrefixedEthBlobCommitments(blobs);
|
|
381
968
|
const args = [
|
|
382
969
|
{
|
|
383
|
-
header:
|
|
384
|
-
archive: toHex(
|
|
385
|
-
stateReference: block.header.state.toViem(),
|
|
970
|
+
header: checkpoint.header.toViem(),
|
|
971
|
+
archive: toHex(checkpoint.archive.root.toBuffer()),
|
|
386
972
|
oracleInput: {
|
|
387
|
-
feeAssetPriceModifier:
|
|
973
|
+
feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
|
|
388
974
|
}
|
|
389
975
|
},
|
|
390
976
|
attestationsAndSigners.getPackedAttestations(),
|
|
@@ -409,9 +995,38 @@ export class SequencerPublisher {
|
|
|
409
995
|
}
|
|
410
996
|
const round = await base.computeRound(slotNumber);
|
|
411
997
|
const roundInfo = await base.getRoundInfo(this.rollupContract.address, round);
|
|
998
|
+
if (roundInfo.quorumReached) {
|
|
999
|
+
return false;
|
|
1000
|
+
}
|
|
412
1001
|
if (roundInfo.lastSignalSlot >= slotNumber) {
|
|
413
1002
|
return false;
|
|
414
1003
|
}
|
|
1004
|
+
if (await this.isPayloadEmpty(payload)) {
|
|
1005
|
+
this.log.warn(`Skipping vote cast for payload with empty code`);
|
|
1006
|
+
return false;
|
|
1007
|
+
}
|
|
1008
|
+
// Check if payload was already submitted to governance
|
|
1009
|
+
const cacheKey = payload.toString();
|
|
1010
|
+
if (!this.payloadProposedCache.has(cacheKey)) {
|
|
1011
|
+
try {
|
|
1012
|
+
const l1StartBlock = await this.rollupContract.getL1StartBlock();
|
|
1013
|
+
const proposed = await retry(()=>base.hasPayloadBeenProposed(payload.toString(), l1StartBlock), 'Check if payload was proposed', makeBackoff([
|
|
1014
|
+
0,
|
|
1015
|
+
1,
|
|
1016
|
+
2
|
|
1017
|
+
]), this.log, true);
|
|
1018
|
+
if (proposed) {
|
|
1019
|
+
this.payloadProposedCache.add(cacheKey);
|
|
1020
|
+
}
|
|
1021
|
+
} catch (err) {
|
|
1022
|
+
this.log.warn(`Failed to check if payload ${payload} was proposed after retries, skipping signal`, err);
|
|
1023
|
+
return false;
|
|
1024
|
+
}
|
|
1025
|
+
}
|
|
1026
|
+
if (this.payloadProposedCache.has(cacheKey)) {
|
|
1027
|
+
this.log.info(`Payload ${payload} was already proposed to governance, stopping signals`);
|
|
1028
|
+
return false;
|
|
1029
|
+
}
|
|
415
1030
|
const cachedLastVote = this.lastActions[signalType];
|
|
416
1031
|
this.lastActions[signalType] = slotNumber;
|
|
417
1032
|
const action = signalType;
|
|
@@ -422,15 +1037,41 @@ export class SequencerPublisher {
|
|
|
422
1037
|
signer: this.l1TxUtils.client.account?.address,
|
|
423
1038
|
lastValidL2Slot: slotNumber
|
|
424
1039
|
});
|
|
1040
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
425
1041
|
try {
|
|
426
1042
|
await this.l1TxUtils.simulate(request, {
|
|
427
1043
|
time: timestamp
|
|
428
|
-
}, [],
|
|
1044
|
+
}, [], mergeAbis([
|
|
1045
|
+
request.abi ?? [],
|
|
1046
|
+
ErrorsAbi
|
|
1047
|
+
]));
|
|
429
1048
|
this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
|
|
430
1049
|
request
|
|
431
1050
|
});
|
|
432
1051
|
} catch (err) {
|
|
433
|
-
|
|
1052
|
+
const viemError = formatViemError(err);
|
|
1053
|
+
this.log.error(`Failed simulation for ${action} at slot ${slotNumber} (enqueuing the action anyway)`, viemError);
|
|
1054
|
+
this.backupFailedTx({
|
|
1055
|
+
id: keccak256(request.data),
|
|
1056
|
+
failureType: 'simulation',
|
|
1057
|
+
request: {
|
|
1058
|
+
to: request.to,
|
|
1059
|
+
data: request.data,
|
|
1060
|
+
value: request.value?.toString()
|
|
1061
|
+
},
|
|
1062
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
1063
|
+
error: {
|
|
1064
|
+
message: viemError.message,
|
|
1065
|
+
name: viemError.name
|
|
1066
|
+
},
|
|
1067
|
+
context: {
|
|
1068
|
+
actions: [
|
|
1069
|
+
action
|
|
1070
|
+
],
|
|
1071
|
+
slot: slotNumber,
|
|
1072
|
+
sender: this.getSenderAddress().toString()
|
|
1073
|
+
}
|
|
1074
|
+
});
|
|
434
1075
|
// Yes, we enqueue the request anyway, in case there was a bug with the simulation itself
|
|
435
1076
|
}
|
|
436
1077
|
// TODO(palla/slash): All votes (governance and slashing) should txTimeoutAt at the end of the slot.
|
|
@@ -450,17 +1091,27 @@ export class SequencerPublisher {
|
|
|
450
1091
|
payload: payload.toString()
|
|
451
1092
|
};
|
|
452
1093
|
if (!success) {
|
|
453
|
-
this.log.error(`Signaling in
|
|
1094
|
+
this.log.error(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
|
|
454
1095
|
this.lastActions[signalType] = cachedLastVote;
|
|
455
1096
|
return false;
|
|
456
1097
|
} else {
|
|
457
|
-
this.log.info(`Signaling in
|
|
1098
|
+
this.log.info(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
|
|
458
1099
|
return true;
|
|
459
1100
|
}
|
|
460
1101
|
}
|
|
461
1102
|
});
|
|
462
1103
|
return true;
|
|
463
1104
|
}
|
|
1105
|
+
async isPayloadEmpty(payload) {
|
|
1106
|
+
const key = payload.toString();
|
|
1107
|
+
const cached = this.isPayloadEmptyCache.get(key);
|
|
1108
|
+
if (cached) {
|
|
1109
|
+
return cached;
|
|
1110
|
+
}
|
|
1111
|
+
const isEmpty = !await this.l1TxUtils.getCode(payload);
|
|
1112
|
+
this.isPayloadEmptyCache.set(key, isEmpty);
|
|
1113
|
+
return isEmpty;
|
|
1114
|
+
}
|
|
464
1115
|
/**
|
|
465
1116
|
* Enqueues a governance castSignal transaction to cast a signal for a given slot number.
|
|
466
1117
|
* @param slotNumber - The slot number to cast a signal for.
|
|
@@ -556,23 +1207,17 @@ export class SequencerPublisher {
|
|
|
556
1207
|
}
|
|
557
1208
|
return true;
|
|
558
1209
|
}
|
|
559
|
-
/**
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
* @returns True if the tx has been enqueued, throws otherwise. See #9315
|
|
564
|
-
*/ async enqueueProposeL2Block(block, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
|
|
565
|
-
const checkpointHeader = block.getCheckpointHeader();
|
|
566
|
-
const blobFields = block.getCheckpointBlobFields();
|
|
567
|
-
const blobs = getBlobsPerL1Block(blobFields);
|
|
1210
|
+
/** Simulates and enqueues a proposal for a checkpoint on L1 */ async enqueueProposeCheckpoint(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
|
|
1211
|
+
const checkpointHeader = checkpoint.header;
|
|
1212
|
+
const blobFields = checkpoint.toBlobFields();
|
|
1213
|
+
const blobs = await getBlobsPerL1Block(blobFields);
|
|
568
1214
|
const proposeTxArgs = {
|
|
569
1215
|
header: checkpointHeader,
|
|
570
|
-
archive:
|
|
571
|
-
stateReference: block.header.state,
|
|
572
|
-
body: block.body.toBuffer(),
|
|
1216
|
+
archive: checkpoint.archive.root.toBuffer(),
|
|
573
1217
|
blobs,
|
|
574
1218
|
attestationsAndSigners,
|
|
575
|
-
attestationsAndSignersSignature
|
|
1219
|
+
attestationsAndSignersSignature,
|
|
1220
|
+
feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
|
|
576
1221
|
};
|
|
577
1222
|
let ts;
|
|
578
1223
|
try {
|
|
@@ -581,36 +1226,35 @@ export class SequencerPublisher {
|
|
|
581
1226
|
// By simulation issue, I mean the fact that the block.timestamp is equal to the last block, not the next, which
|
|
582
1227
|
// make time consistency checks break.
|
|
583
1228
|
// TODO(palla): Check whether we're validating twice, once here and once within addProposeTx, since we call simulateProposeTx in both places.
|
|
584
|
-
ts = await this.
|
|
1229
|
+
ts = await this.validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts);
|
|
585
1230
|
} catch (err) {
|
|
586
|
-
this.log.error(`
|
|
587
|
-
...
|
|
588
|
-
slotNumber:
|
|
589
|
-
|
|
1231
|
+
this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
|
|
1232
|
+
...checkpoint.getStats(),
|
|
1233
|
+
slotNumber: checkpoint.header.slotNumber,
|
|
1234
|
+
forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
|
|
590
1235
|
});
|
|
591
1236
|
throw err;
|
|
592
1237
|
}
|
|
593
|
-
this.log.verbose(`Enqueuing
|
|
594
|
-
...
|
|
1238
|
+
this.log.verbose(`Enqueuing checkpoint propose transaction`, {
|
|
1239
|
+
...checkpoint.toCheckpointInfo(),
|
|
595
1240
|
...opts
|
|
596
1241
|
});
|
|
597
|
-
await this.addProposeTx(
|
|
598
|
-
return true;
|
|
1242
|
+
await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
|
|
599
1243
|
}
|
|
600
|
-
|
|
1244
|
+
enqueueInvalidateCheckpoint(request, opts = {}) {
|
|
601
1245
|
if (!request) {
|
|
602
1246
|
return;
|
|
603
1247
|
}
|
|
604
1248
|
// We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
|
|
605
1249
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
|
|
606
|
-
const { gasUsed,
|
|
1250
|
+
const { gasUsed, checkpointNumber } = request;
|
|
607
1251
|
const logData = {
|
|
608
1252
|
gasUsed,
|
|
609
|
-
|
|
1253
|
+
checkpointNumber,
|
|
610
1254
|
gasLimit,
|
|
611
1255
|
opts
|
|
612
1256
|
};
|
|
613
|
-
this.log.verbose(`Enqueuing invalidate
|
|
1257
|
+
this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
|
|
614
1258
|
this.addRequest({
|
|
615
1259
|
action: `invalidate-by-${request.reason}`,
|
|
616
1260
|
request: request.request,
|
|
@@ -618,16 +1262,16 @@ export class SequencerPublisher {
|
|
|
618
1262
|
gasLimit,
|
|
619
1263
|
txTimeoutAt: opts.txTimeoutAt
|
|
620
1264
|
},
|
|
621
|
-
lastValidL2Slot: this.getCurrentL2Slot() +
|
|
1265
|
+
lastValidL2Slot: SlotNumber(this.getCurrentL2Slot() + 2),
|
|
622
1266
|
checkSuccess: (_req, result)=>{
|
|
623
|
-
const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, '
|
|
1267
|
+
const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
|
|
624
1268
|
if (!success) {
|
|
625
|
-
this.log.warn(`Invalidate
|
|
1269
|
+
this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
|
|
626
1270
|
...result,
|
|
627
1271
|
...logData
|
|
628
1272
|
});
|
|
629
1273
|
} else {
|
|
630
|
-
this.log.info(`Invalidate
|
|
1274
|
+
this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
|
|
631
1275
|
...result,
|
|
632
1276
|
...logData
|
|
633
1277
|
});
|
|
@@ -649,28 +1293,60 @@ export class SequencerPublisher {
|
|
|
649
1293
|
const cachedLastActionSlot = this.lastActions[action];
|
|
650
1294
|
this.lastActions[action] = slotNumber;
|
|
651
1295
|
this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
|
|
1296
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
652
1297
|
let gasUsed;
|
|
1298
|
+
const simulateAbi = mergeAbis([
|
|
1299
|
+
request.abi ?? [],
|
|
1300
|
+
ErrorsAbi
|
|
1301
|
+
]);
|
|
653
1302
|
try {
|
|
654
1303
|
({ gasUsed } = await this.l1TxUtils.simulate(request, {
|
|
655
1304
|
time: timestamp
|
|
656
|
-
}, [],
|
|
1305
|
+
}, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
|
|
657
1306
|
this.log.verbose(`Simulation for ${action} succeeded`, {
|
|
658
1307
|
...logData,
|
|
659
1308
|
request,
|
|
660
1309
|
gasUsed
|
|
661
1310
|
});
|
|
662
1311
|
} catch (err) {
|
|
663
|
-
const viemError = formatViemError(err);
|
|
1312
|
+
const viemError = formatViemError(err, simulateAbi);
|
|
664
1313
|
this.log.error(`Simulation for ${action} at ${slotNumber} failed`, viemError, logData);
|
|
1314
|
+
this.backupFailedTx({
|
|
1315
|
+
id: keccak256(request.data),
|
|
1316
|
+
failureType: 'simulation',
|
|
1317
|
+
request: {
|
|
1318
|
+
to: request.to,
|
|
1319
|
+
data: request.data,
|
|
1320
|
+
value: request.value?.toString()
|
|
1321
|
+
},
|
|
1322
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
1323
|
+
error: {
|
|
1324
|
+
message: viemError.message,
|
|
1325
|
+
name: viemError.name
|
|
1326
|
+
},
|
|
1327
|
+
context: {
|
|
1328
|
+
actions: [
|
|
1329
|
+
action
|
|
1330
|
+
],
|
|
1331
|
+
slot: slotNumber,
|
|
1332
|
+
sender: this.getSenderAddress().toString()
|
|
1333
|
+
}
|
|
1334
|
+
});
|
|
665
1335
|
return false;
|
|
666
1336
|
}
|
|
667
1337
|
// We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
|
|
668
1338
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
|
|
669
1339
|
logData.gasLimit = gasLimit;
|
|
1340
|
+
// Store the ABI used for simulation on the request so Multicall3.forward can decode errors
|
|
1341
|
+
// when the tx is sent and a revert is diagnosed via simulation.
|
|
1342
|
+
const requestWithAbi = {
|
|
1343
|
+
...request,
|
|
1344
|
+
abi: simulateAbi
|
|
1345
|
+
};
|
|
670
1346
|
this.log.debug(`Enqueuing ${action}`, logData);
|
|
671
1347
|
this.addRequest({
|
|
672
1348
|
action,
|
|
673
|
-
request,
|
|
1349
|
+
request: requestWithAbi,
|
|
674
1350
|
gasConfig: {
|
|
675
1351
|
gasLimit
|
|
676
1352
|
},
|
|
@@ -713,34 +1389,70 @@ export class SequencerPublisher {
|
|
|
713
1389
|
this.log.debug('Validating blob input', {
|
|
714
1390
|
blobInput
|
|
715
1391
|
});
|
|
716
|
-
|
|
717
|
-
|
|
718
|
-
|
|
719
|
-
|
|
720
|
-
|
|
721
|
-
|
|
722
|
-
|
|
723
|
-
|
|
724
|
-
|
|
725
|
-
|
|
726
|
-
|
|
727
|
-
|
|
728
|
-
|
|
729
|
-
|
|
730
|
-
|
|
731
|
-
|
|
1392
|
+
// Get blob evaluation gas
|
|
1393
|
+
let blobEvaluationGas;
|
|
1394
|
+
if (this.config.fishermanMode) {
|
|
1395
|
+
// In fisherman mode, we can't estimate blob gas because estimateGas doesn't support state overrides
|
|
1396
|
+
// Use a fixed estimate.
|
|
1397
|
+
blobEvaluationGas = BigInt(encodedData.blobs.length) * 21_000n;
|
|
1398
|
+
this.log.debug(`Using fixed blob evaluation gas estimate in fisherman mode: ${blobEvaluationGas}`);
|
|
1399
|
+
} else {
|
|
1400
|
+
// Normal mode - use estimateGas with blob inputs
|
|
1401
|
+
blobEvaluationGas = await this.l1TxUtils.estimateGas(this.getSenderAddress().toString(), {
|
|
1402
|
+
to: this.rollupContract.address,
|
|
1403
|
+
data: encodeFunctionData({
|
|
1404
|
+
abi: RollupAbi,
|
|
1405
|
+
functionName: 'validateBlobs',
|
|
1406
|
+
args: [
|
|
1407
|
+
blobInput
|
|
1408
|
+
]
|
|
1409
|
+
})
|
|
1410
|
+
}, {}, {
|
|
1411
|
+
blobs: encodedData.blobs.map((b)=>b.data),
|
|
1412
|
+
kzg
|
|
1413
|
+
}).catch(async (err)=>{
|
|
1414
|
+
const viemError = formatViemError(err);
|
|
1415
|
+
this.log.error(`Failed to validate blobs`, viemError.message, {
|
|
1416
|
+
metaMessages: viemError.metaMessages
|
|
1417
|
+
});
|
|
1418
|
+
const validateBlobsData = encodeFunctionData({
|
|
1419
|
+
abi: RollupAbi,
|
|
1420
|
+
functionName: 'validateBlobs',
|
|
1421
|
+
args: [
|
|
1422
|
+
blobInput
|
|
1423
|
+
]
|
|
1424
|
+
});
|
|
1425
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
1426
|
+
this.backupFailedTx({
|
|
1427
|
+
id: keccak256(validateBlobsData),
|
|
1428
|
+
failureType: 'simulation',
|
|
1429
|
+
request: {
|
|
1430
|
+
to: this.rollupContract.address,
|
|
1431
|
+
data: validateBlobsData
|
|
1432
|
+
},
|
|
1433
|
+
blobData: encodedData.blobs.map((b)=>toHex(b.data)),
|
|
1434
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
1435
|
+
error: {
|
|
1436
|
+
message: viemError.message,
|
|
1437
|
+
name: viemError.name
|
|
1438
|
+
},
|
|
1439
|
+
context: {
|
|
1440
|
+
actions: [
|
|
1441
|
+
'validate-blobs'
|
|
1442
|
+
],
|
|
1443
|
+
sender: this.getSenderAddress().toString()
|
|
1444
|
+
}
|
|
1445
|
+
});
|
|
1446
|
+
throw new Error('Failed to validate blobs');
|
|
732
1447
|
});
|
|
733
|
-
|
|
734
|
-
});
|
|
1448
|
+
}
|
|
735
1449
|
const signers = encodedData.attestationsAndSigners.getSigners().map((signer)=>signer.toString());
|
|
736
1450
|
const args = [
|
|
737
1451
|
{
|
|
738
1452
|
header: encodedData.header.toViem(),
|
|
739
1453
|
archive: toHex(encodedData.archive),
|
|
740
|
-
stateReference: encodedData.stateReference.toViem(),
|
|
741
1454
|
oracleInput: {
|
|
742
|
-
|
|
743
|
-
feeAssetPriceModifier: 0n
|
|
1455
|
+
feeAssetPriceModifier: encodedData.feeAssetPriceModifier
|
|
744
1456
|
}
|
|
745
1457
|
},
|
|
746
1458
|
encodedData.attestationsAndSigners.getPackedAttestations(),
|
|
@@ -767,18 +1479,9 @@ export class SequencerPublisher {
|
|
|
767
1479
|
functionName: 'propose',
|
|
768
1480
|
args
|
|
769
1481
|
});
|
|
770
|
-
// override the pending
|
|
771
|
-
const
|
|
772
|
-
const
|
|
773
|
-
to: this.rollupContract.address,
|
|
774
|
-
data: rollupData,
|
|
775
|
-
gas: SequencerPublisher.PROPOSE_GAS_GUESS
|
|
776
|
-
}, {
|
|
777
|
-
// @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
|
|
778
|
-
time: timestamp + 1n,
|
|
779
|
-
// @note reth should have a 30m gas limit per block but throws errors that this tx is beyond limit so we increase here
|
|
780
|
-
gasLimit: SequencerPublisher.PROPOSE_GAS_GUESS * 2n
|
|
781
|
-
}, [
|
|
1482
|
+
// override the pending checkpoint number if requested
|
|
1483
|
+
const forcePendingCheckpointNumberStateDiff = (options.forcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
|
|
1484
|
+
const stateOverrides = [
|
|
782
1485
|
{
|
|
783
1486
|
address: this.rollupContract.address,
|
|
784
1487
|
// @note we override checkBlob to false since blobs are not part simulate()
|
|
@@ -787,14 +1490,65 @@ export class SequencerPublisher {
|
|
|
787
1490
|
slot: toPaddedHex(RollupContract.checkBlobStorageSlot, true),
|
|
788
1491
|
value: toPaddedHex(0n, true)
|
|
789
1492
|
},
|
|
790
|
-
...
|
|
1493
|
+
...forcePendingCheckpointNumberStateDiff
|
|
791
1494
|
]
|
|
792
1495
|
}
|
|
793
|
-
]
|
|
1496
|
+
];
|
|
1497
|
+
// In fisherman mode, simulate as the proposer but with sufficient balance
|
|
1498
|
+
if (this.proposerAddressForSimulation) {
|
|
1499
|
+
stateOverrides.push({
|
|
1500
|
+
address: this.proposerAddressForSimulation.toString(),
|
|
1501
|
+
balance: 10n * WEI_CONST * WEI_CONST
|
|
1502
|
+
});
|
|
1503
|
+
}
|
|
1504
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
1505
|
+
const simulationResult = await this.l1TxUtils.simulate({
|
|
1506
|
+
to: this.rollupContract.address,
|
|
1507
|
+
data: rollupData,
|
|
1508
|
+
gas: MAX_L1_TX_LIMIT,
|
|
1509
|
+
...this.proposerAddressForSimulation && {
|
|
1510
|
+
from: this.proposerAddressForSimulation.toString()
|
|
1511
|
+
}
|
|
1512
|
+
}, {
|
|
1513
|
+
// @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
|
|
1514
|
+
time: timestamp + 1n,
|
|
1515
|
+
// @note reth should have a 30m gas limit per block but throws errors that this tx is beyond limit so we increase here
|
|
1516
|
+
gasLimit: MAX_L1_TX_LIMIT * 2n
|
|
1517
|
+
}, stateOverrides, RollupAbi, {
|
|
794
1518
|
// @note fallback gas estimate to use if the node doesn't support simulation API
|
|
795
|
-
fallbackGasEstimate:
|
|
1519
|
+
fallbackGasEstimate: MAX_L1_TX_LIMIT
|
|
796
1520
|
}).catch((err)=>{
|
|
797
|
-
|
|
1521
|
+
// In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
|
|
1522
|
+
const viemError = formatViemError(err);
|
|
1523
|
+
if (this.config.fishermanMode && viemError.message?.includes('ValidatorSelection__MissingProposerSignature')) {
|
|
1524
|
+
this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
|
|
1525
|
+
// Return a minimal simulation result with the fallback gas estimate
|
|
1526
|
+
return {
|
|
1527
|
+
gasUsed: MAX_L1_TX_LIMIT,
|
|
1528
|
+
logs: []
|
|
1529
|
+
};
|
|
1530
|
+
}
|
|
1531
|
+
this.log.error(`Failed to simulate propose tx`, viemError);
|
|
1532
|
+
this.backupFailedTx({
|
|
1533
|
+
id: keccak256(rollupData),
|
|
1534
|
+
failureType: 'simulation',
|
|
1535
|
+
request: {
|
|
1536
|
+
to: this.rollupContract.address,
|
|
1537
|
+
data: rollupData
|
|
1538
|
+
},
|
|
1539
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
1540
|
+
error: {
|
|
1541
|
+
message: viemError.message,
|
|
1542
|
+
name: viemError.name
|
|
1543
|
+
},
|
|
1544
|
+
context: {
|
|
1545
|
+
actions: [
|
|
1546
|
+
'propose'
|
|
1547
|
+
],
|
|
1548
|
+
slot: Number(args[0].header.slotNumber),
|
|
1549
|
+
sender: this.getSenderAddress().toString()
|
|
1550
|
+
}
|
|
1551
|
+
});
|
|
798
1552
|
throw err;
|
|
799
1553
|
});
|
|
800
1554
|
return {
|
|
@@ -802,24 +1556,25 @@ export class SequencerPublisher {
|
|
|
802
1556
|
simulationResult
|
|
803
1557
|
};
|
|
804
1558
|
}
|
|
805
|
-
async addProposeTx(
|
|
1559
|
+
async addProposeTx(checkpoint, encodedData, opts = {}, timestamp) {
|
|
1560
|
+
const slot = checkpoint.header.slotNumber;
|
|
806
1561
|
const timer = new Timer();
|
|
807
1562
|
const kzg = Blob.getViemKzgInstance();
|
|
808
1563
|
const { rollupData, simulationResult, blobEvaluationGas } = await this.prepareProposeTx(encodedData, timestamp, opts);
|
|
809
1564
|
const startBlock = await this.l1TxUtils.getBlockNumber();
|
|
810
1565
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(simulationResult.gasUsed) * 64 / 63)) + blobEvaluationGas + SequencerPublisher.MULTICALL_OVERHEAD_GAS_GUESS);
|
|
811
|
-
// Send the blobs to the blob
|
|
812
|
-
// tx fails but it does get mined. We make sure that the blobs are sent to the blob
|
|
813
|
-
void this.
|
|
814
|
-
|
|
815
|
-
|
|
1566
|
+
// Send the blobs to the blob client preemptively. This helps in tests where the sequencer mistakingly thinks that the propose
|
|
1567
|
+
// tx fails but it does get mined. We make sure that the blobs are sent to the blob client regardless of the tx outcome.
|
|
1568
|
+
void Promise.resolve().then(()=>this.blobClient.sendBlobsToFilestore(encodedData.blobs).catch((_err)=>{
|
|
1569
|
+
this.log.error('Failed to send blobs to blob client');
|
|
1570
|
+
}));
|
|
816
1571
|
return this.addRequest({
|
|
817
1572
|
action: 'propose',
|
|
818
1573
|
request: {
|
|
819
1574
|
to: this.rollupContract.address,
|
|
820
1575
|
data: rollupData
|
|
821
1576
|
},
|
|
822
|
-
lastValidL2Slot:
|
|
1577
|
+
lastValidL2Slot: checkpoint.header.slotNumber,
|
|
823
1578
|
gasConfig: {
|
|
824
1579
|
...opts,
|
|
825
1580
|
gasLimit
|
|
@@ -833,7 +1588,7 @@ export class SequencerPublisher {
|
|
|
833
1588
|
return false;
|
|
834
1589
|
}
|
|
835
1590
|
const { receipt, stats, errorMsg } = result;
|
|
836
|
-
const success = receipt && receipt.status === 'success' && tryExtractEvent(receipt.logs, this.rollupContract.address, RollupAbi, '
|
|
1591
|
+
const success = receipt && receipt.status === 'success' && tryExtractEvent(receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointProposed');
|
|
837
1592
|
if (success) {
|
|
838
1593
|
const endBlock = receipt.blockNumber;
|
|
839
1594
|
const inclusionBlocks = Number(endBlock - startBlock);
|
|
@@ -847,25 +1602,23 @@ export class SequencerPublisher {
|
|
|
847
1602
|
calldataGas,
|
|
848
1603
|
calldataSize,
|
|
849
1604
|
sender,
|
|
850
|
-
...
|
|
1605
|
+
...checkpoint.getStats(),
|
|
851
1606
|
eventName: 'rollup-published-to-l1',
|
|
852
1607
|
blobCount: encodedData.blobs.length,
|
|
853
1608
|
inclusionBlocks
|
|
854
1609
|
};
|
|
855
|
-
this.log.info(`Published
|
|
1610
|
+
this.log.info(`Published checkpoint ${checkpoint.number} at slot ${slot} to rollup contract`, {
|
|
856
1611
|
...stats,
|
|
857
|
-
...
|
|
858
|
-
...receipt
|
|
1612
|
+
...checkpoint.getStats(),
|
|
1613
|
+
...pick(receipt, 'transactionHash', 'blockHash')
|
|
859
1614
|
});
|
|
860
1615
|
this.metrics.recordProcessBlockTx(timer.ms(), publishStats);
|
|
861
1616
|
return true;
|
|
862
1617
|
} else {
|
|
863
1618
|
this.metrics.recordFailedTx('process');
|
|
864
|
-
this.log.error(`
|
|
865
|
-
...
|
|
866
|
-
receipt
|
|
867
|
-
txHash: receipt.transactionHash,
|
|
868
|
-
slotNumber: block.header.globalVariables.slotNumber.toBigInt()
|
|
1619
|
+
this.log.error(`Publishing checkpoint at slot ${slot} failed with ${errorMsg ?? 'no error message'}`, undefined, {
|
|
1620
|
+
...checkpoint.getStats(),
|
|
1621
|
+
...receipt
|
|
869
1622
|
});
|
|
870
1623
|
return false;
|
|
871
1624
|
}
|