@aztec/sequencer-client 0.0.1-commit.fce3e4f → 0.0.1-commit.ffe5b04ea
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 +32 -16
- package/dest/client/sequencer-client.d.ts.map +1 -1
- package/dest/client/sequencer-client.js +118 -28
- package/dest/config.d.ts +33 -8
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +99 -44
- package/dest/global_variable_builder/global_builder.d.ts +20 -13
- package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
- package/dest/global_variable_builder/global_builder.js +51 -41
- package/dest/index.d.ts +2 -3
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +1 -2
- package/dest/publisher/config.d.ts +41 -20
- package/dest/publisher/config.d.ts.map +1 -1
- package/dest/publisher/config.js +109 -39
- 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 +28 -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 +73 -47
- package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
- package/dest/publisher/sequencer-publisher.js +888 -146
- package/dest/sequencer/checkpoint_proposal_job.d.ts +100 -0
- package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -0
- package/dest/sequencer/checkpoint_proposal_job.js +1244 -0
- package/dest/sequencer/checkpoint_voter.d.ts +35 -0
- package/dest/sequencer/checkpoint_voter.d.ts.map +1 -0
- package/dest/sequencer/checkpoint_voter.js +109 -0
- package/dest/sequencer/config.d.ts +3 -2
- package/dest/sequencer/config.d.ts.map +1 -1
- package/dest/sequencer/events.d.ts +46 -0
- package/dest/sequencer/events.d.ts.map +1 -0
- package/dest/sequencer/events.js +1 -0
- package/dest/sequencer/index.d.ts +4 -2
- package/dest/sequencer/index.d.ts.map +1 -1
- package/dest/sequencer/index.js +3 -1
- package/dest/sequencer/metrics.d.ts +38 -6
- package/dest/sequencer/metrics.d.ts.map +1 -1
- package/dest/sequencer/metrics.js +216 -72
- package/dest/sequencer/sequencer.d.ts +122 -133
- package/dest/sequencer/sequencer.d.ts.map +1 -1
- package/dest/sequencer/sequencer.js +717 -625
- package/dest/sequencer/timetable.d.ts +54 -16
- package/dest/sequencer/timetable.d.ts.map +1 -1
- package/dest/sequencer/timetable.js +147 -62
- package/dest/sequencer/types.d.ts +6 -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 +95 -0
- package/dest/test/mock_checkpoint_builder.d.ts.map +1 -0
- package/dest/test/mock_checkpoint_builder.js +231 -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 +158 -52
- package/src/config.ts +114 -54
- package/src/global_variable_builder/global_builder.ts +65 -61
- package/src/index.ts +1 -7
- package/src/publisher/config.ts +131 -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 +43 -10
- package/src/publisher/sequencer-publisher-metrics.ts +19 -71
- package/src/publisher/sequencer-publisher.ts +587 -191
- package/src/sequencer/README.md +531 -0
- package/src/sequencer/checkpoint_proposal_job.ts +960 -0
- package/src/sequencer/checkpoint_voter.ts +130 -0
- package/src/sequencer/config.ts +2 -1
- package/src/sequencer/events.ts +27 -0
- package/src/sequencer/index.ts +3 -1
- package/src/sequencer/metrics.ts +268 -82
- package/src/sequencer/sequencer.ts +464 -831
- package/src/sequencer/timetable.ts +178 -83
- package/src/sequencer/types.ts +9 -0
- package/src/sequencer/utils.ts +18 -9
- package/src/test/index.ts +5 -6
- package/src/test/mock_checkpoint_builder.ts +323 -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,19 +1,398 @@
|
|
|
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';
|
|
383
|
+
import { TimeoutError } from '@aztec/foundation/error';
|
|
7
384
|
import { EthAddress } from '@aztec/foundation/eth-address';
|
|
8
385
|
import { Signature } from '@aztec/foundation/eth-signature';
|
|
9
386
|
import { createLogger } from '@aztec/foundation/log';
|
|
387
|
+
import { makeBackoff, retry } from '@aztec/foundation/retry';
|
|
10
388
|
import { bufferToHex } from '@aztec/foundation/string';
|
|
11
389
|
import { Timer } from '@aztec/foundation/timer';
|
|
12
390
|
import { EmpireBaseAbi, ErrorsAbi, RollupAbi } from '@aztec/l1-artifacts';
|
|
13
391
|
import { encodeSlashConsensusVotes } from '@aztec/slasher';
|
|
14
|
-
import {
|
|
15
|
-
import { getTelemetryClient } from '@aztec/telemetry-client';
|
|
16
|
-
import { encodeFunctionData, toHex } from 'viem';
|
|
392
|
+
import { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
|
|
393
|
+
import { getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
|
|
394
|
+
import { encodeFunctionData, keccak256, multicall3Abi, toHex } from 'viem';
|
|
395
|
+
import { createL1TxFailedStore } from './l1_tx_failed_store/index.js';
|
|
17
396
|
import { SequencerPublisherMetrics } from './sequencer-publisher-metrics.js';
|
|
18
397
|
export const Actions = [
|
|
19
398
|
'invalidate-by-invalid-attestation',
|
|
@@ -28,22 +407,44 @@ export const Actions = [
|
|
|
28
407
|
];
|
|
29
408
|
// Sorting for actions such that invalidations go before proposals, and proposals go before votes
|
|
30
409
|
export const compareActions = (a, b)=>Actions.indexOf(a) - Actions.indexOf(b);
|
|
410
|
+
_dec = trackSpan('SequencerPublisher.sendRequests'), _dec1 = trackSpan('SequencerPublisher.validateBlockHeader'), _dec2 = trackSpan('SequencerPublisher.validateCheckpointForSubmission');
|
|
31
411
|
export class SequencerPublisher {
|
|
32
412
|
config;
|
|
413
|
+
static{
|
|
414
|
+
({ e: [_initProto] } = _apply_decs_2203_r(this, [
|
|
415
|
+
[
|
|
416
|
+
_dec,
|
|
417
|
+
2,
|
|
418
|
+
"sendRequests"
|
|
419
|
+
],
|
|
420
|
+
[
|
|
421
|
+
_dec1,
|
|
422
|
+
2,
|
|
423
|
+
"validateBlockHeader"
|
|
424
|
+
],
|
|
425
|
+
[
|
|
426
|
+
_dec2,
|
|
427
|
+
2,
|
|
428
|
+
"validateCheckpointForSubmission"
|
|
429
|
+
]
|
|
430
|
+
], []));
|
|
431
|
+
}
|
|
33
432
|
interrupted;
|
|
34
433
|
metrics;
|
|
35
434
|
epochCache;
|
|
435
|
+
failedTxStore;
|
|
36
436
|
governanceLog;
|
|
37
437
|
slashingLog;
|
|
38
438
|
lastActions;
|
|
439
|
+
isPayloadEmptyCache;
|
|
440
|
+
payloadProposedCache;
|
|
39
441
|
log;
|
|
40
442
|
ethereumSlotDuration;
|
|
41
|
-
|
|
443
|
+
blobClient;
|
|
42
444
|
/** Address to use for simulations in fisherman mode (actual proposer's address) */ proposerAddressForSimulation;
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
static PROPOSE_GAS_GUESS = 12_000_000n;
|
|
445
|
+
/** Optional callback to obtain a replacement publisher when the current one fails to send. */ getNextPublisher;
|
|
446
|
+
/** L1 fee analyzer for fisherman mode */ l1FeeAnalyzer;
|
|
447
|
+
/** Fee asset price oracle for computing price modifiers from Uniswap V4 */ feeAssetPriceOracle;
|
|
47
448
|
// A CALL to a cold address is 2700 gas
|
|
48
449
|
static MULTICALL_OVERHEAD_GAS_GUESS = 5000n;
|
|
49
450
|
// Gas report for VotingWithSigTest shows a max gas of 100k, but we've seen it cost 700k+ in testnet
|
|
@@ -53,24 +454,27 @@ export class SequencerPublisher {
|
|
|
53
454
|
govProposerContract;
|
|
54
455
|
slashingProposerContract;
|
|
55
456
|
slashFactoryContract;
|
|
457
|
+
tracer;
|
|
56
458
|
requests;
|
|
57
459
|
constructor(config, deps){
|
|
58
460
|
this.config = config;
|
|
59
|
-
this.interrupted = false;
|
|
461
|
+
this.interrupted = (_initProto(this), false);
|
|
60
462
|
this.governanceLog = createLogger('sequencer:publisher:governance');
|
|
61
463
|
this.slashingLog = createLogger('sequencer:publisher:slashing');
|
|
62
464
|
this.lastActions = {};
|
|
465
|
+
this.isPayloadEmptyCache = new Map();
|
|
466
|
+
this.payloadProposedCache = new Set();
|
|
63
467
|
this.requests = [];
|
|
64
468
|
this.log = deps.log ?? createLogger('sequencer:publisher');
|
|
65
469
|
this.ethereumSlotDuration = BigInt(config.ethereumSlotDuration);
|
|
66
470
|
this.epochCache = deps.epochCache;
|
|
67
471
|
this.lastActions = deps.lastActions;
|
|
68
|
-
this.
|
|
69
|
-
logger: createLogger('sequencer:blob-sink:client')
|
|
70
|
-
});
|
|
472
|
+
this.blobClient = deps.blobClient;
|
|
71
473
|
const telemetry = deps.telemetry ?? getTelemetryClient();
|
|
72
474
|
this.metrics = deps.metrics ?? new SequencerPublisherMetrics(telemetry, 'SequencerPublisher');
|
|
475
|
+
this.tracer = telemetry.getTracer('SequencerPublisher');
|
|
73
476
|
this.l1TxUtils = deps.l1TxUtils;
|
|
477
|
+
this.getNextPublisher = deps.getNextPublisher;
|
|
74
478
|
this.rollupContract = deps.rollupContract;
|
|
75
479
|
this.govProposerContract = deps.governanceProposerContract;
|
|
76
480
|
this.slashingProposerContract = deps.slashingProposerContract;
|
|
@@ -80,14 +484,49 @@ export class SequencerPublisher {
|
|
|
80
484
|
this.slashingProposerContract = newSlashingProposer;
|
|
81
485
|
});
|
|
82
486
|
this.slashFactoryContract = deps.slashFactoryContract;
|
|
487
|
+
// Initialize L1 fee analyzer for fisherman mode
|
|
488
|
+
if (config.fishermanMode) {
|
|
489
|
+
this.l1FeeAnalyzer = new L1FeeAnalyzer(this.l1TxUtils.client, deps.dateProvider, createLogger('sequencer:publisher:fee-analyzer'));
|
|
490
|
+
}
|
|
491
|
+
// Initialize fee asset price oracle
|
|
492
|
+
this.feeAssetPriceOracle = new FeeAssetPriceOracle(this.l1TxUtils.client, this.rollupContract, createLogger('sequencer:publisher:price-oracle'));
|
|
493
|
+
// Initialize failed L1 tx store (optional, for test networks)
|
|
494
|
+
this.failedTxStore = createL1TxFailedStore(config.l1TxFailedStore, this.log);
|
|
495
|
+
}
|
|
496
|
+
/**
|
|
497
|
+
* Backs up a failed L1 transaction to the configured store for debugging.
|
|
498
|
+
* Does nothing if no store is configured.
|
|
499
|
+
*/ backupFailedTx(failedTx) {
|
|
500
|
+
if (!this.failedTxStore) {
|
|
501
|
+
return;
|
|
502
|
+
}
|
|
503
|
+
const tx = {
|
|
504
|
+
...failedTx,
|
|
505
|
+
timestamp: Date.now()
|
|
506
|
+
};
|
|
507
|
+
// Fire and forget - don't block on backup
|
|
508
|
+
void this.failedTxStore.then((store)=>store?.saveFailedTx(tx)).catch((err)=>{
|
|
509
|
+
this.log.warn(`Failed to backup failed L1 tx to store`, err);
|
|
510
|
+
});
|
|
83
511
|
}
|
|
84
512
|
getRollupContract() {
|
|
85
513
|
return this.rollupContract;
|
|
86
514
|
}
|
|
515
|
+
/**
|
|
516
|
+
* Gets the fee asset price modifier from the oracle.
|
|
517
|
+
* Returns 0n if the oracle query fails.
|
|
518
|
+
*/ getFeeAssetPriceModifier() {
|
|
519
|
+
return this.feeAssetPriceOracle.computePriceModifier();
|
|
520
|
+
}
|
|
87
521
|
getSenderAddress() {
|
|
88
522
|
return this.l1TxUtils.getSenderAddress();
|
|
89
523
|
}
|
|
90
524
|
/**
|
|
525
|
+
* Gets the L1 fee analyzer instance (only available in fisherman mode)
|
|
526
|
+
*/ getL1FeeAnalyzer() {
|
|
527
|
+
return this.l1FeeAnalyzer;
|
|
528
|
+
}
|
|
529
|
+
/**
|
|
91
530
|
* Sets the proposer address to use for simulations in fisherman mode.
|
|
92
531
|
* @param proposerAddress - The actual proposer's address to use for balance lookups in simulations
|
|
93
532
|
*/ setProposerAddressForSimulation(proposerAddress) {
|
|
@@ -109,6 +548,46 @@ export class SequencerPublisher {
|
|
|
109
548
|
}
|
|
110
549
|
}
|
|
111
550
|
/**
|
|
551
|
+
* Analyzes L1 fees for the pending requests without sending them.
|
|
552
|
+
* This is used in fisherman mode to validate fee calculations.
|
|
553
|
+
* @param l2SlotNumber - The L2 slot number for this analysis
|
|
554
|
+
* @param onComplete - Optional callback to invoke when analysis completes (after block is mined)
|
|
555
|
+
* @returns The analysis result (incomplete until block mines), or undefined if no requests
|
|
556
|
+
*/ async analyzeL1Fees(l2SlotNumber, onComplete) {
|
|
557
|
+
if (!this.l1FeeAnalyzer) {
|
|
558
|
+
this.log.warn('L1 fee analyzer not available (not in fisherman mode)');
|
|
559
|
+
return undefined;
|
|
560
|
+
}
|
|
561
|
+
const requestsToAnalyze = [
|
|
562
|
+
...this.requests
|
|
563
|
+
];
|
|
564
|
+
if (requestsToAnalyze.length === 0) {
|
|
565
|
+
this.log.debug('No requests to analyze for L1 fees');
|
|
566
|
+
return undefined;
|
|
567
|
+
}
|
|
568
|
+
// Extract blob config from requests (if any)
|
|
569
|
+
const blobConfigs = requestsToAnalyze.filter((request)=>request.blobConfig).map((request)=>request.blobConfig);
|
|
570
|
+
const blobConfig = blobConfigs[0];
|
|
571
|
+
// Get gas configs
|
|
572
|
+
const gasConfigs = requestsToAnalyze.filter((request)=>request.gasConfig).map((request)=>request.gasConfig);
|
|
573
|
+
const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
|
|
574
|
+
const gasLimit = gasLimits.length > 0 ? gasLimits.reduce((sum, g)=>sum + g, 0n) : 0n;
|
|
575
|
+
// Get the transaction requests
|
|
576
|
+
const l1Requests = requestsToAnalyze.map((r)=>r.request);
|
|
577
|
+
// Start the analysis
|
|
578
|
+
const analysisId = await this.l1FeeAnalyzer.startAnalysis(l2SlotNumber, gasLimit > 0n ? gasLimit : MAX_L1_TX_LIMIT, l1Requests, blobConfig, onComplete);
|
|
579
|
+
this.log.info('Started L1 fee analysis', {
|
|
580
|
+
analysisId,
|
|
581
|
+
l2SlotNumber: l2SlotNumber.toString(),
|
|
582
|
+
requestCount: requestsToAnalyze.length,
|
|
583
|
+
hasBlobConfig: !!blobConfig,
|
|
584
|
+
gasLimit: gasLimit.toString(),
|
|
585
|
+
actions: requestsToAnalyze.map((r)=>r.action)
|
|
586
|
+
});
|
|
587
|
+
// Return the analysis result (will be incomplete until block mines)
|
|
588
|
+
return this.l1FeeAnalyzer.getAnalysis(analysisId);
|
|
589
|
+
}
|
|
590
|
+
/**
|
|
112
591
|
* Sends all requests that are still valid.
|
|
113
592
|
* @returns one of:
|
|
114
593
|
* - A receipt and stats if the tx succeeded
|
|
@@ -119,7 +598,7 @@ export class SequencerPublisher {
|
|
|
119
598
|
...this.requests
|
|
120
599
|
];
|
|
121
600
|
this.requests = [];
|
|
122
|
-
if (this.interrupted) {
|
|
601
|
+
if (this.interrupted || requestsToProcess.length === 0) {
|
|
123
602
|
return undefined;
|
|
124
603
|
}
|
|
125
604
|
const currentL2Slot = this.getCurrentL2Slot();
|
|
@@ -154,7 +633,16 @@ export class SequencerPublisher {
|
|
|
154
633
|
const blobConfig = blobConfigs[0];
|
|
155
634
|
// Merge gasConfigs. Yields the sum of gasLimits, and the earliest txTimeoutAt, or undefined if no gasConfig sets them.
|
|
156
635
|
const gasLimits = gasConfigs.map((g)=>g?.gasLimit).filter((g)=>g !== undefined);
|
|
157
|
-
|
|
636
|
+
let gasLimit = gasLimits.length > 0 ? sumBigint(gasLimits) : undefined; // sum
|
|
637
|
+
// Cap at L1 block gas limit so the node accepts the tx ("gas limit too high" otherwise).
|
|
638
|
+
const maxGas = MAX_L1_TX_LIMIT;
|
|
639
|
+
if (gasLimit !== undefined && gasLimit > maxGas) {
|
|
640
|
+
this.log.debug('Capping bundled tx gas limit to L1 max', {
|
|
641
|
+
requested: gasLimit,
|
|
642
|
+
capped: maxGas
|
|
643
|
+
});
|
|
644
|
+
gasLimit = maxGas;
|
|
645
|
+
}
|
|
158
646
|
const txTimeoutAts = gasConfigs.map((g)=>g?.txTimeoutAt).filter((g)=>g !== undefined);
|
|
159
647
|
const txTimeoutAt = txTimeoutAts.length > 0 ? new Date(Math.min(...txTimeoutAts.map((g)=>g.getTime()))) : undefined; // earliest
|
|
160
648
|
const txConfig = {
|
|
@@ -165,12 +653,34 @@ export class SequencerPublisher {
|
|
|
165
653
|
// This ensures the committee gets precomputed correctly
|
|
166
654
|
validRequests.sort((a, b)=>compareActions(a.action, b.action));
|
|
167
655
|
try {
|
|
656
|
+
// Capture context for failed tx backup before sending
|
|
657
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
658
|
+
const multicallData = encodeFunctionData({
|
|
659
|
+
abi: multicall3Abi,
|
|
660
|
+
functionName: 'aggregate3',
|
|
661
|
+
args: [
|
|
662
|
+
validRequests.map((r)=>({
|
|
663
|
+
target: r.request.to,
|
|
664
|
+
callData: r.request.data,
|
|
665
|
+
allowFailure: true
|
|
666
|
+
}))
|
|
667
|
+
]
|
|
668
|
+
});
|
|
669
|
+
const blobDataHex = blobConfig?.blobs?.map((b)=>toHex(b));
|
|
670
|
+
const txContext = {
|
|
671
|
+
multicallData,
|
|
672
|
+
blobData: blobDataHex,
|
|
673
|
+
l1BlockNumber
|
|
674
|
+
};
|
|
168
675
|
this.log.debug('Forwarding transactions', {
|
|
169
676
|
validRequests: validRequests.map((request)=>request.action),
|
|
170
677
|
txConfig
|
|
171
678
|
});
|
|
172
|
-
const result = await
|
|
173
|
-
|
|
679
|
+
const result = await this.forwardWithPublisherRotation(validRequests, txConfig, blobConfig);
|
|
680
|
+
if (result === undefined) {
|
|
681
|
+
return undefined;
|
|
682
|
+
}
|
|
683
|
+
const { successfulActions = [], failedActions = [] } = this.callbackBundledTransactions(validRequests, result, txContext);
|
|
174
684
|
return {
|
|
175
685
|
result,
|
|
176
686
|
expiredActions,
|
|
@@ -190,10 +700,67 @@ export class SequencerPublisher {
|
|
|
190
700
|
}
|
|
191
701
|
}
|
|
192
702
|
}
|
|
193
|
-
|
|
703
|
+
/**
|
|
704
|
+
* Forwards transactions via Multicall3, rotating to the next available publisher if a send
|
|
705
|
+
* failure occurs (i.e. the tx never reached the chain).
|
|
706
|
+
* On-chain reverts and simulation errors are returned as-is without rotation.
|
|
707
|
+
*/ async forwardWithPublisherRotation(validRequests, txConfig, blobConfig) {
|
|
708
|
+
const triedAddresses = [];
|
|
709
|
+
let currentPublisher = this.l1TxUtils;
|
|
710
|
+
while(true){
|
|
711
|
+
triedAddresses.push(currentPublisher.getSenderAddress());
|
|
712
|
+
try {
|
|
713
|
+
const result = await Multicall3.forward(validRequests.map((r)=>r.request), currentPublisher, txConfig, blobConfig, this.rollupContract.address, this.log);
|
|
714
|
+
this.l1TxUtils = currentPublisher;
|
|
715
|
+
return result;
|
|
716
|
+
} catch (err) {
|
|
717
|
+
if (err instanceof TimeoutError) {
|
|
718
|
+
throw err;
|
|
719
|
+
}
|
|
720
|
+
const viemError = formatViemError(err);
|
|
721
|
+
if (!this.getNextPublisher) {
|
|
722
|
+
this.log.error('Failed to publish bundled transactions', viemError);
|
|
723
|
+
return undefined;
|
|
724
|
+
}
|
|
725
|
+
this.log.warn(`Publisher ${currentPublisher.getSenderAddress()} failed to send, rotating to next publisher`, viemError);
|
|
726
|
+
const nextPublisher = await this.getNextPublisher([
|
|
727
|
+
...triedAddresses
|
|
728
|
+
]);
|
|
729
|
+
if (!nextPublisher) {
|
|
730
|
+
this.log.error('All available publishers exhausted, failed to publish bundled transactions');
|
|
731
|
+
return undefined;
|
|
732
|
+
}
|
|
733
|
+
currentPublisher = nextPublisher;
|
|
734
|
+
}
|
|
735
|
+
}
|
|
736
|
+
}
|
|
737
|
+
callbackBundledTransactions(requests, result, txContext) {
|
|
194
738
|
const actionsListStr = requests.map((r)=>r.action).join(', ');
|
|
195
739
|
if (result instanceof FormattedViemError) {
|
|
196
740
|
this.log.error(`Failed to publish bundled transactions (${actionsListStr})`, result);
|
|
741
|
+
this.backupFailedTx({
|
|
742
|
+
id: keccak256(txContext.multicallData),
|
|
743
|
+
failureType: 'send-error',
|
|
744
|
+
request: {
|
|
745
|
+
to: MULTI_CALL_3_ADDRESS,
|
|
746
|
+
data: txContext.multicallData
|
|
747
|
+
},
|
|
748
|
+
blobData: txContext.blobData,
|
|
749
|
+
l1BlockNumber: txContext.l1BlockNumber.toString(),
|
|
750
|
+
error: {
|
|
751
|
+
message: result.message,
|
|
752
|
+
name: result.name
|
|
753
|
+
},
|
|
754
|
+
context: {
|
|
755
|
+
actions: requests.map((r)=>r.action),
|
|
756
|
+
requests: requests.map((r)=>({
|
|
757
|
+
action: r.action,
|
|
758
|
+
to: r.request.to,
|
|
759
|
+
data: r.request.data
|
|
760
|
+
})),
|
|
761
|
+
sender: this.getSenderAddress().toString()
|
|
762
|
+
}
|
|
763
|
+
});
|
|
197
764
|
return {
|
|
198
765
|
failedActions: requests.map((r)=>r.action)
|
|
199
766
|
};
|
|
@@ -211,6 +778,37 @@ export class SequencerPublisher {
|
|
|
211
778
|
failedActions.push(request.action);
|
|
212
779
|
}
|
|
213
780
|
}
|
|
781
|
+
// Single backup for the whole reverted tx
|
|
782
|
+
if (failedActions.length > 0 && result?.receipt?.status === 'reverted') {
|
|
783
|
+
this.backupFailedTx({
|
|
784
|
+
id: result.receipt.transactionHash,
|
|
785
|
+
failureType: 'revert',
|
|
786
|
+
request: {
|
|
787
|
+
to: MULTI_CALL_3_ADDRESS,
|
|
788
|
+
data: txContext.multicallData
|
|
789
|
+
},
|
|
790
|
+
blobData: txContext.blobData,
|
|
791
|
+
l1BlockNumber: result.receipt.blockNumber.toString(),
|
|
792
|
+
receipt: {
|
|
793
|
+
transactionHash: result.receipt.transactionHash,
|
|
794
|
+
blockNumber: result.receipt.blockNumber.toString(),
|
|
795
|
+
gasUsed: (result.receipt.gasUsed ?? 0n).toString(),
|
|
796
|
+
status: 'reverted'
|
|
797
|
+
},
|
|
798
|
+
error: {
|
|
799
|
+
message: result.errorMsg ?? 'Transaction reverted'
|
|
800
|
+
},
|
|
801
|
+
context: {
|
|
802
|
+
actions: failedActions,
|
|
803
|
+
requests: requests.filter((r)=>failedActions.includes(r.action)).map((r)=>({
|
|
804
|
+
action: r.action,
|
|
805
|
+
to: r.request.to,
|
|
806
|
+
data: r.request.data
|
|
807
|
+
})),
|
|
808
|
+
sender: this.getSenderAddress().toString()
|
|
809
|
+
}
|
|
810
|
+
});
|
|
811
|
+
}
|
|
214
812
|
return {
|
|
215
813
|
successfulActions,
|
|
216
814
|
failedActions
|
|
@@ -229,7 +827,7 @@ export class SequencerPublisher {
|
|
|
229
827
|
'InvalidArchive'
|
|
230
828
|
];
|
|
231
829
|
return this.rollupContract.canProposeAtNextEthBlock(tipArchive.toBuffer(), msgSender.toString(), Number(this.ethereumSlotDuration), {
|
|
232
|
-
forcePendingCheckpointNumber: opts.
|
|
830
|
+
forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
|
|
233
831
|
}).catch((err)=>{
|
|
234
832
|
if (err instanceof FormattedViemError && ignoredErrors.find((e)=>err.message.includes(e))) {
|
|
235
833
|
this.log.warn(`Failed canProposeAtTime check with ${ignoredErrors.find((e)=>err.message.includes(e))}`, {
|
|
@@ -257,11 +855,11 @@ export class SequencerPublisher {
|
|
|
257
855
|
[],
|
|
258
856
|
Signature.empty().toViemSignature(),
|
|
259
857
|
`0x${'0'.repeat(64)}`,
|
|
260
|
-
header.
|
|
858
|
+
header.blobsHash.toString(),
|
|
261
859
|
flags
|
|
262
860
|
];
|
|
263
861
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
264
|
-
const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.
|
|
862
|
+
const stateOverrides = await this.rollupContract.makePendingCheckpointNumberOverride(opts?.forcePendingCheckpointNumber);
|
|
265
863
|
let balance = 0n;
|
|
266
864
|
if (this.config.fishermanMode) {
|
|
267
865
|
// In fisherman mode, we can't know where the proposer is publishing from
|
|
@@ -288,34 +886,38 @@ export class SequencerPublisher {
|
|
|
288
886
|
this.log.debug(`Simulated validateHeader`);
|
|
289
887
|
}
|
|
290
888
|
/**
|
|
291
|
-
* Simulate making a call to invalidate a
|
|
292
|
-
* @param
|
|
293
|
-
*/ async
|
|
889
|
+
* Simulate making a call to invalidate a checkpoint with invalid attestations. Returns undefined if no need to invalidate.
|
|
890
|
+
* @param validationResult - The validation result indicating which checkpoint to invalidate (as returned by the archiver)
|
|
891
|
+
*/ async simulateInvalidateCheckpoint(validationResult) {
|
|
294
892
|
if (validationResult.valid) {
|
|
295
893
|
return undefined;
|
|
296
894
|
}
|
|
297
|
-
const { reason,
|
|
298
|
-
const
|
|
895
|
+
const { reason, checkpoint } = validationResult;
|
|
896
|
+
const checkpointNumber = checkpoint.checkpointNumber;
|
|
299
897
|
const logData = {
|
|
300
|
-
...
|
|
898
|
+
...checkpoint,
|
|
301
899
|
reason
|
|
302
900
|
};
|
|
303
|
-
const
|
|
304
|
-
if (
|
|
305
|
-
this.log.verbose(`Skipping
|
|
306
|
-
|
|
901
|
+
const currentCheckpointNumber = await this.rollupContract.getCheckpointNumber();
|
|
902
|
+
if (currentCheckpointNumber < checkpointNumber) {
|
|
903
|
+
this.log.verbose(`Skipping checkpoint ${checkpointNumber} invalidation since it has already been removed from the pending chain`, {
|
|
904
|
+
currentCheckpointNumber,
|
|
307
905
|
...logData
|
|
308
906
|
});
|
|
309
907
|
return undefined;
|
|
310
908
|
}
|
|
311
|
-
const request = this.
|
|
312
|
-
this.log.debug(`Simulating invalidate
|
|
909
|
+
const request = this.buildInvalidateCheckpointRequest(validationResult);
|
|
910
|
+
this.log.debug(`Simulating invalidate checkpoint ${checkpointNumber}`, {
|
|
313
911
|
...logData,
|
|
314
912
|
request
|
|
315
913
|
});
|
|
914
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
316
915
|
try {
|
|
317
|
-
const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined,
|
|
318
|
-
|
|
916
|
+
const { gasUsed } = await this.l1TxUtils.simulate(request, undefined, undefined, mergeAbis([
|
|
917
|
+
request.abi ?? [],
|
|
918
|
+
ErrorsAbi
|
|
919
|
+
]));
|
|
920
|
+
this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} succeeded`, {
|
|
319
921
|
...logData,
|
|
320
922
|
request,
|
|
321
923
|
gasUsed
|
|
@@ -323,90 +925,92 @@ export class SequencerPublisher {
|
|
|
323
925
|
return {
|
|
324
926
|
request,
|
|
325
927
|
gasUsed,
|
|
326
|
-
|
|
327
|
-
|
|
928
|
+
checkpointNumber,
|
|
929
|
+
forcePendingCheckpointNumber: CheckpointNumber(checkpointNumber - 1),
|
|
328
930
|
reason
|
|
329
931
|
};
|
|
330
932
|
} catch (err) {
|
|
331
933
|
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
|
|
934
|
+
// If the error is due to the checkpoint not being in the pending chain, and it was indeed removed by someone else,
|
|
935
|
+
// we can safely ignore it and return undefined so we go ahead with checkpoint building.
|
|
936
|
+
if (viemError.message?.includes('Rollup__CheckpointNotInPendingChain')) {
|
|
937
|
+
this.log.verbose(`Simulation for invalidate checkpoint ${checkpointNumber} failed due to checkpoint not being in pending chain`, {
|
|
336
938
|
...logData,
|
|
337
939
|
request,
|
|
338
940
|
error: viemError.message
|
|
339
941
|
});
|
|
340
|
-
const
|
|
341
|
-
if (
|
|
342
|
-
this.log.verbose(`
|
|
942
|
+
const latestPendingCheckpointNumber = await this.rollupContract.getCheckpointNumber();
|
|
943
|
+
if (latestPendingCheckpointNumber < checkpointNumber) {
|
|
944
|
+
this.log.verbose(`Checkpoint ${checkpointNumber} has already been invalidated`, {
|
|
343
945
|
...logData
|
|
344
946
|
});
|
|
345
947
|
return undefined;
|
|
346
948
|
} else {
|
|
347
|
-
this.log.error(`Simulation for invalidate ${
|
|
348
|
-
throw new Error(`Failed to simulate invalidate
|
|
949
|
+
this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed and it is still in pending chain`, viemError, logData);
|
|
950
|
+
throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber} while it is still in pending chain`, {
|
|
349
951
|
cause: viemError
|
|
350
952
|
});
|
|
351
953
|
}
|
|
352
954
|
}
|
|
353
|
-
// Otherwise, throw. We cannot build the next
|
|
354
|
-
this.log.error(`Simulation for invalidate
|
|
355
|
-
|
|
955
|
+
// Otherwise, throw. We cannot build the next checkpoint if we cannot invalidate the previous one.
|
|
956
|
+
this.log.error(`Simulation for invalidate checkpoint ${checkpointNumber} failed`, viemError, logData);
|
|
957
|
+
this.backupFailedTx({
|
|
958
|
+
id: keccak256(request.data),
|
|
959
|
+
failureType: 'simulation',
|
|
960
|
+
request: {
|
|
961
|
+
to: request.to,
|
|
962
|
+
data: request.data,
|
|
963
|
+
value: request.value?.toString()
|
|
964
|
+
},
|
|
965
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
966
|
+
error: {
|
|
967
|
+
message: viemError.message,
|
|
968
|
+
name: viemError.name
|
|
969
|
+
},
|
|
970
|
+
context: {
|
|
971
|
+
actions: [
|
|
972
|
+
`invalidate-${reason}`
|
|
973
|
+
],
|
|
974
|
+
checkpointNumber,
|
|
975
|
+
sender: this.getSenderAddress().toString()
|
|
976
|
+
}
|
|
977
|
+
});
|
|
978
|
+
throw new Error(`Failed to simulate invalidate checkpoint ${checkpointNumber}`, {
|
|
356
979
|
cause: viemError
|
|
357
980
|
});
|
|
358
981
|
}
|
|
359
982
|
}
|
|
360
|
-
|
|
983
|
+
buildInvalidateCheckpointRequest(validationResult) {
|
|
361
984
|
if (validationResult.valid) {
|
|
362
|
-
throw new Error('Cannot invalidate a valid
|
|
985
|
+
throw new Error('Cannot invalidate a valid checkpoint');
|
|
363
986
|
}
|
|
364
|
-
const {
|
|
987
|
+
const { checkpoint, committee, reason } = validationResult;
|
|
365
988
|
const logData = {
|
|
366
|
-
...
|
|
989
|
+
...checkpoint,
|
|
367
990
|
reason
|
|
368
991
|
};
|
|
369
|
-
this.log.debug(`
|
|
992
|
+
this.log.debug(`Building invalidate checkpoint ${checkpoint.checkpointNumber} request`, logData);
|
|
370
993
|
const attestationsAndSigners = new CommitteeAttestationsAndSigners(validationResult.attestations).getPackedAttestations();
|
|
371
994
|
if (reason === 'invalid-attestation') {
|
|
372
|
-
return this.rollupContract.buildInvalidateBadAttestationRequest(
|
|
995
|
+
return this.rollupContract.buildInvalidateBadAttestationRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee, validationResult.invalidIndex);
|
|
373
996
|
} else if (reason === 'insufficient-attestations') {
|
|
374
|
-
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(
|
|
997
|
+
return this.rollupContract.buildInvalidateInsufficientAttestationsRequest(checkpoint.checkpointNumber, attestationsAndSigners, committee);
|
|
375
998
|
} else {
|
|
376
999
|
const _ = reason;
|
|
377
1000
|
throw new Error(`Unknown reason for invalidation`);
|
|
378
1001
|
}
|
|
379
1002
|
}
|
|
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) {
|
|
1003
|
+
/** Simulates `propose` to make sure that the checkpoint is valid for submission */ async validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, options) {
|
|
389
1004
|
const ts = BigInt((await this.l1TxUtils.getBlock()).timestamp + this.ethereumSlotDuration);
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
const ignoreSignatures = attestationsAndSigners.attestations.length === 0;
|
|
393
|
-
if (ignoreSignatures) {
|
|
394
|
-
const { committee } = await this.epochCache.getCommittee(block.header.globalVariables.slotNumber);
|
|
395
|
-
if (!committee) {
|
|
396
|
-
this.log.warn(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
|
|
397
|
-
throw new Error(`No committee found for slot ${block.header.globalVariables.slotNumber}`);
|
|
398
|
-
}
|
|
399
|
-
attestationsAndSigners.attestations = committee.map((committeeMember)=>CommitteeAttestation.fromAddress(committeeMember));
|
|
400
|
-
}
|
|
401
|
-
const blobFields = block.getCheckpointBlobFields();
|
|
402
|
-
const blobs = getBlobsPerL1Block(blobFields);
|
|
1005
|
+
const blobFields = checkpoint.toBlobFields();
|
|
1006
|
+
const blobs = await getBlobsPerL1Block(blobFields);
|
|
403
1007
|
const blobInput = getPrefixedEthBlobCommitments(blobs);
|
|
404
1008
|
const args = [
|
|
405
1009
|
{
|
|
406
|
-
header:
|
|
407
|
-
archive: toHex(
|
|
1010
|
+
header: checkpoint.header.toViem(),
|
|
1011
|
+
archive: toHex(checkpoint.archive.root.toBuffer()),
|
|
408
1012
|
oracleInput: {
|
|
409
|
-
feeAssetPriceModifier:
|
|
1013
|
+
feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
|
|
410
1014
|
}
|
|
411
1015
|
},
|
|
412
1016
|
attestationsAndSigners.getPackedAttestations(),
|
|
@@ -431,9 +1035,38 @@ export class SequencerPublisher {
|
|
|
431
1035
|
}
|
|
432
1036
|
const round = await base.computeRound(slotNumber);
|
|
433
1037
|
const roundInfo = await base.getRoundInfo(this.rollupContract.address, round);
|
|
1038
|
+
if (roundInfo.quorumReached) {
|
|
1039
|
+
return false;
|
|
1040
|
+
}
|
|
434
1041
|
if (roundInfo.lastSignalSlot >= slotNumber) {
|
|
435
1042
|
return false;
|
|
436
1043
|
}
|
|
1044
|
+
if (await this.isPayloadEmpty(payload)) {
|
|
1045
|
+
this.log.warn(`Skipping vote cast for payload with empty code`);
|
|
1046
|
+
return false;
|
|
1047
|
+
}
|
|
1048
|
+
// Check if payload was already submitted to governance
|
|
1049
|
+
const cacheKey = payload.toString();
|
|
1050
|
+
if (!this.payloadProposedCache.has(cacheKey)) {
|
|
1051
|
+
try {
|
|
1052
|
+
const l1StartBlock = await this.rollupContract.getL1StartBlock();
|
|
1053
|
+
const proposed = await retry(()=>base.hasPayloadBeenProposed(payload.toString(), l1StartBlock), 'Check if payload was proposed', makeBackoff([
|
|
1054
|
+
0,
|
|
1055
|
+
1,
|
|
1056
|
+
2
|
|
1057
|
+
]), this.log, true);
|
|
1058
|
+
if (proposed) {
|
|
1059
|
+
this.payloadProposedCache.add(cacheKey);
|
|
1060
|
+
}
|
|
1061
|
+
} catch (err) {
|
|
1062
|
+
this.log.warn(`Failed to check if payload ${payload} was proposed after retries, skipping signal`, err);
|
|
1063
|
+
return false;
|
|
1064
|
+
}
|
|
1065
|
+
}
|
|
1066
|
+
if (this.payloadProposedCache.has(cacheKey)) {
|
|
1067
|
+
this.log.info(`Payload ${payload} was already proposed to governance, stopping signals`);
|
|
1068
|
+
return false;
|
|
1069
|
+
}
|
|
437
1070
|
const cachedLastVote = this.lastActions[signalType];
|
|
438
1071
|
this.lastActions[signalType] = slotNumber;
|
|
439
1072
|
const action = signalType;
|
|
@@ -444,15 +1077,41 @@ export class SequencerPublisher {
|
|
|
444
1077
|
signer: this.l1TxUtils.client.account?.address,
|
|
445
1078
|
lastValidL2Slot: slotNumber
|
|
446
1079
|
});
|
|
1080
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
447
1081
|
try {
|
|
448
1082
|
await this.l1TxUtils.simulate(request, {
|
|
449
1083
|
time: timestamp
|
|
450
|
-
}, [],
|
|
1084
|
+
}, [], mergeAbis([
|
|
1085
|
+
request.abi ?? [],
|
|
1086
|
+
ErrorsAbi
|
|
1087
|
+
]));
|
|
451
1088
|
this.log.debug(`Simulation for ${action} at slot ${slotNumber} succeeded`, {
|
|
452
1089
|
request
|
|
453
1090
|
});
|
|
454
1091
|
} catch (err) {
|
|
455
|
-
|
|
1092
|
+
const viemError = formatViemError(err);
|
|
1093
|
+
this.log.error(`Failed simulation for ${action} at slot ${slotNumber} (enqueuing the action anyway)`, viemError);
|
|
1094
|
+
this.backupFailedTx({
|
|
1095
|
+
id: keccak256(request.data),
|
|
1096
|
+
failureType: 'simulation',
|
|
1097
|
+
request: {
|
|
1098
|
+
to: request.to,
|
|
1099
|
+
data: request.data,
|
|
1100
|
+
value: request.value?.toString()
|
|
1101
|
+
},
|
|
1102
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
1103
|
+
error: {
|
|
1104
|
+
message: viemError.message,
|
|
1105
|
+
name: viemError.name
|
|
1106
|
+
},
|
|
1107
|
+
context: {
|
|
1108
|
+
actions: [
|
|
1109
|
+
action
|
|
1110
|
+
],
|
|
1111
|
+
slot: slotNumber,
|
|
1112
|
+
sender: this.getSenderAddress().toString()
|
|
1113
|
+
}
|
|
1114
|
+
});
|
|
456
1115
|
// Yes, we enqueue the request anyway, in case there was a bug with the simulation itself
|
|
457
1116
|
}
|
|
458
1117
|
// TODO(palla/slash): All votes (governance and slashing) should txTimeoutAt at the end of the slot.
|
|
@@ -472,17 +1131,27 @@ export class SequencerPublisher {
|
|
|
472
1131
|
payload: payload.toString()
|
|
473
1132
|
};
|
|
474
1133
|
if (!success) {
|
|
475
|
-
this.log.error(`Signaling in
|
|
1134
|
+
this.log.error(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} failed`, logData);
|
|
476
1135
|
this.lastActions[signalType] = cachedLastVote;
|
|
477
1136
|
return false;
|
|
478
1137
|
} else {
|
|
479
|
-
this.log.info(`Signaling in
|
|
1138
|
+
this.log.info(`Signaling in ${action} for ${payload} at slot ${slotNumber} in round ${round} succeeded`, logData);
|
|
480
1139
|
return true;
|
|
481
1140
|
}
|
|
482
1141
|
}
|
|
483
1142
|
});
|
|
484
1143
|
return true;
|
|
485
1144
|
}
|
|
1145
|
+
async isPayloadEmpty(payload) {
|
|
1146
|
+
const key = payload.toString();
|
|
1147
|
+
const cached = this.isPayloadEmptyCache.get(key);
|
|
1148
|
+
if (cached) {
|
|
1149
|
+
return cached;
|
|
1150
|
+
}
|
|
1151
|
+
const isEmpty = !await this.l1TxUtils.getCode(payload);
|
|
1152
|
+
this.isPayloadEmptyCache.set(key, isEmpty);
|
|
1153
|
+
return isEmpty;
|
|
1154
|
+
}
|
|
486
1155
|
/**
|
|
487
1156
|
* Enqueues a governance castSignal transaction to cast a signal for a given slot number.
|
|
488
1157
|
* @param slotNumber - The slot number to cast a signal for.
|
|
@@ -578,22 +1247,17 @@ export class SequencerPublisher {
|
|
|
578
1247
|
}
|
|
579
1248
|
return true;
|
|
580
1249
|
}
|
|
581
|
-
/**
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
* @returns True if the tx has been enqueued, throws otherwise. See #9315
|
|
586
|
-
*/ async enqueueProposeL2Block(block, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
|
|
587
|
-
const checkpointHeader = block.getCheckpointHeader();
|
|
588
|
-
const blobFields = block.getCheckpointBlobFields();
|
|
589
|
-
const blobs = getBlobsPerL1Block(blobFields);
|
|
1250
|
+
/** Simulates and enqueues a proposal for a checkpoint on L1 */ async enqueueProposeCheckpoint(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts = {}) {
|
|
1251
|
+
const checkpointHeader = checkpoint.header;
|
|
1252
|
+
const blobFields = checkpoint.toBlobFields();
|
|
1253
|
+
const blobs = await getBlobsPerL1Block(blobFields);
|
|
590
1254
|
const proposeTxArgs = {
|
|
591
1255
|
header: checkpointHeader,
|
|
592
|
-
archive:
|
|
593
|
-
body: block.body.toBuffer(),
|
|
1256
|
+
archive: checkpoint.archive.root.toBuffer(),
|
|
594
1257
|
blobs,
|
|
595
1258
|
attestationsAndSigners,
|
|
596
|
-
attestationsAndSignersSignature
|
|
1259
|
+
attestationsAndSignersSignature,
|
|
1260
|
+
feeAssetPriceModifier: checkpoint.feeAssetPriceModifier
|
|
597
1261
|
};
|
|
598
1262
|
let ts;
|
|
599
1263
|
try {
|
|
@@ -602,36 +1266,35 @@ export class SequencerPublisher {
|
|
|
602
1266
|
// By simulation issue, I mean the fact that the block.timestamp is equal to the last block, not the next, which
|
|
603
1267
|
// make time consistency checks break.
|
|
604
1268
|
// 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.
|
|
1269
|
+
ts = await this.validateCheckpointForSubmission(checkpoint, attestationsAndSigners, attestationsAndSignersSignature, opts);
|
|
606
1270
|
} catch (err) {
|
|
607
|
-
this.log.error(`
|
|
608
|
-
...
|
|
609
|
-
slotNumber:
|
|
610
|
-
|
|
1271
|
+
this.log.error(`Checkpoint validation failed. ${err instanceof Error ? err.message : 'No error message'}`, err, {
|
|
1272
|
+
...checkpoint.getStats(),
|
|
1273
|
+
slotNumber: checkpoint.header.slotNumber,
|
|
1274
|
+
forcePendingCheckpointNumber: opts.forcePendingCheckpointNumber
|
|
611
1275
|
});
|
|
612
1276
|
throw err;
|
|
613
1277
|
}
|
|
614
|
-
this.log.verbose(`Enqueuing
|
|
615
|
-
...
|
|
1278
|
+
this.log.verbose(`Enqueuing checkpoint propose transaction`, {
|
|
1279
|
+
...checkpoint.toCheckpointInfo(),
|
|
616
1280
|
...opts
|
|
617
1281
|
});
|
|
618
|
-
await this.addProposeTx(
|
|
619
|
-
return true;
|
|
1282
|
+
await this.addProposeTx(checkpoint, proposeTxArgs, opts, ts);
|
|
620
1283
|
}
|
|
621
|
-
|
|
1284
|
+
enqueueInvalidateCheckpoint(request, opts = {}) {
|
|
622
1285
|
if (!request) {
|
|
623
1286
|
return;
|
|
624
1287
|
}
|
|
625
1288
|
// We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
|
|
626
1289
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(request.gasUsed) * 64 / 63)));
|
|
627
|
-
const { gasUsed,
|
|
1290
|
+
const { gasUsed, checkpointNumber } = request;
|
|
628
1291
|
const logData = {
|
|
629
1292
|
gasUsed,
|
|
630
|
-
|
|
1293
|
+
checkpointNumber,
|
|
631
1294
|
gasLimit,
|
|
632
1295
|
opts
|
|
633
1296
|
};
|
|
634
|
-
this.log.verbose(`Enqueuing invalidate
|
|
1297
|
+
this.log.verbose(`Enqueuing invalidate checkpoint request`, logData);
|
|
635
1298
|
this.addRequest({
|
|
636
1299
|
action: `invalidate-by-${request.reason}`,
|
|
637
1300
|
request: request.request,
|
|
@@ -643,12 +1306,12 @@ export class SequencerPublisher {
|
|
|
643
1306
|
checkSuccess: (_req, result)=>{
|
|
644
1307
|
const success = result && result.receipt && result.receipt.status === 'success' && tryExtractEvent(result.receipt.logs, this.rollupContract.address, RollupAbi, 'CheckpointInvalidated');
|
|
645
1308
|
if (!success) {
|
|
646
|
-
this.log.warn(`Invalidate
|
|
1309
|
+
this.log.warn(`Invalidate checkpoint ${request.checkpointNumber} failed`, {
|
|
647
1310
|
...result,
|
|
648
1311
|
...logData
|
|
649
1312
|
});
|
|
650
1313
|
} else {
|
|
651
|
-
this.log.info(`Invalidate
|
|
1314
|
+
this.log.info(`Invalidate checkpoint ${request.checkpointNumber} succeeded`, {
|
|
652
1315
|
...result,
|
|
653
1316
|
...logData
|
|
654
1317
|
});
|
|
@@ -670,28 +1333,60 @@ export class SequencerPublisher {
|
|
|
670
1333
|
const cachedLastActionSlot = this.lastActions[action];
|
|
671
1334
|
this.lastActions[action] = slotNumber;
|
|
672
1335
|
this.log.debug(`Simulating ${action} for slot ${slotNumber}`, logData);
|
|
1336
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
673
1337
|
let gasUsed;
|
|
1338
|
+
const simulateAbi = mergeAbis([
|
|
1339
|
+
request.abi ?? [],
|
|
1340
|
+
ErrorsAbi
|
|
1341
|
+
]);
|
|
674
1342
|
try {
|
|
675
1343
|
({ gasUsed } = await this.l1TxUtils.simulate(request, {
|
|
676
1344
|
time: timestamp
|
|
677
|
-
}, [],
|
|
1345
|
+
}, [], simulateAbi)); // TODO(palla/slash): Check the timestamp logic
|
|
678
1346
|
this.log.verbose(`Simulation for ${action} succeeded`, {
|
|
679
1347
|
...logData,
|
|
680
1348
|
request,
|
|
681
1349
|
gasUsed
|
|
682
1350
|
});
|
|
683
1351
|
} catch (err) {
|
|
684
|
-
const viemError = formatViemError(err);
|
|
1352
|
+
const viemError = formatViemError(err, simulateAbi);
|
|
685
1353
|
this.log.error(`Simulation for ${action} at ${slotNumber} failed`, viemError, logData);
|
|
1354
|
+
this.backupFailedTx({
|
|
1355
|
+
id: keccak256(request.data),
|
|
1356
|
+
failureType: 'simulation',
|
|
1357
|
+
request: {
|
|
1358
|
+
to: request.to,
|
|
1359
|
+
data: request.data,
|
|
1360
|
+
value: request.value?.toString()
|
|
1361
|
+
},
|
|
1362
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
1363
|
+
error: {
|
|
1364
|
+
message: viemError.message,
|
|
1365
|
+
name: viemError.name
|
|
1366
|
+
},
|
|
1367
|
+
context: {
|
|
1368
|
+
actions: [
|
|
1369
|
+
action
|
|
1370
|
+
],
|
|
1371
|
+
slot: slotNumber,
|
|
1372
|
+
sender: this.getSenderAddress().toString()
|
|
1373
|
+
}
|
|
1374
|
+
});
|
|
686
1375
|
return false;
|
|
687
1376
|
}
|
|
688
1377
|
// We issued the simulation against the rollup contract, so we need to account for the overhead of the multicall3
|
|
689
1378
|
const gasLimit = this.l1TxUtils.bumpGasLimit(BigInt(Math.ceil(Number(gasUsed) * 64 / 63)));
|
|
690
1379
|
logData.gasLimit = gasLimit;
|
|
1380
|
+
// Store the ABI used for simulation on the request so Multicall3.forward can decode errors
|
|
1381
|
+
// when the tx is sent and a revert is diagnosed via simulation.
|
|
1382
|
+
const requestWithAbi = {
|
|
1383
|
+
...request,
|
|
1384
|
+
abi: simulateAbi
|
|
1385
|
+
};
|
|
691
1386
|
this.log.debug(`Enqueuing ${action}`, logData);
|
|
692
1387
|
this.addRequest({
|
|
693
1388
|
action,
|
|
694
|
-
request,
|
|
1389
|
+
request: requestWithAbi,
|
|
695
1390
|
gasConfig: {
|
|
696
1391
|
gasLimit
|
|
697
1392
|
},
|
|
@@ -755,10 +1450,38 @@ export class SequencerPublisher {
|
|
|
755
1450
|
}, {}, {
|
|
756
1451
|
blobs: encodedData.blobs.map((b)=>b.data),
|
|
757
1452
|
kzg
|
|
758
|
-
}).catch((err)=>{
|
|
759
|
-
const
|
|
760
|
-
this.log.error(`Failed to validate blobs`, message, {
|
|
761
|
-
metaMessages
|
|
1453
|
+
}).catch(async (err)=>{
|
|
1454
|
+
const viemError = formatViemError(err);
|
|
1455
|
+
this.log.error(`Failed to validate blobs`, viemError.message, {
|
|
1456
|
+
metaMessages: viemError.metaMessages
|
|
1457
|
+
});
|
|
1458
|
+
const validateBlobsData = encodeFunctionData({
|
|
1459
|
+
abi: RollupAbi,
|
|
1460
|
+
functionName: 'validateBlobs',
|
|
1461
|
+
args: [
|
|
1462
|
+
blobInput
|
|
1463
|
+
]
|
|
1464
|
+
});
|
|
1465
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
1466
|
+
this.backupFailedTx({
|
|
1467
|
+
id: keccak256(validateBlobsData),
|
|
1468
|
+
failureType: 'simulation',
|
|
1469
|
+
request: {
|
|
1470
|
+
to: this.rollupContract.address,
|
|
1471
|
+
data: validateBlobsData
|
|
1472
|
+
},
|
|
1473
|
+
blobData: encodedData.blobs.map((b)=>toHex(b.data)),
|
|
1474
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
1475
|
+
error: {
|
|
1476
|
+
message: viemError.message,
|
|
1477
|
+
name: viemError.name
|
|
1478
|
+
},
|
|
1479
|
+
context: {
|
|
1480
|
+
actions: [
|
|
1481
|
+
'validate-blobs'
|
|
1482
|
+
],
|
|
1483
|
+
sender: this.getSenderAddress().toString()
|
|
1484
|
+
}
|
|
762
1485
|
});
|
|
763
1486
|
throw new Error('Failed to validate blobs');
|
|
764
1487
|
});
|
|
@@ -769,8 +1492,7 @@ export class SequencerPublisher {
|
|
|
769
1492
|
header: encodedData.header.toViem(),
|
|
770
1493
|
archive: toHex(encodedData.archive),
|
|
771
1494
|
oracleInput: {
|
|
772
|
-
|
|
773
|
-
feeAssetPriceModifier: 0n
|
|
1495
|
+
feeAssetPriceModifier: encodedData.feeAssetPriceModifier
|
|
774
1496
|
}
|
|
775
1497
|
},
|
|
776
1498
|
encodedData.attestationsAndSigners.getPackedAttestations(),
|
|
@@ -797,8 +1519,8 @@ export class SequencerPublisher {
|
|
|
797
1519
|
functionName: 'propose',
|
|
798
1520
|
args
|
|
799
1521
|
});
|
|
800
|
-
// override the pending
|
|
801
|
-
const
|
|
1522
|
+
// override the pending checkpoint number if requested
|
|
1523
|
+
const forcePendingCheckpointNumberStateDiff = (options.forcePendingCheckpointNumber !== undefined ? await this.rollupContract.makePendingCheckpointNumberOverride(options.forcePendingCheckpointNumber) : []).flatMap((override)=>override.stateDiff ?? []);
|
|
802
1524
|
const stateOverrides = [
|
|
803
1525
|
{
|
|
804
1526
|
address: this.rollupContract.address,
|
|
@@ -808,7 +1530,7 @@ export class SequencerPublisher {
|
|
|
808
1530
|
slot: toPaddedHex(RollupContract.checkBlobStorageSlot, true),
|
|
809
1531
|
value: toPaddedHex(0n, true)
|
|
810
1532
|
},
|
|
811
|
-
...
|
|
1533
|
+
...forcePendingCheckpointNumberStateDiff
|
|
812
1534
|
]
|
|
813
1535
|
}
|
|
814
1536
|
];
|
|
@@ -819,10 +1541,11 @@ export class SequencerPublisher {
|
|
|
819
1541
|
balance: 10n * WEI_CONST * WEI_CONST
|
|
820
1542
|
});
|
|
821
1543
|
}
|
|
1544
|
+
const l1BlockNumber = await this.l1TxUtils.getBlockNumber();
|
|
822
1545
|
const simulationResult = await this.l1TxUtils.simulate({
|
|
823
1546
|
to: this.rollupContract.address,
|
|
824
1547
|
data: rollupData,
|
|
825
|
-
gas:
|
|
1548
|
+
gas: MAX_L1_TX_LIMIT,
|
|
826
1549
|
...this.proposerAddressForSimulation && {
|
|
827
1550
|
from: this.proposerAddressForSimulation.toString()
|
|
828
1551
|
}
|
|
@@ -830,10 +1553,10 @@ export class SequencerPublisher {
|
|
|
830
1553
|
// @note we add 1n to the timestamp because geth implementation doesn't like simulation timestamp to be equal to the current block timestamp
|
|
831
1554
|
time: timestamp + 1n,
|
|
832
1555
|
// @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:
|
|
1556
|
+
gasLimit: MAX_L1_TX_LIMIT * 2n
|
|
834
1557
|
}, stateOverrides, RollupAbi, {
|
|
835
1558
|
// @note fallback gas estimate to use if the node doesn't support simulation API
|
|
836
|
-
fallbackGasEstimate:
|
|
1559
|
+
fallbackGasEstimate: MAX_L1_TX_LIMIT
|
|
837
1560
|
}).catch((err)=>{
|
|
838
1561
|
// In fisherman mode, we expect ValidatorSelection__MissingProposerSignature since fisherman doesn't have proposer signature
|
|
839
1562
|
const viemError = formatViemError(err);
|
|
@@ -841,11 +1564,31 @@ export class SequencerPublisher {
|
|
|
841
1564
|
this.log.debug(`Ignoring expected ValidatorSelection__MissingProposerSignature error in fisherman mode`);
|
|
842
1565
|
// Return a minimal simulation result with the fallback gas estimate
|
|
843
1566
|
return {
|
|
844
|
-
gasUsed:
|
|
1567
|
+
gasUsed: MAX_L1_TX_LIMIT,
|
|
845
1568
|
logs: []
|
|
846
1569
|
};
|
|
847
1570
|
}
|
|
848
1571
|
this.log.error(`Failed to simulate propose tx`, viemError);
|
|
1572
|
+
this.backupFailedTx({
|
|
1573
|
+
id: keccak256(rollupData),
|
|
1574
|
+
failureType: 'simulation',
|
|
1575
|
+
request: {
|
|
1576
|
+
to: this.rollupContract.address,
|
|
1577
|
+
data: rollupData
|
|
1578
|
+
},
|
|
1579
|
+
l1BlockNumber: l1BlockNumber.toString(),
|
|
1580
|
+
error: {
|
|
1581
|
+
message: viemError.message,
|
|
1582
|
+
name: viemError.name
|
|
1583
|
+
},
|
|
1584
|
+
context: {
|
|
1585
|
+
actions: [
|
|
1586
|
+
'propose'
|
|
1587
|
+
],
|
|
1588
|
+
slot: Number(args[0].header.slotNumber),
|
|
1589
|
+
sender: this.getSenderAddress().toString()
|
|
1590
|
+
}
|
|
1591
|
+
});
|
|
849
1592
|
throw err;
|
|
850
1593
|
});
|
|
851
1594
|
return {
|
|
@@ -853,24 +1596,25 @@ export class SequencerPublisher {
|
|
|
853
1596
|
simulationResult
|
|
854
1597
|
};
|
|
855
1598
|
}
|
|
856
|
-
async addProposeTx(
|
|
1599
|
+
async addProposeTx(checkpoint, encodedData, opts = {}, timestamp) {
|
|
1600
|
+
const slot = checkpoint.header.slotNumber;
|
|
857
1601
|
const timer = new Timer();
|
|
858
1602
|
const kzg = Blob.getViemKzgInstance();
|
|
859
1603
|
const { rollupData, simulationResult, blobEvaluationGas } = await this.prepareProposeTx(encodedData, timestamp, opts);
|
|
860
1604
|
const startBlock = await this.l1TxUtils.getBlockNumber();
|
|
861
1605
|
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
|
-
|
|
1606
|
+
// Send the blobs to the blob client preemptively. This helps in tests where the sequencer mistakingly thinks that the propose
|
|
1607
|
+
// tx fails but it does get mined. We make sure that the blobs are sent to the blob client regardless of the tx outcome.
|
|
1608
|
+
void Promise.resolve().then(()=>this.blobClient.sendBlobsToFilestore(encodedData.blobs).catch((_err)=>{
|
|
1609
|
+
this.log.error('Failed to send blobs to blob client');
|
|
1610
|
+
}));
|
|
867
1611
|
return this.addRequest({
|
|
868
1612
|
action: 'propose',
|
|
869
1613
|
request: {
|
|
870
1614
|
to: this.rollupContract.address,
|
|
871
1615
|
data: rollupData
|
|
872
1616
|
},
|
|
873
|
-
lastValidL2Slot:
|
|
1617
|
+
lastValidL2Slot: checkpoint.header.slotNumber,
|
|
874
1618
|
gasConfig: {
|
|
875
1619
|
...opts,
|
|
876
1620
|
gasLimit
|
|
@@ -898,25 +1642,23 @@ export class SequencerPublisher {
|
|
|
898
1642
|
calldataGas,
|
|
899
1643
|
calldataSize,
|
|
900
1644
|
sender,
|
|
901
|
-
...
|
|
1645
|
+
...checkpoint.getStats(),
|
|
902
1646
|
eventName: 'rollup-published-to-l1',
|
|
903
1647
|
blobCount: encodedData.blobs.length,
|
|
904
1648
|
inclusionBlocks
|
|
905
1649
|
};
|
|
906
|
-
this.log.info(`Published
|
|
1650
|
+
this.log.info(`Published checkpoint ${checkpoint.number} at slot ${slot} to rollup contract`, {
|
|
907
1651
|
...stats,
|
|
908
|
-
...
|
|
909
|
-
...receipt
|
|
1652
|
+
...checkpoint.getStats(),
|
|
1653
|
+
...pick(receipt, 'transactionHash', 'blockHash')
|
|
910
1654
|
});
|
|
911
1655
|
this.metrics.recordProcessBlockTx(timer.ms(), publishStats);
|
|
912
1656
|
return true;
|
|
913
1657
|
} else {
|
|
914
1658
|
this.metrics.recordFailedTx('process');
|
|
915
|
-
this.log.error(`
|
|
916
|
-
...
|
|
917
|
-
receipt
|
|
918
|
-
txHash: receipt.transactionHash,
|
|
919
|
-
slotNumber: block.header.globalVariables.slotNumber
|
|
1659
|
+
this.log.error(`Publishing checkpoint at slot ${slot} failed with ${errorMsg ?? 'no error message'}`, undefined, {
|
|
1660
|
+
...checkpoint.getStats(),
|
|
1661
|
+
...receipt
|
|
920
1662
|
});
|
|
921
1663
|
return false;
|
|
922
1664
|
}
|