@aztec/epoch-cache 0.0.1-commit.b33fc05d0 → 0.0.1-commit.b3d3157a
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/README.md +210 -0
- package/dest/config.d.ts +2 -2
- package/dest/config.d.ts.map +1 -1
- package/dest/epoch_cache.d.ts +70 -17
- package/dest/epoch_cache.d.ts.map +1 -1
- package/dest/epoch_cache.js +207 -67
- package/dest/test/test_epoch_cache.d.ts +15 -3
- package/dest/test/test_epoch_cache.d.ts.map +1 -1
- package/dest/test/test_epoch_cache.js +47 -11
- package/package.json +6 -6
- package/src/config.ts +1 -1
- package/src/epoch_cache.ts +264 -62
- package/src/test/test_epoch_cache.ts +68 -12
package/src/epoch_cache.ts
CHANGED
|
@@ -1,5 +1,7 @@
|
|
|
1
1
|
import { createEthereumChain } from '@aztec/ethereum/chain';
|
|
2
|
+
import { makeL1HttpTransport } from '@aztec/ethereum/client';
|
|
2
3
|
import { NoCommitteeError, RollupContract } from '@aztec/ethereum/contracts';
|
|
4
|
+
import { getFinalizedL1Block } from '@aztec/ethereum/queries';
|
|
3
5
|
import { EpochNumber, SlotNumber } from '@aztec/foundation/branded-types';
|
|
4
6
|
import { EthAddress } from '@aztec/foundation/eth-address';
|
|
5
7
|
import { type Logger, createLogger } from '@aztec/foundation/log';
|
|
@@ -8,19 +10,25 @@ import {
|
|
|
8
10
|
type L1RollupConstants,
|
|
9
11
|
getEpochAtSlot,
|
|
10
12
|
getEpochNumberAtTimestamp,
|
|
13
|
+
getNextL1SlotTimestamp,
|
|
14
|
+
getSlotAtNextL1Block,
|
|
11
15
|
getSlotAtTimestamp,
|
|
12
16
|
getSlotRangeForEpoch,
|
|
17
|
+
getStartTimestampForEpoch,
|
|
13
18
|
getTimestampForSlot,
|
|
14
|
-
getTimestampRangeForEpoch,
|
|
15
19
|
} from '@aztec/stdlib/epoch-helpers';
|
|
16
20
|
|
|
17
|
-
import { createPublicClient, encodeAbiParameters,
|
|
21
|
+
import { createPublicClient, encodeAbiParameters, keccak256 } from 'viem';
|
|
18
22
|
|
|
19
23
|
import { type EpochCacheConfig, getEpochCacheConfigEnvVars } from './config.js';
|
|
20
24
|
|
|
25
|
+
/** The proposer pipelines by building one slot ahead. */
|
|
26
|
+
export const PROPOSER_PIPELINING_SLOT_OFFSET = 1;
|
|
27
|
+
|
|
28
|
+
/** Flat return type for compound epoch/slot getters. */
|
|
21
29
|
export type EpochAndSlot = {
|
|
22
|
-
epoch: EpochNumber;
|
|
23
30
|
slot: SlotNumber;
|
|
31
|
+
epoch: EpochNumber;
|
|
24
32
|
ts: bigint;
|
|
25
33
|
};
|
|
26
34
|
|
|
@@ -34,13 +42,38 @@ export type EpochCommitteeInfo = {
|
|
|
34
42
|
|
|
35
43
|
export type SlotTag = 'now' | 'next' | SlotNumber;
|
|
36
44
|
|
|
45
|
+
/** Minimal L1 block info used for cache provenance. */
|
|
46
|
+
type L1BlockInfo = { number: bigint; hash: `0x${string}`; timestamp: bigint };
|
|
47
|
+
|
|
48
|
+
/** Resolved cache entry with L1 provenance metadata. */
|
|
49
|
+
type CachedEpochEntry = {
|
|
50
|
+
data: EpochCommitteeInfo;
|
|
51
|
+
/** L1 block number at which the committee data was originally queried. */
|
|
52
|
+
lastQueryL1BlockNumber: bigint;
|
|
53
|
+
/** L1 block hash at which the committee data was originally queried. Used to detect reorgs. */
|
|
54
|
+
lastQueryL1BlockHash: `0x${string}`;
|
|
55
|
+
/** Latest L1 block timestamp at the time of the most recent refresh (full fetch or lightweight check). */
|
|
56
|
+
lastRefreshL1Timestamp: bigint;
|
|
57
|
+
/** Whether the epoch's sampling data falls within finalized L1 history. */
|
|
58
|
+
finalized: boolean;
|
|
59
|
+
};
|
|
60
|
+
|
|
37
61
|
export interface EpochCacheInterface {
|
|
38
62
|
getCommittee(slot: SlotTag | undefined): Promise<EpochCommitteeInfo>;
|
|
63
|
+
getSlotNow(): SlotNumber;
|
|
64
|
+
getTargetSlot(): SlotNumber;
|
|
65
|
+
getEpochNow(): EpochNumber;
|
|
66
|
+
getTargetEpoch(): EpochNumber;
|
|
39
67
|
getEpochAndSlotNow(): EpochAndSlot & { nowMs: bigint };
|
|
40
|
-
getEpochAndSlotInNextL1Slot(): EpochAndSlot & {
|
|
68
|
+
getEpochAndSlotInNextL1Slot(): EpochAndSlot & { nowSeconds: bigint };
|
|
69
|
+
/** Returns epoch/slot info for the next L1 slot with pipeline offset applied. */
|
|
70
|
+
getTargetEpochAndSlotInNextL1Slot(): EpochAndSlot & { nowSeconds: bigint };
|
|
71
|
+
isEscapeHatchOpen(epoch: EpochNumber): Promise<boolean>;
|
|
72
|
+
isEscapeHatchOpenAtSlot(slot: SlotTag): Promise<boolean>;
|
|
41
73
|
getProposerIndexEncoding(epoch: EpochNumber, slot: SlotNumber, seed: bigint): `0x${string}`;
|
|
42
74
|
computeProposerIndex(slot: SlotNumber, epoch: EpochNumber, seed: bigint, size: bigint): bigint;
|
|
43
75
|
getCurrentAndNextSlot(): { currentSlot: SlotNumber; nextSlot: SlotNumber };
|
|
76
|
+
getTargetAndNextSlot(): { targetSlot: SlotNumber; nextSlot: SlotNumber };
|
|
44
77
|
getProposerAttesterAddressInSlot(slot: SlotNumber): Promise<EthAddress | undefined>;
|
|
45
78
|
getRegisteredValidators(): Promise<EthAddress[]>;
|
|
46
79
|
isInCommittee(slot: SlotTag, validator: EthAddress): Promise<boolean>;
|
|
@@ -58,8 +91,11 @@ export interface EpochCacheInterface {
|
|
|
58
91
|
* Note: This class is very dependent on the system clock being in sync.
|
|
59
92
|
*/
|
|
60
93
|
export class EpochCache implements EpochCacheInterface {
|
|
61
|
-
|
|
62
|
-
|
|
94
|
+
/**
|
|
95
|
+
* Single map holding both resolved entries and in-flight promises.
|
|
96
|
+
* A `Promise` value means a fetch is in progress; concurrent callers await it.
|
|
97
|
+
*/
|
|
98
|
+
protected cache: Map<EpochNumber, CachedEpochEntry | Promise<CachedEpochEntry>> = new Map();
|
|
63
99
|
private allValidators: Set<string> = new Set();
|
|
64
100
|
private lastValidatorRefresh = 0;
|
|
65
101
|
private readonly log: Logger = createLogger('epoch-cache');
|
|
@@ -93,7 +129,7 @@ export class EpochCache implements EpochCacheInterface {
|
|
|
93
129
|
const chain = createEthereumChain(config.l1RpcUrls, config.l1ChainId);
|
|
94
130
|
const publicClient = createPublicClient({
|
|
95
131
|
chain: chain.chainInfo,
|
|
96
|
-
transport:
|
|
132
|
+
transport: makeL1HttpTransport(config.l1RpcUrls, { timeout: config.l1HttpTimeoutMS }),
|
|
97
133
|
pollingInterval: config.viemPollingIntervalMS,
|
|
98
134
|
});
|
|
99
135
|
rollup = new RollupContract(publicClient, rollupOrAddress.toString());
|
|
@@ -134,41 +170,64 @@ export class EpochCache implements EpochCacheInterface {
|
|
|
134
170
|
rollupManaLimit: Number(rollupManaLimit),
|
|
135
171
|
};
|
|
136
172
|
|
|
137
|
-
return new EpochCache(rollup, l1RollupConstants, deps.dateProvider
|
|
173
|
+
return new EpochCache(rollup, l1RollupConstants, deps.dateProvider, {
|
|
174
|
+
cacheSize: 12,
|
|
175
|
+
validatorRefreshIntervalSeconds: 60,
|
|
176
|
+
});
|
|
138
177
|
}
|
|
139
178
|
|
|
140
179
|
public getL1Constants(): L1RollupConstants {
|
|
141
180
|
return this.l1constants;
|
|
142
181
|
}
|
|
143
182
|
|
|
183
|
+
public getSlotNow(): SlotNumber {
|
|
184
|
+
return this.getEpochAndSlotNow().slot;
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
public getTargetSlot(): SlotNumber {
|
|
188
|
+
const slotNow = this.getSlotNow();
|
|
189
|
+
const offset = PROPOSER_PIPELINING_SLOT_OFFSET;
|
|
190
|
+
return SlotNumber(slotNow + offset);
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
public getEpochNow(): EpochNumber {
|
|
194
|
+
return this.getEpochAndSlotNow().epoch;
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
public getTargetEpoch(): EpochNumber {
|
|
198
|
+
return getEpochAtSlot(this.getTargetSlot(), this.l1constants);
|
|
199
|
+
}
|
|
200
|
+
|
|
144
201
|
public getEpochAndSlotNow(): EpochAndSlot & { nowMs: bigint } {
|
|
145
202
|
const nowMs = BigInt(this.dateProvider.now());
|
|
146
203
|
const nowSeconds = nowMs / 1000n;
|
|
147
204
|
return { ...this.getEpochAndSlotAtTimestamp(nowSeconds), nowMs };
|
|
148
205
|
}
|
|
149
206
|
|
|
150
|
-
|
|
151
|
-
return
|
|
207
|
+
private getEpochAndSlotAtSlot(slot: SlotNumber): EpochAndSlot {
|
|
208
|
+
return this.getEpochAndSlotAtTimestamp(getTimestampForSlot(slot, this.l1constants));
|
|
152
209
|
}
|
|
153
210
|
|
|
154
|
-
|
|
155
|
-
const
|
|
156
|
-
const
|
|
157
|
-
return {
|
|
211
|
+
public getEpochAndSlotInNextL1Slot(): EpochAndSlot & { nowSeconds: bigint } {
|
|
212
|
+
const nowSeconds = this.dateProvider.nowInSeconds();
|
|
213
|
+
const nextSlotTs = getNextL1SlotTimestamp(nowSeconds, this.l1constants);
|
|
214
|
+
return { ...this.getEpochAndSlotAtTimestamp(nextSlotTs), nowSeconds: BigInt(nowSeconds) };
|
|
158
215
|
}
|
|
159
216
|
|
|
160
|
-
public
|
|
161
|
-
const
|
|
162
|
-
const
|
|
163
|
-
|
|
217
|
+
public getTargetEpochAndSlotInNextL1Slot(): EpochAndSlot & { nowSeconds: bigint } {
|
|
218
|
+
const result = this.getEpochAndSlotInNextL1Slot();
|
|
219
|
+
const offset = PROPOSER_PIPELINING_SLOT_OFFSET;
|
|
220
|
+
const targetSlot = SlotNumber(result.slot + offset);
|
|
221
|
+
return { ...result, slot: targetSlot, epoch: getEpochAtSlot(targetSlot, this.l1constants) };
|
|
164
222
|
}
|
|
165
223
|
|
|
166
224
|
private getEpochAndSlotAtTimestamp(ts: bigint): EpochAndSlot {
|
|
167
225
|
const slot = getSlotAtTimestamp(ts, this.l1constants);
|
|
226
|
+
const epoch = getEpochNumberAtTimestamp(ts, this.l1constants);
|
|
168
227
|
return {
|
|
169
|
-
epoch: getEpochNumberAtTimestamp(ts, this.l1constants),
|
|
170
|
-
ts: getTimestampForSlot(slot, this.l1constants),
|
|
171
228
|
slot,
|
|
229
|
+
epoch,
|
|
230
|
+
ts: getTimestampForSlot(slot, this.l1constants),
|
|
172
231
|
};
|
|
173
232
|
}
|
|
174
233
|
|
|
@@ -177,17 +236,8 @@ export class EpochCache implements EpochCacheInterface {
|
|
|
177
236
|
return this.getCommittee(startSlot);
|
|
178
237
|
}
|
|
179
238
|
|
|
180
|
-
/**
|
|
181
|
-
* Returns whether the escape hatch is open for the given epoch.
|
|
182
|
-
*
|
|
183
|
-
* Uses the already-cached EpochCommitteeInfo when available. If not cached, it will fetch
|
|
184
|
-
* the epoch committee info (which includes the escape hatch flag) and return it.
|
|
185
|
-
*/
|
|
239
|
+
/** Returns whether the escape hatch is open for the given epoch. */
|
|
186
240
|
public async isEscapeHatchOpen(epoch: EpochNumber): Promise<boolean> {
|
|
187
|
-
const cached = this.cache.get(epoch);
|
|
188
|
-
if (cached) {
|
|
189
|
-
return cached.isEscapeHatchOpen;
|
|
190
|
-
}
|
|
191
241
|
const info = await this.getCommitteeForEpoch(epoch);
|
|
192
242
|
return info.isEscapeHatchOpen;
|
|
193
243
|
}
|
|
@@ -201,7 +251,7 @@ export class EpochCache implements EpochCacheInterface {
|
|
|
201
251
|
public async isEscapeHatchOpenAtSlot(slot: SlotTag = 'now'): Promise<boolean> {
|
|
202
252
|
const epoch =
|
|
203
253
|
slot === 'now'
|
|
204
|
-
? this.
|
|
254
|
+
? this.getEpochNow()
|
|
205
255
|
: slot === 'next'
|
|
206
256
|
? this.getEpochAndSlotInNextL1Slot().epoch
|
|
207
257
|
: getEpochAtSlot(slot, this.l1constants);
|
|
@@ -210,33 +260,52 @@ export class EpochCache implements EpochCacheInterface {
|
|
|
210
260
|
}
|
|
211
261
|
|
|
212
262
|
/**
|
|
213
|
-
* Get the current validator set
|
|
214
|
-
*
|
|
215
|
-
*
|
|
263
|
+
* Get the current validator set.
|
|
264
|
+
*
|
|
265
|
+
* Returns cached data if the entry is finalized or still fresh (queried less than one
|
|
266
|
+
* Ethereum slot ago). Stale non-finalized entries are re-queried, and concurrent callers
|
|
267
|
+
* coalesce on the same in-flight promise so the L1 query happens only once.
|
|
216
268
|
*/
|
|
217
269
|
public async getCommittee(slot: SlotTag = 'now'): Promise<EpochCommitteeInfo> {
|
|
218
270
|
const { epoch, ts } = this.getEpochAndTimestamp(slot);
|
|
219
271
|
|
|
220
|
-
|
|
221
|
-
|
|
272
|
+
const cached = this.cache.get(epoch);
|
|
273
|
+
|
|
274
|
+
// In-flight promise: another caller is already fetching this epoch — just await it.
|
|
275
|
+
if (cached instanceof Promise) {
|
|
276
|
+
return (await cached).data;
|
|
222
277
|
}
|
|
223
278
|
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
return epochData;
|
|
279
|
+
// Resolved entry: return it if finalized or still fresh.
|
|
280
|
+
if (cached && (cached.finalized || !this.isStale(cached))) {
|
|
281
|
+
return cached.data;
|
|
228
282
|
}
|
|
229
|
-
this.cache.set(epoch, epochData);
|
|
230
283
|
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
284
|
+
// Stale non-finalized entry: do a lightweight refresh first (check block hash + finalized ts).
|
|
285
|
+
// Only fall back to a full re-fetch if the L1 block was reorged.
|
|
286
|
+
if (cached) {
|
|
287
|
+
const promise = this.refreshStaleEntry(cached, epoch, ts);
|
|
288
|
+
this.cache.set(epoch, promise);
|
|
289
|
+
try {
|
|
290
|
+
return (await promise).data;
|
|
291
|
+
} catch (err) {
|
|
292
|
+
this.cache.set(epoch, cached);
|
|
293
|
+
throw err;
|
|
294
|
+
}
|
|
295
|
+
}
|
|
235
296
|
|
|
236
|
-
|
|
297
|
+
// No entry at all: full fetch.
|
|
298
|
+
const promise = this.fetchAndCache(epoch, ts);
|
|
299
|
+
this.cache.set(epoch, promise);
|
|
300
|
+
try {
|
|
301
|
+
return (await promise).data;
|
|
302
|
+
} catch (err) {
|
|
303
|
+
this.cache.delete(epoch);
|
|
304
|
+
throw err;
|
|
305
|
+
}
|
|
237
306
|
}
|
|
238
307
|
|
|
239
|
-
private getEpochAndTimestamp(slot: SlotTag = 'now') {
|
|
308
|
+
private getEpochAndTimestamp(slot: SlotTag = 'now'): { epoch: EpochNumber; ts: bigint } {
|
|
240
309
|
if (slot === 'now') {
|
|
241
310
|
return this.getEpochAndSlotNow();
|
|
242
311
|
} else if (slot === 'next') {
|
|
@@ -246,22 +315,140 @@ export class EpochCache implements EpochCacheInterface {
|
|
|
246
315
|
}
|
|
247
316
|
}
|
|
248
317
|
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
318
|
+
/** Evicts oldest cache entries (resolved or in-flight) beyond cacheSize. */
|
|
319
|
+
private purgeCache(): void {
|
|
320
|
+
if (this.cache.size <= this.config.cacheSize) {
|
|
321
|
+
return;
|
|
322
|
+
}
|
|
323
|
+
const toPurge = Array.from(this.cache.keys())
|
|
324
|
+
.sort((a, b) => Number(b - a))
|
|
325
|
+
.slice(this.config.cacheSize);
|
|
326
|
+
toPurge.forEach(key => this.cache.delete(key));
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
/** Returns true if a non-finalized cache entry is older than one Ethereum slot. */
|
|
330
|
+
private isStale(entry: CachedEpochEntry): boolean {
|
|
331
|
+
const nowSeconds = BigInt(this.dateProvider.nowInSeconds());
|
|
332
|
+
return nowSeconds - entry.lastRefreshL1Timestamp >= BigInt(this.l1constants.ethereumSlotDuration);
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
/** Whether a cached epoch entry has been marked as finalized. Returns undefined if not cached or still in-flight. */
|
|
336
|
+
public isFinalized(epoch: EpochNumber): boolean | undefined {
|
|
337
|
+
const entry = this.cache.get(epoch);
|
|
338
|
+
if (!entry || entry instanceof Promise) {
|
|
339
|
+
return undefined;
|
|
340
|
+
}
|
|
341
|
+
return entry.finalized;
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
/** Returns the latest L1 timestamp stored in the cached entry. Undefined if not cached or in-flight. */
|
|
345
|
+
public getCachedLastRefreshL1Timestamp(epoch: EpochNumber): bigint | undefined {
|
|
346
|
+
const entry = this.cache.get(epoch);
|
|
347
|
+
if (!entry || entry instanceof Promise) {
|
|
348
|
+
return undefined;
|
|
349
|
+
}
|
|
350
|
+
return entry.lastRefreshL1Timestamp;
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
/** Computes the sampling timestamp for an epoch's committee data. */
|
|
354
|
+
private getSamplingTimestamp(epoch: EpochNumber): bigint {
|
|
355
|
+
const { lagInEpochsForRandao, epochDuration, slotDuration } = this.l1constants;
|
|
356
|
+
const epochStartTs = getStartTimestampForEpoch(epoch, this.l1constants);
|
|
357
|
+
return epochStartTs - BigInt(lagInEpochsForRandao) * BigInt(epochDuration) * BigInt(slotDuration);
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
/**
|
|
361
|
+
* Lightweight refresh for a stale non-finalized entry. Queries only the block hash at
|
|
362
|
+
* the original block number and the finalized block timestamp — avoids the expensive
|
|
363
|
+
* getCommitteeAt and getSampleSeedAt calls on the rollup contract.
|
|
364
|
+
*
|
|
365
|
+
* If the block hash still matches (no L1 reorg), we keep the existing data and just
|
|
366
|
+
* update the provenance timestamp. If the finalized block has caught up, we promote the
|
|
367
|
+
* entry to finalized. If there was a reorg (hash mismatch), we fall back to a full fetch.
|
|
368
|
+
*/
|
|
369
|
+
private async refreshStaleEntry(stale: CachedEpochEntry, epoch: EpochNumber, ts: bigint): Promise<CachedEpochEntry> {
|
|
370
|
+
const [blockAtOriginal, l1FinalizedBlock, latestBlock] = await Promise.all([
|
|
371
|
+
this.rollup.client.getBlock({ blockNumber: stale.lastQueryL1BlockNumber, includeTransactions: false }),
|
|
372
|
+
getFinalizedL1Block(this.rollup.client),
|
|
373
|
+
this.rollup.client.getBlock({ includeTransactions: false }),
|
|
374
|
+
]);
|
|
375
|
+
|
|
376
|
+
if (blockAtOriginal.hash === stale.lastQueryL1BlockHash) {
|
|
377
|
+
// No reorg: the data is still valid. Check if we can now mark it as finalized.
|
|
378
|
+
const samplingTs = this.getSamplingTimestamp(epoch);
|
|
379
|
+
const finalized =
|
|
380
|
+
!!(stale.data.committee && stale.data.committee.length > 0) &&
|
|
381
|
+
l1FinalizedBlock !== undefined &&
|
|
382
|
+
samplingTs <= l1FinalizedBlock.timestamp;
|
|
383
|
+
|
|
384
|
+
const refreshed: CachedEpochEntry = {
|
|
385
|
+
...stale,
|
|
386
|
+
lastRefreshL1Timestamp: latestBlock.timestamp,
|
|
387
|
+
finalized,
|
|
388
|
+
};
|
|
389
|
+
this.cache.set(epoch, refreshed);
|
|
390
|
+
return refreshed;
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
// Reorg detected: block hash mismatch. Do a full re-fetch.
|
|
394
|
+
// Pass the already-fetched block timestamps to avoid redundant queries.
|
|
395
|
+
this.log.warn(`L1 reorg detected for epoch ${epoch}: block ${stale.lastQueryL1BlockNumber} hash changed`, {
|
|
396
|
+
epoch,
|
|
397
|
+
expectedHash: stale.lastQueryL1BlockHash,
|
|
398
|
+
actualHash: blockAtOriginal.hash,
|
|
399
|
+
});
|
|
400
|
+
return this.fetchAndCache(epoch, ts, { latestBlock, finalizedBlock: l1FinalizedBlock });
|
|
401
|
+
}
|
|
402
|
+
|
|
403
|
+
/**
|
|
404
|
+
* Fetches committee data from L1, determines finalization status, and stores in the cache.
|
|
405
|
+
*
|
|
406
|
+
* Uses `lagInEpochsForRandao` (the binding constraint, always <= lagInEpochsForValidatorSet)
|
|
407
|
+
* and computes the sampling timestamp from the epoch start to match the L1 contract's logic.
|
|
408
|
+
*
|
|
409
|
+
* When called from refreshStaleEntry after a reorg, the latest and finalized blocks are
|
|
410
|
+
* passed in to avoid redundant L1 queries.
|
|
411
|
+
*/
|
|
412
|
+
private async fetchAndCache(
|
|
413
|
+
epoch: EpochNumber,
|
|
414
|
+
ts: bigint,
|
|
415
|
+
prefetched?: { latestBlock: L1BlockInfo; finalizedBlock: { timestamp: bigint } | undefined },
|
|
416
|
+
): Promise<CachedEpochEntry> {
|
|
417
|
+
const [committee, seedBuffer, latestBlock, finalizedBlock, isEscapeHatchOpen] = await Promise.all([
|
|
252
418
|
this.rollup.getCommitteeAt(ts),
|
|
253
419
|
this.rollup.getSampleSeedAt(ts),
|
|
254
|
-
this.rollup.client.getBlock({ includeTransactions: false })
|
|
420
|
+
prefetched?.latestBlock ?? this.rollup.client.getBlock({ includeTransactions: false }),
|
|
421
|
+
prefetched !== undefined ? prefetched.finalizedBlock : getFinalizedL1Block(this.rollup.client),
|
|
255
422
|
this.rollup.isEscapeHatchOpen(epoch),
|
|
256
423
|
]);
|
|
257
|
-
|
|
258
|
-
const
|
|
259
|
-
|
|
424
|
+
|
|
425
|
+
const samplingTs = this.getSamplingTimestamp(epoch);
|
|
426
|
+
|
|
427
|
+
if (samplingTs > latestBlock.timestamp) {
|
|
260
428
|
throw new Error(
|
|
261
|
-
`Cannot query committee for future epoch ${epoch}
|
|
429
|
+
`Cannot query committee for future epoch ${epoch}: ` +
|
|
430
|
+
`sampling timestamp ${samplingTs} is beyond latest L1 block at ${latestBlock.timestamp}. ` +
|
|
431
|
+
`Check your Ethereum node is synced.`,
|
|
262
432
|
);
|
|
263
433
|
}
|
|
264
|
-
|
|
434
|
+
|
|
435
|
+
// Empty committees are never marked finalized so they always get re-queried after TTL.
|
|
436
|
+
// If L1 has no finalized block yet (devnet startup), entries stay unfinalized.
|
|
437
|
+
const hasCommittee = !!(committee && committee.length > 0);
|
|
438
|
+
const finalized = hasCommittee && finalizedBlock !== undefined && samplingTs <= finalizedBlock.timestamp;
|
|
439
|
+
const data: EpochCommitteeInfo = { committee, seed: seedBuffer.toBigInt(), epoch, isEscapeHatchOpen };
|
|
440
|
+
const entry: CachedEpochEntry = {
|
|
441
|
+
data,
|
|
442
|
+
lastQueryL1BlockNumber: latestBlock.number!,
|
|
443
|
+
lastQueryL1BlockHash: latestBlock.hash!,
|
|
444
|
+
lastRefreshL1Timestamp: latestBlock.timestamp,
|
|
445
|
+
finalized,
|
|
446
|
+
};
|
|
447
|
+
|
|
448
|
+
this.cache.set(epoch, entry);
|
|
449
|
+
this.purgeCache();
|
|
450
|
+
|
|
451
|
+
return entry;
|
|
265
452
|
}
|
|
266
453
|
|
|
267
454
|
/**
|
|
@@ -286,17 +473,31 @@ export class EpochCache implements EpochCacheInterface {
|
|
|
286
473
|
return BigInt(keccak256(this.getProposerIndexEncoding(epoch, slot, seed))) % size;
|
|
287
474
|
}
|
|
288
475
|
|
|
289
|
-
/** Returns the current and next L2 slot
|
|
476
|
+
/** Returns the current and next L2 slot in next eth L1 Slot. */
|
|
290
477
|
public getCurrentAndNextSlot(): { currentSlot: SlotNumber; nextSlot: SlotNumber } {
|
|
291
|
-
const
|
|
478
|
+
const currentSlot = this.getSlotNow();
|
|
292
479
|
const next = this.getEpochAndSlotInNextL1Slot();
|
|
293
480
|
|
|
294
481
|
return {
|
|
295
|
-
currentSlot
|
|
482
|
+
currentSlot,
|
|
296
483
|
nextSlot: next.slot,
|
|
297
484
|
};
|
|
298
485
|
}
|
|
299
486
|
|
|
487
|
+
/** Returns the target and next L2 slot in the next L1 slot. */
|
|
488
|
+
public getTargetAndNextSlot(): { targetSlot: SlotNumber; nextSlot: SlotNumber } {
|
|
489
|
+
const nowSeconds = BigInt(this.dateProvider.nowInSeconds());
|
|
490
|
+
const offset = PROPOSER_PIPELINING_SLOT_OFFSET;
|
|
491
|
+
|
|
492
|
+
const currentSlot = getSlotAtTimestamp(nowSeconds, this.l1constants);
|
|
493
|
+
const targetSlot = SlotNumber(currentSlot + offset);
|
|
494
|
+
|
|
495
|
+
const nextL2SlotOnL1 = getSlotAtNextL1Block(nowSeconds, this.l1constants);
|
|
496
|
+
const nextSlot = SlotNumber(nextL2SlotOnL1 + offset);
|
|
497
|
+
|
|
498
|
+
return { targetSlot, nextSlot };
|
|
499
|
+
}
|
|
500
|
+
|
|
300
501
|
/**
|
|
301
502
|
* Get the proposer attester address in the given L2 slot
|
|
302
503
|
* @returns The proposer attester address. If the committee does not exist, we throw a NoCommitteeError.
|
|
@@ -375,10 +576,11 @@ export class EpochCache implements EpochCacheInterface {
|
|
|
375
576
|
async getRegisteredValidators(): Promise<EthAddress[]> {
|
|
376
577
|
const validatorRefreshIntervalMs = this.config.validatorRefreshIntervalSeconds * 1000;
|
|
377
578
|
const validatorRefreshTime = this.lastValidatorRefresh + validatorRefreshIntervalMs;
|
|
378
|
-
|
|
379
|
-
|
|
579
|
+
const now = this.dateProvider.now();
|
|
580
|
+
if (validatorRefreshTime < now) {
|
|
581
|
+
const currentSet = await this.rollup.getAttesters(BigInt(Math.floor(now / 1000)));
|
|
380
582
|
this.allValidators = new Set(currentSet.map(v => v.toString()));
|
|
381
|
-
this.lastValidatorRefresh =
|
|
583
|
+
this.lastValidatorRefresh = now;
|
|
382
584
|
}
|
|
383
585
|
return Array.from(this.allValidators.keys()).map(v => EthAddress.fromString(v));
|
|
384
586
|
}
|
|
@@ -3,7 +3,13 @@ import { EthAddress } from '@aztec/foundation/eth-address';
|
|
|
3
3
|
import type { L1RollupConstants } from '@aztec/stdlib/epoch-helpers';
|
|
4
4
|
import { getEpochAtSlot, getSlotAtTimestamp, getTimestampRangeForEpoch } from '@aztec/stdlib/epoch-helpers';
|
|
5
5
|
|
|
6
|
-
import
|
|
6
|
+
import {
|
|
7
|
+
type EpochAndSlot,
|
|
8
|
+
type EpochCacheInterface,
|
|
9
|
+
type EpochCommitteeInfo,
|
|
10
|
+
PROPOSER_PIPELINING_SLOT_OFFSET,
|
|
11
|
+
type SlotTag,
|
|
12
|
+
} from '../epoch_cache.js';
|
|
7
13
|
|
|
8
14
|
/** Default L1 constants for testing. */
|
|
9
15
|
const DEFAULT_L1_CONSTANTS: L1RollupConstants = {
|
|
@@ -114,19 +120,52 @@ export class TestEpochCache implements EpochCacheInterface {
|
|
|
114
120
|
});
|
|
115
121
|
}
|
|
116
122
|
|
|
123
|
+
getSlotNow(): SlotNumber {
|
|
124
|
+
return this.currentSlot;
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
getTargetSlot(): SlotNumber {
|
|
128
|
+
return SlotNumber(this.currentSlot + PROPOSER_PIPELINING_SLOT_OFFSET);
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
getEpochNow(): EpochNumber {
|
|
132
|
+
return getEpochAtSlot(this.currentSlot, this.l1Constants);
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
getTargetEpoch(): EpochNumber {
|
|
136
|
+
return getEpochAtSlot(this.getTargetSlot(), this.l1Constants);
|
|
137
|
+
}
|
|
138
|
+
|
|
117
139
|
getEpochAndSlotNow(): EpochAndSlot & { nowMs: bigint } {
|
|
118
|
-
const
|
|
119
|
-
const ts = getTimestampRangeForEpoch(
|
|
120
|
-
return {
|
|
140
|
+
const epochNow = getEpochAtSlot(this.currentSlot, this.l1Constants);
|
|
141
|
+
const ts = getTimestampRangeForEpoch(epochNow, this.l1Constants)[0];
|
|
142
|
+
return {
|
|
143
|
+
epoch: epochNow,
|
|
144
|
+
slot: this.currentSlot,
|
|
145
|
+
ts,
|
|
146
|
+
nowMs: ts * 1000n,
|
|
147
|
+
};
|
|
121
148
|
}
|
|
122
149
|
|
|
123
|
-
getEpochAndSlotInNextL1Slot(): EpochAndSlot & {
|
|
124
|
-
const
|
|
125
|
-
const nextSlotTs =
|
|
150
|
+
getEpochAndSlotInNextL1Slot(): EpochAndSlot & { nowSeconds: bigint } {
|
|
151
|
+
const nowTs = getTimestampRangeForEpoch(getEpochAtSlot(this.currentSlot, this.l1Constants), this.l1Constants)[0];
|
|
152
|
+
const nextSlotTs = nowTs + BigInt(this.l1Constants.ethereumSlotDuration);
|
|
126
153
|
const nextSlot = getSlotAtTimestamp(nextSlotTs, this.l1Constants);
|
|
127
|
-
const
|
|
128
|
-
const ts = getTimestampRangeForEpoch(
|
|
129
|
-
return {
|
|
154
|
+
const epochNow = getEpochAtSlot(nextSlot, this.l1Constants);
|
|
155
|
+
const ts = getTimestampRangeForEpoch(epochNow, this.l1Constants)[0];
|
|
156
|
+
return {
|
|
157
|
+
epoch: epochNow,
|
|
158
|
+
slot: nextSlot,
|
|
159
|
+
ts,
|
|
160
|
+
nowSeconds: nowTs,
|
|
161
|
+
};
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
getTargetEpochAndSlotInNextL1Slot(): EpochAndSlot & { nowSeconds: bigint } {
|
|
165
|
+
const result = this.getEpochAndSlotInNextL1Slot();
|
|
166
|
+
const offset = PROPOSER_PIPELINING_SLOT_OFFSET;
|
|
167
|
+
const targetSlot = SlotNumber(result.slot + offset);
|
|
168
|
+
return { ...result, slot: targetSlot, epoch: getEpochAtSlot(targetSlot, this.l1Constants) };
|
|
130
169
|
}
|
|
131
170
|
|
|
132
171
|
getProposerIndexEncoding(epoch: EpochNumber, slot: SlotNumber, seed: bigint): `0x${string}` {
|
|
@@ -142,9 +181,22 @@ export class TestEpochCache implements EpochCacheInterface {
|
|
|
142
181
|
}
|
|
143
182
|
|
|
144
183
|
getCurrentAndNextSlot(): { currentSlot: SlotNumber; nextSlot: SlotNumber } {
|
|
184
|
+
const currentSlot = this.getSlotNow();
|
|
185
|
+
const next = this.getEpochAndSlotInNextL1Slot();
|
|
186
|
+
|
|
187
|
+
return {
|
|
188
|
+
currentSlot,
|
|
189
|
+
nextSlot: next.slot,
|
|
190
|
+
};
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
getTargetAndNextSlot(): { targetSlot: SlotNumber; nextSlot: SlotNumber } {
|
|
194
|
+
const targetSlot = this.getTargetSlot();
|
|
195
|
+
const next = this.getTargetEpochAndSlotInNextL1Slot();
|
|
196
|
+
|
|
145
197
|
return {
|
|
146
|
-
|
|
147
|
-
nextSlot:
|
|
198
|
+
targetSlot,
|
|
199
|
+
nextSlot: next.slot,
|
|
148
200
|
};
|
|
149
201
|
}
|
|
150
202
|
|
|
@@ -165,6 +217,10 @@ export class TestEpochCache implements EpochCacheInterface {
|
|
|
165
217
|
return Promise.resolve(validators.filter(v => committeeSet.has(v.toString())));
|
|
166
218
|
}
|
|
167
219
|
|
|
220
|
+
isEscapeHatchOpen(_epoch: EpochNumber): Promise<boolean> {
|
|
221
|
+
return Promise.resolve(this.escapeHatchOpen);
|
|
222
|
+
}
|
|
223
|
+
|
|
168
224
|
isEscapeHatchOpenAtSlot(_slot?: SlotTag): Promise<boolean> {
|
|
169
225
|
return Promise.resolve(this.escapeHatchOpen);
|
|
170
226
|
}
|