@aztec/epoch-cache 0.0.1-commit.9b94fc1 → 0.0.1-commit.9badcec54

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