@aztec/epoch-cache 5.0.0-private.20260318 → 5.0.0-rc.1

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,12 +1,15 @@
1
1
  import { createEthereumChain } from '@aztec/ethereum/chain';
2
2
  import { makeL1HttpTransport } from '@aztec/ethereum/client';
3
3
  import { NoCommitteeError, RollupContract } from '@aztec/ethereum/contracts';
4
+ import { getFinalizedL1Block } from '@aztec/ethereum/queries';
5
+ import { SlotNumber } from '@aztec/foundation/branded-types';
4
6
  import { EthAddress } from '@aztec/foundation/eth-address';
5
7
  import { createLogger } from '@aztec/foundation/log';
6
8
  import { DateProvider } from '@aztec/foundation/timer';
7
- import { getEpochAtSlot, getEpochNumberAtTimestamp, getSlotAtTimestamp, getSlotRangeForEpoch, getTimestampForSlot, getTimestampRangeForEpoch } from '@aztec/stdlib/epoch-helpers';
9
+ import { getEpochAtSlot, getEpochNumberAtTimestamp, getNextL1SlotTimestamp, getSlotAtNextL1Block, getSlotAtTimestamp, getSlotRangeForEpoch, getStartTimestampForEpoch, getTimestampForSlot } from '@aztec/stdlib/epoch-helpers';
8
10
  import { createPublicClient, encodeAbiParameters, keccak256 } from 'viem';
9
11
  import { getEpochCacheConfigEnvVars } from './config.js';
12
+ /** The proposer pipelines by building one slot ahead. */ export const PROPOSER_PIPELINING_SLOT_OFFSET = 1;
10
13
  /**
11
14
  * Epoch cache
12
15
  *
@@ -20,8 +23,10 @@ import { getEpochCacheConfigEnvVars } from './config.js';
20
23
  l1constants;
21
24
  dateProvider;
22
25
  config;
23
- // eslint-disable-next-line aztec-custom/no-non-primitive-in-collections
24
- cache;
26
+ /**
27
+ * Single map holding both resolved entries and in-flight promises.
28
+ * A `Promise` value means a fetch is in progress; concurrent callers await it.
29
+ */ cache;
25
30
  allValidators;
26
31
  lastValidatorRefresh;
27
32
  log;
@@ -81,11 +86,28 @@ import { getEpochCacheConfigEnvVars } from './config.js';
81
86
  targetCommitteeSize: Number(targetCommitteeSize),
82
87
  rollupManaLimit: Number(rollupManaLimit)
83
88
  };
84
- return new EpochCache(rollup, l1RollupConstants, deps.dateProvider);
89
+ return new EpochCache(rollup, l1RollupConstants, deps.dateProvider, {
90
+ cacheSize: 12,
91
+ validatorRefreshIntervalSeconds: 60
92
+ });
85
93
  }
86
94
  getL1Constants() {
87
95
  return this.l1constants;
88
96
  }
97
+ getSlotNow() {
98
+ return this.getEpochAndSlotNow().slot;
99
+ }
100
+ getTargetSlot() {
101
+ const slotNow = this.getSlotNow();
102
+ const offset = PROPOSER_PIPELINING_SLOT_OFFSET;
103
+ return SlotNumber(slotNow + offset);
104
+ }
105
+ getEpochNow() {
106
+ return this.getEpochAndSlotNow().epoch;
107
+ }
108
+ getTargetEpoch() {
109
+ return getEpochAtSlot(this.getTargetSlot(), this.l1constants);
110
+ }
89
111
  getEpochAndSlotNow() {
90
112
  const nowMs = BigInt(this.dateProvider.now());
91
113
  const nowSeconds = nowMs / 1000n;
@@ -94,48 +116,41 @@ import { getEpochCacheConfigEnvVars } from './config.js';
94
116
  nowMs
95
117
  };
96
118
  }
97
- nowInSeconds() {
98
- return BigInt(Math.floor(this.dateProvider.now() / 1000));
99
- }
100
119
  getEpochAndSlotAtSlot(slot) {
101
- const epoch = getEpochAtSlot(slot, this.l1constants);
102
- const ts = getTimestampRangeForEpoch(epoch, this.l1constants)[0];
103
- return {
104
- epoch,
105
- ts,
106
- slot
107
- };
120
+ return this.getEpochAndSlotAtTimestamp(getTimestampForSlot(slot, this.l1constants));
108
121
  }
109
122
  getEpochAndSlotInNextL1Slot() {
110
- const now = this.nowInSeconds();
111
- const nextSlotTs = now + BigInt(this.l1constants.ethereumSlotDuration);
123
+ const nowSeconds = this.dateProvider.nowInSeconds();
124
+ const nextSlotTs = getNextL1SlotTimestamp(nowSeconds, this.l1constants);
112
125
  return {
113
126
  ...this.getEpochAndSlotAtTimestamp(nextSlotTs),
114
- now
127
+ nowSeconds: BigInt(nowSeconds)
128
+ };
129
+ }
130
+ getTargetEpochAndSlotInNextL1Slot() {
131
+ const result = this.getEpochAndSlotInNextL1Slot();
132
+ const offset = PROPOSER_PIPELINING_SLOT_OFFSET;
133
+ const targetSlot = SlotNumber(result.slot + offset);
134
+ return {
135
+ ...result,
136
+ slot: targetSlot,
137
+ epoch: getEpochAtSlot(targetSlot, this.l1constants)
115
138
  };
116
139
  }
117
140
  getEpochAndSlotAtTimestamp(ts) {
118
141
  const slot = getSlotAtTimestamp(ts, this.l1constants);
142
+ const epoch = getEpochNumberAtTimestamp(ts, this.l1constants);
119
143
  return {
120
- epoch: getEpochNumberAtTimestamp(ts, this.l1constants),
121
- ts: getTimestampForSlot(slot, this.l1constants),
122
- slot
144
+ slot,
145
+ epoch,
146
+ ts: getTimestampForSlot(slot, this.l1constants)
123
147
  };
124
148
  }
125
149
  getCommitteeForEpoch(epoch) {
126
150
  const [startSlot] = getSlotRangeForEpoch(epoch, this.l1constants);
127
151
  return this.getCommittee(startSlot);
128
152
  }
129
- /**
130
- * Returns whether the escape hatch is open for the given epoch.
131
- *
132
- * Uses the already-cached EpochCommitteeInfo when available. If not cached, it will fetch
133
- * the epoch committee info (which includes the escape hatch flag) and return it.
134
- */ async isEscapeHatchOpen(epoch) {
135
- const cached = this.cache.get(epoch);
136
- if (cached) {
137
- return cached.isEscapeHatchOpen;
138
- }
153
+ /** Returns whether the escape hatch is open for the given epoch. */ async isEscapeHatchOpen(epoch) {
139
154
  const info = await this.getCommitteeForEpoch(epoch);
140
155
  return info.isEscapeHatchOpen;
141
156
  }
@@ -145,30 +160,47 @@ import { getEpochCacheConfigEnvVars } from './config.js';
145
160
  * This is a lightweight helper intended for callers that already have a slot number and only
146
161
  * need the escape hatch flag (without pulling full committee info).
147
162
  */ async isEscapeHatchOpenAtSlot(slot = 'now') {
148
- const epoch = slot === 'now' ? this.getEpochAndSlotNow().epoch : slot === 'next' ? this.getEpochAndSlotInNextL1Slot().epoch : getEpochAtSlot(slot, this.l1constants);
163
+ const epoch = slot === 'now' ? this.getEpochNow() : slot === 'next' ? this.getEpochAndSlotInNextL1Slot().epoch : getEpochAtSlot(slot, this.l1constants);
149
164
  return await this.isEscapeHatchOpen(epoch);
150
165
  }
151
166
  /**
152
- * Get the current validator set
153
- * @param nextSlot - If true, get the validator set for the next slot.
154
- * @returns The current validator set.
167
+ * Get the current validator set.
168
+ *
169
+ * Returns cached data if the entry is finalized or still fresh (queried less than one
170
+ * Ethereum slot ago). Stale non-finalized entries are re-queried, and concurrent callers
171
+ * coalesce on the same in-flight promise so the L1 query happens only once.
155
172
  */ async getCommittee(slot = 'now') {
156
173
  const { epoch, ts } = this.getEpochAndTimestamp(slot);
157
- if (this.cache.has(epoch)) {
158
- return this.cache.get(epoch);
174
+ const cached = this.cache.get(epoch);
175
+ // In-flight promise: another caller is already fetching this epoch — just await it.
176
+ if (cached instanceof Promise) {
177
+ return (await cached).data;
159
178
  }
160
- const epochData = await this.computeCommittee({
161
- epoch,
162
- ts
163
- });
164
- // If the committee size is 0 or undefined, then do not cache
165
- if (!epochData.committee || epochData.committee.length === 0) {
166
- return epochData;
179
+ // Resolved entry: return it if finalized or still fresh.
180
+ if (cached && (cached.finalized || !this.isStale(cached))) {
181
+ return cached.data;
182
+ }
183
+ // Stale non-finalized entry: do a lightweight refresh first (check block hash + finalized ts).
184
+ // Only fall back to a full re-fetch if the L1 block was reorged.
185
+ if (cached) {
186
+ const promise = this.refreshStaleEntry(cached, epoch, ts);
187
+ this.cache.set(epoch, promise);
188
+ try {
189
+ return (await promise).data;
190
+ } catch (err) {
191
+ this.cache.set(epoch, cached);
192
+ throw err;
193
+ }
194
+ }
195
+ // No entry at all: full fetch.
196
+ const promise = this.fetchAndCache(epoch, ts);
197
+ this.cache.set(epoch, promise);
198
+ try {
199
+ return (await promise).data;
200
+ } catch (err) {
201
+ this.cache.delete(epoch);
202
+ throw err;
167
203
  }
168
- this.cache.set(epoch, epochData);
169
- const toPurge = Array.from(this.cache.keys()).sort((a, b)=>Number(b - a)).slice(this.config.cacheSize);
170
- toPurge.forEach((key)=>this.cache.delete(key));
171
- return epochData;
172
204
  }
173
205
  getEpochAndTimestamp(slot = 'now') {
174
206
  if (slot === 'now') {
@@ -179,27 +211,121 @@ import { getEpochCacheConfigEnvVars } from './config.js';
179
211
  return this.getEpochAndSlotAtSlot(slot);
180
212
  }
181
213
  }
182
- async computeCommittee(when) {
183
- const { ts, epoch } = when;
184
- const [committee, seedBuffer, l1Timestamp, isEscapeHatchOpen] = await Promise.all([
214
+ /** Evicts oldest cache entries (resolved or in-flight) beyond cacheSize. */ purgeCache() {
215
+ if (this.cache.size <= this.config.cacheSize) {
216
+ return;
217
+ }
218
+ const toPurge = Array.from(this.cache.keys()).sort((a, b)=>Number(b - a)).slice(this.config.cacheSize);
219
+ toPurge.forEach((key)=>this.cache.delete(key));
220
+ }
221
+ /** Returns true if a non-finalized cache entry is older than one Ethereum slot. */ isStale(entry) {
222
+ const nowSeconds = BigInt(this.dateProvider.nowInSeconds());
223
+ return nowSeconds - entry.lastRefreshL1Timestamp >= BigInt(this.l1constants.ethereumSlotDuration);
224
+ }
225
+ /** Whether a cached epoch entry has been marked as finalized. Returns undefined if not cached or still in-flight. */ isFinalized(epoch) {
226
+ const entry = this.cache.get(epoch);
227
+ if (!entry || entry instanceof Promise) {
228
+ return undefined;
229
+ }
230
+ return entry.finalized;
231
+ }
232
+ /** Returns the latest L1 timestamp stored in the cached entry. Undefined if not cached or in-flight. */ getCachedLastRefreshL1Timestamp(epoch) {
233
+ const entry = this.cache.get(epoch);
234
+ if (!entry || entry instanceof Promise) {
235
+ return undefined;
236
+ }
237
+ return entry.lastRefreshL1Timestamp;
238
+ }
239
+ /** Computes the sampling timestamp for an epoch's committee data. */ getSamplingTimestamp(epoch) {
240
+ const { lagInEpochsForRandao, epochDuration, slotDuration } = this.l1constants;
241
+ const epochStartTs = getStartTimestampForEpoch(epoch, this.l1constants);
242
+ return epochStartTs - BigInt(lagInEpochsForRandao) * BigInt(epochDuration) * BigInt(slotDuration);
243
+ }
244
+ /**
245
+ * Lightweight refresh for a stale non-finalized entry. Queries only the block hash at
246
+ * the original block number and the finalized block timestamp — avoids the expensive
247
+ * getCommitteeAt and getSampleSeedAt calls on the rollup contract.
248
+ *
249
+ * If the block hash still matches (no L1 reorg), we keep the existing data and just
250
+ * update the provenance timestamp. If the finalized block has caught up, we promote the
251
+ * entry to finalized. If there was a reorg (hash mismatch), we fall back to a full fetch.
252
+ */ async refreshStaleEntry(stale, epoch, ts) {
253
+ const [blockAtOriginal, l1FinalizedBlock, latestBlock] = await Promise.all([
254
+ this.rollup.client.getBlock({
255
+ blockNumber: stale.lastQueryL1BlockNumber,
256
+ includeTransactions: false
257
+ }),
258
+ getFinalizedL1Block(this.rollup.client),
259
+ this.rollup.client.getBlock({
260
+ includeTransactions: false
261
+ })
262
+ ]);
263
+ if (blockAtOriginal.hash === stale.lastQueryL1BlockHash) {
264
+ // No reorg: the data is still valid. Check if we can now mark it as finalized.
265
+ const samplingTs = this.getSamplingTimestamp(epoch);
266
+ const finalized = !!(stale.data.committee && stale.data.committee.length > 0) && l1FinalizedBlock !== undefined && samplingTs <= l1FinalizedBlock.timestamp;
267
+ const refreshed = {
268
+ ...stale,
269
+ lastRefreshL1Timestamp: latestBlock.timestamp,
270
+ finalized
271
+ };
272
+ this.cache.set(epoch, refreshed);
273
+ return refreshed;
274
+ }
275
+ // Reorg detected: block hash mismatch. Do a full re-fetch.
276
+ // Pass the already-fetched block timestamps to avoid redundant queries.
277
+ this.log.warn(`L1 reorg detected for epoch ${epoch}: block ${stale.lastQueryL1BlockNumber} hash changed`, {
278
+ epoch,
279
+ expectedHash: stale.lastQueryL1BlockHash,
280
+ actualHash: blockAtOriginal.hash
281
+ });
282
+ return this.fetchAndCache(epoch, ts, {
283
+ latestBlock,
284
+ finalizedBlock: l1FinalizedBlock
285
+ });
286
+ }
287
+ /**
288
+ * Fetches committee data from L1, determines finalization status, and stores in the cache.
289
+ *
290
+ * Uses `lagInEpochsForRandao` (the binding constraint, always <= lagInEpochsForValidatorSet)
291
+ * and computes the sampling timestamp from the epoch start to match the L1 contract's logic.
292
+ *
293
+ * When called from refreshStaleEntry after a reorg, the latest and finalized blocks are
294
+ * passed in to avoid redundant L1 queries.
295
+ */ async fetchAndCache(epoch, ts, prefetched) {
296
+ const [committee, seedBuffer, latestBlock, finalizedBlock, isEscapeHatchOpen] = await Promise.all([
185
297
  this.rollup.getCommitteeAt(ts),
186
298
  this.rollup.getSampleSeedAt(ts),
187
- this.rollup.client.getBlock({
299
+ prefetched?.latestBlock ?? this.rollup.client.getBlock({
188
300
  includeTransactions: false
189
- }).then((b)=>b.timestamp),
301
+ }),
302
+ prefetched !== undefined ? prefetched.finalizedBlock : getFinalizedL1Block(this.rollup.client),
190
303
  this.rollup.isEscapeHatchOpen(epoch)
191
304
  ]);
192
- const { lagInEpochsForValidatorSet, epochDuration, slotDuration } = this.l1constants;
193
- const sub = BigInt(lagInEpochsForValidatorSet) * BigInt(epochDuration) * BigInt(slotDuration);
194
- if (ts - sub > l1Timestamp) {
195
- throw new Error(`Cannot query committee for future epoch ${epoch} with timestamp ${ts} (current L1 time is ${l1Timestamp}). Check your Ethereum node is synced.`);
305
+ const samplingTs = this.getSamplingTimestamp(epoch);
306
+ if (samplingTs > latestBlock.timestamp) {
307
+ throw new Error(`Cannot query committee for future epoch ${epoch}: ` + `sampling timestamp ${samplingTs} is beyond latest L1 block at ${latestBlock.timestamp}. ` + `Check your Ethereum node is synced.`);
196
308
  }
197
- return {
309
+ // Empty committees are never marked finalized so they always get re-queried after TTL.
310
+ // If L1 has no finalized block yet (devnet startup), entries stay unfinalized.
311
+ const hasCommittee = !!(committee && committee.length > 0);
312
+ const finalized = hasCommittee && finalizedBlock !== undefined && samplingTs <= finalizedBlock.timestamp;
313
+ const data = {
198
314
  committee,
199
315
  seed: seedBuffer.toBigInt(),
200
316
  epoch,
201
317
  isEscapeHatchOpen
202
318
  };
319
+ const entry = {
320
+ data,
321
+ lastQueryL1BlockNumber: latestBlock.number,
322
+ lastQueryL1BlockHash: latestBlock.hash,
323
+ lastRefreshL1Timestamp: latestBlock.timestamp,
324
+ finalized
325
+ };
326
+ this.cache.set(epoch, entry);
327
+ this.purgeCache();
328
+ return entry;
203
329
  }
204
330
  /**
205
331
  * Get the ABI encoding of the proposer index - see ValidatorSelectionLib.sol computeProposerIndex
@@ -230,14 +356,26 @@ import { getEpochCacheConfigEnvVars } from './config.js';
230
356
  }
231
357
  return BigInt(keccak256(this.getProposerIndexEncoding(epoch, slot, seed))) % size;
232
358
  }
233
- /** Returns the current and next L2 slot numbers. */ getCurrentAndNextSlot() {
234
- const current = this.getEpochAndSlotNow();
359
+ /** Returns the current and next L2 slot in next eth L1 Slot. */ getCurrentAndNextSlot() {
360
+ const currentSlot = this.getSlotNow();
235
361
  const next = this.getEpochAndSlotInNextL1Slot();
236
362
  return {
237
- currentSlot: current.slot,
363
+ currentSlot,
238
364
  nextSlot: next.slot
239
365
  };
240
366
  }
367
+ /** Returns the target and next L2 slot in the next L1 slot. */ getTargetAndNextSlot() {
368
+ const nowSeconds = BigInt(this.dateProvider.nowInSeconds());
369
+ const offset = PROPOSER_PIPELINING_SLOT_OFFSET;
370
+ const currentSlot = getSlotAtTimestamp(nowSeconds, this.l1constants);
371
+ const targetSlot = SlotNumber(currentSlot + offset);
372
+ const nextL2SlotOnL1 = getSlotAtNextL1Block(nowSeconds, this.l1constants);
373
+ const nextSlot = SlotNumber(nextL2SlotOnL1 + offset);
374
+ return {
375
+ targetSlot,
376
+ nextSlot
377
+ };
378
+ }
241
379
  /**
242
380
  * Get the proposer attester address in the given L2 slot
243
381
  * @returns The proposer attester address. If the committee does not exist, we throw a NoCommitteeError.
@@ -295,10 +433,11 @@ import { getEpochCacheConfigEnvVars } from './config.js';
295
433
  async getRegisteredValidators() {
296
434
  const validatorRefreshIntervalMs = this.config.validatorRefreshIntervalSeconds * 1000;
297
435
  const validatorRefreshTime = this.lastValidatorRefresh + validatorRefreshIntervalMs;
298
- if (validatorRefreshTime < this.dateProvider.now()) {
299
- const currentSet = await this.rollup.getAttesters();
436
+ const now = this.dateProvider.now();
437
+ if (validatorRefreshTime < now) {
438
+ const currentSet = await this.rollup.getAttesters(BigInt(Math.floor(now / 1000)));
300
439
  this.allValidators = new Set(currentSet.map((v)=>v.toString()));
301
- this.lastValidatorRefresh = this.dateProvider.now();
440
+ this.lastValidatorRefresh = now;
302
441
  }
303
442
  return Array.from(this.allValidators.keys()).map((v)=>EthAddress.fromString(v));
304
443
  }
@@ -1,7 +1,7 @@
1
1
  import { EpochNumber, SlotNumber } from '@aztec/foundation/branded-types';
2
2
  import { EthAddress } from '@aztec/foundation/eth-address';
3
3
  import type { L1RollupConstants } from '@aztec/stdlib/epoch-helpers';
4
- import type { EpochAndSlot, EpochCacheInterface, EpochCommitteeInfo, SlotTag } from '../epoch_cache.js';
4
+ import { type EpochAndSlot, type EpochCacheInterface, type EpochCommitteeInfo, type SlotTag } from '../epoch_cache.js';
5
5
  /**
6
6
  * A test implementation of EpochCacheInterface that allows manual configuration
7
7
  * of committee, proposer, slot, and escape hatch state for use in tests.
@@ -55,11 +55,18 @@ export declare class TestEpochCache implements EpochCacheInterface {
55
55
  setL1Constants(constants: Partial<L1RollupConstants>): this;
56
56
  getL1Constants(): L1RollupConstants;
57
57
  getCommittee(_slot?: SlotTag): Promise<EpochCommitteeInfo>;
58
+ getSlotNow(): SlotNumber;
59
+ getTargetSlot(): SlotNumber;
60
+ getEpochNow(): EpochNumber;
61
+ getTargetEpoch(): EpochNumber;
58
62
  getEpochAndSlotNow(): EpochAndSlot & {
59
63
  nowMs: bigint;
60
64
  };
61
65
  getEpochAndSlotInNextL1Slot(): EpochAndSlot & {
62
- now: bigint;
66
+ nowSeconds: bigint;
67
+ };
68
+ getTargetEpochAndSlotInNextL1Slot(): EpochAndSlot & {
69
+ nowSeconds: bigint;
63
70
  };
64
71
  getProposerIndexEncoding(epoch: EpochNumber, slot: SlotNumber, seed: bigint): `0x${string}`;
65
72
  computeProposerIndex(slot: SlotNumber, _epoch: EpochNumber, _seed: bigint, size: bigint): bigint;
@@ -67,10 +74,15 @@ export declare class TestEpochCache implements EpochCacheInterface {
67
74
  currentSlot: SlotNumber;
68
75
  nextSlot: SlotNumber;
69
76
  };
77
+ getTargetAndNextSlot(): {
78
+ targetSlot: SlotNumber;
79
+ nextSlot: SlotNumber;
80
+ };
70
81
  getProposerAttesterAddressInSlot(_slot: SlotNumber): Promise<EthAddress | undefined>;
71
82
  getRegisteredValidators(): Promise<EthAddress[]>;
72
83
  isInCommittee(_slot: SlotTag, validator: EthAddress): Promise<boolean>;
73
84
  filterInCommittee(_slot: SlotTag, validators: EthAddress[]): Promise<EthAddress[]>;
85
+ isEscapeHatchOpen(_epoch: EpochNumber): Promise<boolean>;
74
86
  isEscapeHatchOpenAtSlot(_slot?: SlotTag): Promise<boolean>;
75
87
  }
76
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdF9lcG9jaF9jYWNoZS5kLnRzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3Rlc3QvdGVzdF9lcG9jaF9jYWNoZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsV0FBVyxFQUFFLFVBQVUsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQzFFLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUMzRCxPQUFPLEtBQUssRUFBRSxpQkFBaUIsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBR3JFLE9BQU8sS0FBSyxFQUFFLFlBQVksRUFBRSxtQkFBbUIsRUFBRSxrQkFBa0IsRUFBRSxPQUFPLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQWN4Rzs7Ozs7O0dBTUc7QUFDSCxxQkFBYSxjQUFlLFlBQVcsbUJBQW1CO0lBQ3hELE9BQU8sQ0FBQyxTQUFTLENBQW9CO0lBQ3JDLE9BQU8sQ0FBQyxlQUFlLENBQXlCO0lBQ2hELE9BQU8sQ0FBQyxXQUFXLENBQTZCO0lBQ2hELE9BQU8sQ0FBQyxlQUFlLENBQWtCO0lBQ3pDLE9BQU8sQ0FBQyxJQUFJLENBQWM7SUFDMUIsT0FBTyxDQUFDLG9CQUFvQixDQUFvQjtJQUNoRCxPQUFPLENBQUMsV0FBVyxDQUFvQjtJQUV2QyxZQUFZLFdBQVcsR0FBRSxPQUFPLENBQUMsaUJBQWlCLENBQU0sRUFFdkQ7SUFFRDs7O09BR0c7SUFDSCxZQUFZLENBQUMsU0FBUyxFQUFFLFVBQVUsRUFBRSxHQUFHLElBQUksQ0FHMUM7SUFFRDs7O09BR0c7SUFDSCxXQUFXLENBQUMsUUFBUSxFQUFFLFVBQVUsR0FBRyxTQUFTLEdBQUcsSUFBSSxDQUdsRDtJQUVEOzs7T0FHRztJQUNILGNBQWMsQ0FBQyxJQUFJLEVBQUUsVUFBVSxHQUFHLElBQUksQ0FHckM7SUFFRDs7O09BR0c7SUFDSCxrQkFBa0IsQ0FBQyxJQUFJLEVBQUUsT0FBTyxHQUFHLElBQUksQ0FHdEM7SUFFRDs7O09BR0c7SUFDSCxPQUFPLENBQUMsSUFBSSxFQUFFLE1BQU0sR0FBRyxJQUFJLENBRzFCO0lBRUQ7OztPQUdHO0lBQ0gsdUJBQXVCLENBQUMsVUFBVSxFQUFFLFVBQVUsRUFBRSxHQUFHLElBQUksQ0FHdEQ7SUFFRDs7O09BR0c7SUFDSCxjQUFjLENBQUMsU0FBUyxFQUFFLE9BQU8sQ0FBQyxpQkFBaUIsQ0FBQyxHQUFHLElBQUksQ0FHMUQ7SUFFRCxjQUFjLElBQUksaUJBQWlCLENBRWxDO0lBRUQsWUFBWSxDQUFDLEtBQUssQ0FBQyxFQUFFLE9BQU8sR0FBRyxPQUFPLENBQUMsa0JBQWtCLENBQUMsQ0FRekQ7SUFFRCxrQkFBa0IsSUFBSSxZQUFZLEdBQUc7UUFBRSxLQUFLLEVBQUUsTUFBTSxDQUFBO0tBQUUsQ0FJckQ7SUFFRCwyQkFBMkIsSUFBSSxZQUFZLEdBQUc7UUFBRSxHQUFHLEVBQUUsTUFBTSxDQUFBO0tBQUUsQ0FPNUQ7SUFFRCx3QkFBd0IsQ0FBQyxLQUFLLEVBQUUsV0FBVyxFQUFFLElBQUksRUFBRSxVQUFVLEVBQUUsSUFBSSxFQUFFLE1BQU0sR0FBRyxLQUFLLE1BQU0sRUFBRSxDQUcxRjtJQUVELG9CQUFvQixDQUFDLElBQUksRUFBRSxVQUFVLEVBQUUsTUFBTSxFQUFFLFdBQVcsRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLElBQUksRUFBRSxNQUFNLEdBQUcsTUFBTSxDQUsvRjtJQUVELHFCQUFxQixJQUFJO1FBQUUsV0FBVyxFQUFFLFVBQVUsQ0FBQztRQUFDLFFBQVEsRUFBRSxVQUFVLENBQUE7S0FBRSxDQUt6RTtJQUVELGdDQUFnQyxDQUFDLEtBQUssRUFBRSxVQUFVLEdBQUcsT0FBTyxDQUFDLFVBQVUsR0FBRyxTQUFTLENBQUMsQ0FFbkY7SUFFRCx1QkFBdUIsSUFBSSxPQUFPLENBQUMsVUFBVSxFQUFFLENBQUMsQ0FFL0M7SUFFRCxhQUFhLENBQUMsS0FBSyxFQUFFLE9BQU8sRUFBRSxTQUFTLEVBQUUsVUFBVSxHQUFHLE9BQU8sQ0FBQyxPQUFPLENBQUMsQ0FFckU7SUFFRCxpQkFBaUIsQ0FBQyxLQUFLLEVBQUUsT0FBTyxFQUFFLFVBQVUsRUFBRSxVQUFVLEVBQUUsR0FBRyxPQUFPLENBQUMsVUFBVSxFQUFFLENBQUMsQ0FHakY7SUFFRCx1QkFBdUIsQ0FBQyxLQUFLLENBQUMsRUFBRSxPQUFPLEdBQUcsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUV6RDtDQUNGIn0=
88
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGVzdF9lcG9jaF9jYWNoZS5kLnRzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3Rlc3QvdGVzdF9lcG9jaF9jYWNoZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsV0FBVyxFQUFFLFVBQVUsRUFBRSxNQUFNLGlDQUFpQyxDQUFDO0FBQzFFLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUMzRCxPQUFPLEtBQUssRUFBRSxpQkFBaUIsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBUXJFLE9BQU8sRUFDTCxLQUFLLFlBQVksRUFDakIsS0FBSyxtQkFBbUIsRUFDeEIsS0FBSyxrQkFBa0IsRUFFdkIsS0FBSyxPQUFPLEVBQ2IsTUFBTSxtQkFBbUIsQ0FBQztBQWMzQjs7Ozs7O0dBTUc7QUFDSCxxQkFBYSxjQUFlLFlBQVcsbUJBQW1CO0lBQ3hELE9BQU8sQ0FBQyxTQUFTLENBQW9CO0lBQ3JDLE9BQU8sQ0FBQyxlQUFlLENBQXlCO0lBQ2hELE9BQU8sQ0FBQyxXQUFXLENBQTZCO0lBQ2hELE9BQU8sQ0FBQyxlQUFlLENBQWtCO0lBQ3pDLE9BQU8sQ0FBQyxJQUFJLENBQWM7SUFDMUIsT0FBTyxDQUFDLG9CQUFvQixDQUFvQjtJQUNoRCxPQUFPLENBQUMsV0FBVyxDQUFvQjtJQUV2QyxZQUFZLFdBQVcsR0FBRSxPQUFPLENBQUMsaUJBQWlCLENBQU0sRUFFdkQ7SUFFRDs7O09BR0c7SUFDSCxZQUFZLENBQUMsU0FBUyxFQUFFLFVBQVUsRUFBRSxHQUFHLElBQUksQ0FHMUM7SUFFRDs7O09BR0c7SUFDSCxXQUFXLENBQUMsUUFBUSxFQUFFLFVBQVUsR0FBRyxTQUFTLEdBQUcsSUFBSSxDQUdsRDtJQUVEOzs7T0FHRztJQUNILGNBQWMsQ0FBQyxJQUFJLEVBQUUsVUFBVSxHQUFHLElBQUksQ0FHckM7SUFFRDs7O09BR0c7SUFDSCxrQkFBa0IsQ0FBQyxJQUFJLEVBQUUsT0FBTyxHQUFHLElBQUksQ0FHdEM7SUFFRDs7O09BR0c7SUFDSCxPQUFPLENBQUMsSUFBSSxFQUFFLE1BQU0sR0FBRyxJQUFJLENBRzFCO0lBRUQ7OztPQUdHO0lBQ0gsdUJBQXVCLENBQUMsVUFBVSxFQUFFLFVBQVUsRUFBRSxHQUFHLElBQUksQ0FHdEQ7SUFFRDs7O09BR0c7SUFDSCxjQUFjLENBQUMsU0FBUyxFQUFFLE9BQU8sQ0FBQyxpQkFBaUIsQ0FBQyxHQUFHLElBQUksQ0FHMUQ7SUFFRCxjQUFjLElBQUksaUJBQWlCLENBRWxDO0lBRUQsWUFBWSxDQUFDLEtBQUssQ0FBQyxFQUFFLE9BQU8sR0FBRyxPQUFPLENBQUMsa0JBQWtCLENBQUMsQ0FRekQ7SUFFRCxVQUFVLElBQUksVUFBVSxDQUV2QjtJQUVELGFBQWEsSUFBSSxVQUFVLENBRTFCO0lBRUQsV0FBVyxJQUFJLFdBQVcsQ0FFekI7SUFFRCxjQUFjLElBQUksV0FBVyxDQUU1QjtJQUVELGtCQUFrQixJQUFJLFlBQVksR0FBRztRQUFFLEtBQUssRUFBRSxNQUFNLENBQUE7S0FBRSxDQVlyRDtJQUVELDJCQUEyQixJQUFJLFlBQVksR0FBRztRQUFFLFVBQVUsRUFBRSxNQUFNLENBQUE7S0FBRSxDQVluRTtJQUVELGlDQUFpQyxJQUFJLFlBQVksR0FBRztRQUFFLFVBQVUsRUFBRSxNQUFNLENBQUE7S0FBRSxDQUt6RTtJQUVELHdCQUF3QixDQUFDLEtBQUssRUFBRSxXQUFXLEVBQUUsSUFBSSxFQUFFLFVBQVUsRUFBRSxJQUFJLEVBQUUsTUFBTSxHQUFHLEtBQUssTUFBTSxFQUFFLENBRzFGO0lBRUQsb0JBQW9CLENBQUMsSUFBSSxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUsV0FBVyxFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUUsSUFBSSxFQUFFLE1BQU0sR0FBRyxNQUFNLENBSy9GO0lBRUQscUJBQXFCLElBQUk7UUFBRSxXQUFXLEVBQUUsVUFBVSxDQUFDO1FBQUMsUUFBUSxFQUFFLFVBQVUsQ0FBQTtLQUFFLENBUXpFO0lBRUQsb0JBQW9CLElBQUk7UUFBRSxVQUFVLEVBQUUsVUFBVSxDQUFDO1FBQUMsUUFBUSxFQUFFLFVBQVUsQ0FBQTtLQUFFLENBUXZFO0lBRUQsZ0NBQWdDLENBQUMsS0FBSyxFQUFFLFVBQVUsR0FBRyxPQUFPLENBQUMsVUFBVSxHQUFHLFNBQVMsQ0FBQyxDQUVuRjtJQUVELHVCQUF1QixJQUFJLE9BQU8sQ0FBQyxVQUFVLEVBQUUsQ0FBQyxDQUUvQztJQUVELGFBQWEsQ0FBQyxLQUFLLEVBQUUsT0FBTyxFQUFFLFNBQVMsRUFBRSxVQUFVLEdBQUcsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUVyRTtJQUVELGlCQUFpQixDQUFDLEtBQUssRUFBRSxPQUFPLEVBQUUsVUFBVSxFQUFFLFVBQVUsRUFBRSxHQUFHLE9BQU8sQ0FBQyxVQUFVLEVBQUUsQ0FBQyxDQUdqRjtJQUVELGlCQUFpQixDQUFDLE1BQU0sRUFBRSxXQUFXLEdBQUcsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUV2RDtJQUVELHVCQUF1QixDQUFDLEtBQUssQ0FBQyxFQUFFLE9BQU8sR0FBRyxPQUFPLENBQUMsT0FBTyxDQUFDLENBRXpEO0NBQ0YifQ==
@@ -1 +1 @@
1
- {"version":3,"file":"test_epoch_cache.d.ts","sourceRoot":"","sources":["../../src/test/test_epoch_cache.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,iCAAiC,CAAC;AAC1E,OAAO,EAAE,UAAU,EAAE,MAAM,+BAA+B,CAAC;AAC3D,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,6BAA6B,CAAC;AAGrE,OAAO,KAAK,EAAE,YAAY,EAAE,mBAAmB,EAAE,kBAAkB,EAAE,OAAO,EAAE,MAAM,mBAAmB,CAAC;AAcxG;;;;;;GAMG;AACH,qBAAa,cAAe,YAAW,mBAAmB;IACxD,OAAO,CAAC,SAAS,CAAoB;IACrC,OAAO,CAAC,eAAe,CAAyB;IAChD,OAAO,CAAC,WAAW,CAA6B;IAChD,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,IAAI,CAAc;IAC1B,OAAO,CAAC,oBAAoB,CAAoB;IAChD,OAAO,CAAC,WAAW,CAAoB;IAEvC,YAAY,WAAW,GAAE,OAAO,CAAC,iBAAiB,CAAM,EAEvD;IAED;;;OAGG;IACH,YAAY,CAAC,SAAS,EAAE,UAAU,EAAE,GAAG,IAAI,CAG1C;IAED;;;OAGG;IACH,WAAW,CAAC,QAAQ,EAAE,UAAU,GAAG,SAAS,GAAG,IAAI,CAGlD;IAED;;;OAGG;IACH,cAAc,CAAC,IAAI,EAAE,UAAU,GAAG,IAAI,CAGrC;IAED;;;OAGG;IACH,kBAAkB,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI,CAGtC;IAED;;;OAGG;IACH,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI,CAG1B;IAED;;;OAGG;IACH,uBAAuB,CAAC,UAAU,EAAE,UAAU,EAAE,GAAG,IAAI,CAGtD;IAED;;;OAGG;IACH,cAAc,CAAC,SAAS,EAAE,OAAO,CAAC,iBAAiB,CAAC,GAAG,IAAI,CAG1D;IAED,cAAc,IAAI,iBAAiB,CAElC;IAED,YAAY,CAAC,KAAK,CAAC,EAAE,OAAO,GAAG,OAAO,CAAC,kBAAkB,CAAC,CAQzD;IAED,kBAAkB,IAAI,YAAY,GAAG;QAAE,KAAK,EAAE,MAAM,CAAA;KAAE,CAIrD;IAED,2BAA2B,IAAI,YAAY,GAAG;QAAE,GAAG,EAAE,MAAM,CAAA;KAAE,CAO5D;IAED,wBAAwB,CAAC,KAAK,EAAE,WAAW,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,GAAG,KAAK,MAAM,EAAE,CAG1F;IAED,oBAAoB,CAAC,IAAI,EAAE,UAAU,EAAE,MAAM,EAAE,WAAW,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,MAAM,CAK/F;IAED,qBAAqB,IAAI;QAAE,WAAW,EAAE,UAAU,CAAC;QAAC,QAAQ,EAAE,UAAU,CAAA;KAAE,CAKzE;IAED,gCAAgC,CAAC,KAAK,EAAE,UAAU,GAAG,OAAO,CAAC,UAAU,GAAG,SAAS,CAAC,CAEnF;IAED,uBAAuB,IAAI,OAAO,CAAC,UAAU,EAAE,CAAC,CAE/C;IAED,aAAa,CAAC,KAAK,EAAE,OAAO,EAAE,SAAS,EAAE,UAAU,GAAG,OAAO,CAAC,OAAO,CAAC,CAErE;IAED,iBAAiB,CAAC,KAAK,EAAE,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC,CAGjF;IAED,uBAAuB,CAAC,KAAK,CAAC,EAAE,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,CAEzD;CACF"}
1
+ {"version":3,"file":"test_epoch_cache.d.ts","sourceRoot":"","sources":["../../src/test/test_epoch_cache.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,iCAAiC,CAAC;AAC1E,OAAO,EAAE,UAAU,EAAE,MAAM,+BAA+B,CAAC;AAC3D,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,6BAA6B,CAAC;AAQrE,OAAO,EACL,KAAK,YAAY,EACjB,KAAK,mBAAmB,EACxB,KAAK,kBAAkB,EAEvB,KAAK,OAAO,EACb,MAAM,mBAAmB,CAAC;AAc3B;;;;;;GAMG;AACH,qBAAa,cAAe,YAAW,mBAAmB;IACxD,OAAO,CAAC,SAAS,CAAoB;IACrC,OAAO,CAAC,eAAe,CAAyB;IAChD,OAAO,CAAC,WAAW,CAA6B;IAChD,OAAO,CAAC,eAAe,CAAkB;IACzC,OAAO,CAAC,IAAI,CAAc;IAC1B,OAAO,CAAC,oBAAoB,CAAoB;IAChD,OAAO,CAAC,WAAW,CAAoB;IAEvC,YAAY,WAAW,GAAE,OAAO,CAAC,iBAAiB,CAAM,EAEvD;IAED;;;OAGG;IACH,YAAY,CAAC,SAAS,EAAE,UAAU,EAAE,GAAG,IAAI,CAG1C;IAED;;;OAGG;IACH,WAAW,CAAC,QAAQ,EAAE,UAAU,GAAG,SAAS,GAAG,IAAI,CAGlD;IAED;;;OAGG;IACH,cAAc,CAAC,IAAI,EAAE,UAAU,GAAG,IAAI,CAGrC;IAED;;;OAGG;IACH,kBAAkB,CAAC,IAAI,EAAE,OAAO,GAAG,IAAI,CAGtC;IAED;;;OAGG;IACH,OAAO,CAAC,IAAI,EAAE,MAAM,GAAG,IAAI,CAG1B;IAED;;;OAGG;IACH,uBAAuB,CAAC,UAAU,EAAE,UAAU,EAAE,GAAG,IAAI,CAGtD;IAED;;;OAGG;IACH,cAAc,CAAC,SAAS,EAAE,OAAO,CAAC,iBAAiB,CAAC,GAAG,IAAI,CAG1D;IAED,cAAc,IAAI,iBAAiB,CAElC;IAED,YAAY,CAAC,KAAK,CAAC,EAAE,OAAO,GAAG,OAAO,CAAC,kBAAkB,CAAC,CAQzD;IAED,UAAU,IAAI,UAAU,CAEvB;IAED,aAAa,IAAI,UAAU,CAE1B;IAED,WAAW,IAAI,WAAW,CAEzB;IAED,cAAc,IAAI,WAAW,CAE5B;IAED,kBAAkB,IAAI,YAAY,GAAG;QAAE,KAAK,EAAE,MAAM,CAAA;KAAE,CAYrD;IAED,2BAA2B,IAAI,YAAY,GAAG;QAAE,UAAU,EAAE,MAAM,CAAA;KAAE,CAYnE;IAED,iCAAiC,IAAI,YAAY,GAAG;QAAE,UAAU,EAAE,MAAM,CAAA;KAAE,CAKzE;IAED,wBAAwB,CAAC,KAAK,EAAE,WAAW,EAAE,IAAI,EAAE,UAAU,EAAE,IAAI,EAAE,MAAM,GAAG,KAAK,MAAM,EAAE,CAG1F;IAED,oBAAoB,CAAC,IAAI,EAAE,UAAU,EAAE,MAAM,EAAE,WAAW,EAAE,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,GAAG,MAAM,CAK/F;IAED,qBAAqB,IAAI;QAAE,WAAW,EAAE,UAAU,CAAC;QAAC,QAAQ,EAAE,UAAU,CAAA;KAAE,CAQzE;IAED,oBAAoB,IAAI;QAAE,UAAU,EAAE,UAAU,CAAC;QAAC,QAAQ,EAAE,UAAU,CAAA;KAAE,CAQvE;IAED,gCAAgC,CAAC,KAAK,EAAE,UAAU,GAAG,OAAO,CAAC,UAAU,GAAG,SAAS,CAAC,CAEnF;IAED,uBAAuB,IAAI,OAAO,CAAC,UAAU,EAAE,CAAC,CAE/C;IAED,aAAa,CAAC,KAAK,EAAE,OAAO,EAAE,SAAS,EAAE,UAAU,GAAG,OAAO,CAAC,OAAO,CAAC,CAErE;IAED,iBAAiB,CAAC,KAAK,EAAE,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC,CAGjF;IAED,iBAAiB,CAAC,MAAM,EAAE,WAAW,GAAG,OAAO,CAAC,OAAO,CAAC,CAEvD;IAED,uBAAuB,CAAC,KAAK,CAAC,EAAE,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,CAEzD;CACF"}
@@ -1,5 +1,6 @@
1
1
  import { SlotNumber } from '@aztec/foundation/branded-types';
2
- import { getEpochAtSlot, getSlotAtTimestamp, getTimestampRangeForEpoch } from '@aztec/stdlib/epoch-helpers';
2
+ import { getEpochAtSlot, getSlotAtTimestamp, getTimestampForSlot, getTimestampRangeForEpoch } from '@aztec/stdlib/epoch-helpers';
3
+ import { PROPOSER_PIPELINING_SLOT_OFFSET } from '../epoch_cache.js';
3
4
  /** Default L1 constants for testing. */ const DEFAULT_L1_CONSTANTS = {
4
5
  l1StartBlock: 0n,
5
6
  l1GenesisTime: 0n,
@@ -94,27 +95,52 @@ import { getEpochAtSlot, getSlotAtTimestamp, getTimestampRangeForEpoch } from '@
94
95
  isEscapeHatchOpen: this.escapeHatchOpen
95
96
  });
96
97
  }
98
+ getSlotNow() {
99
+ return this.currentSlot;
100
+ }
101
+ getTargetSlot() {
102
+ return SlotNumber(this.currentSlot + PROPOSER_PIPELINING_SLOT_OFFSET);
103
+ }
104
+ getEpochNow() {
105
+ return getEpochAtSlot(this.currentSlot, this.l1Constants);
106
+ }
107
+ getTargetEpoch() {
108
+ return getEpochAtSlot(this.getTargetSlot(), this.l1Constants);
109
+ }
97
110
  getEpochAndSlotNow() {
98
- const epoch = getEpochAtSlot(this.currentSlot, this.l1Constants);
99
- const ts = getTimestampRangeForEpoch(epoch, this.l1Constants)[0];
111
+ // Model "now" as the start of the current slot (mirroring the real EpochCache, which derives nowMs
112
+ // from the wall clock). Using the slot start rather than the epoch start keeps nowMs consistent with
113
+ // currentSlot, which the pipelining receive-window check (clock_tolerance) relies on.
114
+ const epochNow = getEpochAtSlot(this.currentSlot, this.l1Constants);
115
+ const ts = getTimestampForSlot(this.currentSlot, this.l1Constants);
100
116
  return {
101
- epoch,
117
+ epoch: epochNow,
102
118
  slot: this.currentSlot,
103
119
  ts,
104
120
  nowMs: ts * 1000n
105
121
  };
106
122
  }
107
123
  getEpochAndSlotInNextL1Slot() {
108
- const now = getTimestampRangeForEpoch(getEpochAtSlot(this.currentSlot, this.l1Constants), this.l1Constants)[0];
109
- const nextSlotTs = now + BigInt(this.l1Constants.ethereumSlotDuration);
124
+ const nowTs = getTimestampRangeForEpoch(getEpochAtSlot(this.currentSlot, this.l1Constants), this.l1Constants)[0];
125
+ const nextSlotTs = nowTs + BigInt(this.l1Constants.ethereumSlotDuration);
110
126
  const nextSlot = getSlotAtTimestamp(nextSlotTs, this.l1Constants);
111
- const epoch = getEpochAtSlot(nextSlot, this.l1Constants);
112
- const ts = getTimestampRangeForEpoch(epoch, this.l1Constants)[0];
127
+ const epochNow = getEpochAtSlot(nextSlot, this.l1Constants);
128
+ const ts = getTimestampRangeForEpoch(epochNow, this.l1Constants)[0];
113
129
  return {
114
- epoch,
130
+ epoch: epochNow,
115
131
  slot: nextSlot,
116
132
  ts,
117
- now
133
+ nowSeconds: nowTs
134
+ };
135
+ }
136
+ getTargetEpochAndSlotInNextL1Slot() {
137
+ const result = this.getEpochAndSlotInNextL1Slot();
138
+ const offset = PROPOSER_PIPELINING_SLOT_OFFSET;
139
+ const targetSlot = SlotNumber(result.slot + offset);
140
+ return {
141
+ ...result,
142
+ slot: targetSlot,
143
+ epoch: getEpochAtSlot(targetSlot, this.l1Constants)
118
144
  };
119
145
  }
120
146
  getProposerIndexEncoding(epoch, slot, seed) {
@@ -128,9 +154,19 @@ import { getEpochAtSlot, getSlotAtTimestamp, getTimestampRangeForEpoch } from '@
128
154
  return BigInt(slot) % size;
129
155
  }
130
156
  getCurrentAndNextSlot() {
157
+ const currentSlot = this.getSlotNow();
158
+ const next = this.getEpochAndSlotInNextL1Slot();
131
159
  return {
132
- currentSlot: this.currentSlot,
133
- nextSlot: SlotNumber(this.currentSlot + 1)
160
+ currentSlot,
161
+ nextSlot: next.slot
162
+ };
163
+ }
164
+ getTargetAndNextSlot() {
165
+ const targetSlot = this.getTargetSlot();
166
+ const next = this.getTargetEpochAndSlotInNextL1Slot();
167
+ return {
168
+ targetSlot,
169
+ nextSlot: next.slot
134
170
  };
135
171
  }
136
172
  getProposerAttesterAddressInSlot(_slot) {
@@ -146,6 +182,9 @@ import { getEpochAtSlot, getSlotAtTimestamp, getTimestampRangeForEpoch } from '@
146
182
  const committeeSet = new Set(this.committee.map((v)=>v.toString()));
147
183
  return Promise.resolve(validators.filter((v)=>committeeSet.has(v.toString())));
148
184
  }
185
+ isEscapeHatchOpen(_epoch) {
186
+ return Promise.resolve(this.escapeHatchOpen);
187
+ }
149
188
  isEscapeHatchOpenAtSlot(_slot) {
150
189
  return Promise.resolve(this.escapeHatchOpen);
151
190
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@aztec/epoch-cache",
3
- "version": "5.0.0-private.20260318",
3
+ "version": "5.0.0-rc.1",
4
4
  "type": "module",
5
5
  "exports": {
6
6
  ".": "./dest/index.js",
@@ -26,16 +26,16 @@
26
26
  "../package.common.json"
27
27
  ],
28
28
  "dependencies": {
29
- "@aztec/ethereum": "5.0.0-private.20260318",
30
- "@aztec/foundation": "5.0.0-private.20260318",
31
- "@aztec/l1-artifacts": "5.0.0-private.20260318",
32
- "@aztec/stdlib": "5.0.0-private.20260318",
29
+ "@aztec/ethereum": "5.0.0-rc.1",
30
+ "@aztec/foundation": "5.0.0-rc.1",
31
+ "@aztec/l1-artifacts": "5.0.0-rc.1",
32
+ "@aztec/stdlib": "5.0.0-rc.1",
33
33
  "dotenv": "^16.0.3",
34
34
  "get-port": "^7.1.0",
35
35
  "jest-mock-extended": "^4.0.0",
36
36
  "tslib": "^2.4.0",
37
37
  "viem": "npm:@aztec/viem@2.38.2",
38
- "zod": "^3.23.8"
38
+ "zod": "^4"
39
39
  },
40
40
  "devDependencies": {
41
41
  "@jest/globals": "^30.0.0",
package/src/config.ts CHANGED
@@ -3,7 +3,7 @@ import { type L1ReaderConfig, getL1ReaderConfigFromEnv } from '@aztec/ethereum/l
3
3
 
4
4
  export type EpochCacheConfig = Pick<
5
5
  L1ReaderConfig & L1ContractsConfig,
6
- 'l1RpcUrls' | 'l1ChainId' | 'viemPollingIntervalMS' | 'l1HttpTimeoutMS' | 'ethereumSlotDuration'
6
+ 'l1RpcUrls' | 'l1ChainId' | 'viemPollingIntervalMS' | 'ethereumSlotDuration' | 'l1HttpTimeoutMS'
7
7
  >;
8
8
 
9
9
  export function getEpochCacheConfigEnvVars(): EpochCacheConfig {