@aztec/archiver 0.0.1-commit.fcb71a6 → 0.0.1-commit.ff7989d6c

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.
Files changed (200) hide show
  1. package/README.md +156 -22
  2. package/dest/archiver.d.ts +139 -0
  3. package/dest/archiver.d.ts.map +1 -0
  4. package/dest/archiver.js +699 -0
  5. package/dest/{archiver/config.d.ts → config.d.ts} +9 -1
  6. package/dest/config.d.ts.map +1 -0
  7. package/dest/{archiver/config.js → config.js} +11 -2
  8. package/dest/errors.d.ts +41 -0
  9. package/dest/errors.d.ts.map +1 -0
  10. package/dest/{archiver/errors.js → errors.js} +8 -0
  11. package/dest/factory.d.ts +9 -7
  12. package/dest/factory.d.ts.map +1 -1
  13. package/dest/factory.js +94 -11
  14. package/dest/index.d.ts +11 -4
  15. package/dest/index.d.ts.map +1 -1
  16. package/dest/index.js +9 -3
  17. package/dest/interfaces.d.ts +9 -0
  18. package/dest/interfaces.d.ts.map +1 -0
  19. package/dest/interfaces.js +3 -0
  20. package/dest/{archiver/l1 → l1}/bin/retrieve-calldata.d.ts +1 -1
  21. package/dest/l1/bin/retrieve-calldata.d.ts.map +1 -0
  22. package/dest/{archiver/l1 → l1}/bin/retrieve-calldata.js +17 -18
  23. package/dest/{archiver/l1 → l1}/calldata_retriever.d.ts +9 -3
  24. package/dest/l1/calldata_retriever.d.ts.map +1 -0
  25. package/dest/{archiver/l1 → l1}/calldata_retriever.js +19 -6
  26. package/dest/l1/data_retrieval.d.ts +89 -0
  27. package/dest/l1/data_retrieval.d.ts.map +1 -0
  28. package/dest/{archiver/l1 → l1}/data_retrieval.js +39 -57
  29. package/dest/{archiver/l1 → l1}/debug_tx.d.ts +1 -1
  30. package/dest/l1/debug_tx.d.ts.map +1 -0
  31. package/dest/{archiver/l1 → l1}/spire_proposer.d.ts +1 -1
  32. package/dest/l1/spire_proposer.d.ts.map +1 -0
  33. package/dest/{archiver/l1 → l1}/trace_tx.d.ts +1 -1
  34. package/dest/l1/trace_tx.d.ts.map +1 -0
  35. package/dest/l1/types.d.ts +12 -0
  36. package/dest/l1/types.d.ts.map +1 -0
  37. package/dest/{archiver/l1 → l1}/validate_trace.d.ts +6 -3
  38. package/dest/l1/validate_trace.d.ts.map +1 -0
  39. package/dest/{archiver/l1 → l1}/validate_trace.js +14 -10
  40. package/dest/modules/data_source_base.d.ts +89 -0
  41. package/dest/modules/data_source_base.d.ts.map +1 -0
  42. package/dest/modules/data_source_base.js +216 -0
  43. package/dest/modules/data_store_updater.d.ts +80 -0
  44. package/dest/modules/data_store_updater.d.ts.map +1 -0
  45. package/dest/modules/data_store_updater.js +323 -0
  46. package/dest/modules/instrumentation.d.ts +39 -0
  47. package/dest/modules/instrumentation.d.ts.map +1 -0
  48. package/dest/{archiver → modules}/instrumentation.js +33 -67
  49. package/dest/modules/l1_synchronizer.d.ts +76 -0
  50. package/dest/modules/l1_synchronizer.d.ts.map +1 -0
  51. package/dest/modules/l1_synchronizer.js +1112 -0
  52. package/dest/modules/validation.d.ts +17 -0
  53. package/dest/modules/validation.d.ts.map +1 -0
  54. package/dest/{archiver → modules}/validation.js +7 -1
  55. package/dest/store/block_store.d.ts +196 -0
  56. package/dest/store/block_store.d.ts.map +1 -0
  57. package/dest/{archiver/kv_archiver_store → store}/block_store.js +228 -62
  58. package/dest/store/contract_class_store.d.ts +18 -0
  59. package/dest/store/contract_class_store.d.ts.map +1 -0
  60. package/dest/{archiver/kv_archiver_store → store}/contract_class_store.js +12 -8
  61. package/dest/store/contract_instance_store.d.ts +24 -0
  62. package/dest/store/contract_instance_store.d.ts.map +1 -0
  63. package/dest/{archiver/kv_archiver_store → store}/contract_instance_store.js +1 -1
  64. package/dest/store/kv_archiver_store.d.ts +354 -0
  65. package/dest/store/kv_archiver_store.d.ts.map +1 -0
  66. package/dest/store/kv_archiver_store.js +464 -0
  67. package/dest/store/l2_tips_cache.d.ts +19 -0
  68. package/dest/store/l2_tips_cache.d.ts.map +1 -0
  69. package/dest/store/l2_tips_cache.js +89 -0
  70. package/dest/store/log_store.d.ts +54 -0
  71. package/dest/store/log_store.d.ts.map +1 -0
  72. package/dest/{archiver/kv_archiver_store → store}/log_store.js +146 -91
  73. package/dest/{archiver/kv_archiver_store → store}/message_store.d.ts +1 -1
  74. package/dest/store/message_store.d.ts.map +1 -0
  75. package/dest/{archiver/structs → structs}/data_retrieval.d.ts +1 -1
  76. package/dest/structs/data_retrieval.d.ts.map +1 -0
  77. package/dest/structs/inbox_message.d.ts +15 -0
  78. package/dest/structs/inbox_message.d.ts.map +1 -0
  79. package/dest/{archiver/structs → structs}/published.d.ts +1 -1
  80. package/dest/structs/published.d.ts.map +1 -0
  81. package/dest/test/fake_l1_state.d.ts +193 -0
  82. package/dest/test/fake_l1_state.d.ts.map +1 -0
  83. package/dest/test/fake_l1_state.js +389 -0
  84. package/dest/test/index.d.ts +2 -1
  85. package/dest/test/index.d.ts.map +1 -1
  86. package/dest/test/index.js +4 -1
  87. package/dest/test/mock_archiver.d.ts +2 -2
  88. package/dest/test/mock_archiver.d.ts.map +1 -1
  89. package/dest/test/mock_archiver.js +3 -3
  90. package/dest/test/mock_l1_to_l2_message_source.d.ts +2 -2
  91. package/dest/test/mock_l1_to_l2_message_source.d.ts.map +1 -1
  92. package/dest/test/mock_l1_to_l2_message_source.js +12 -3
  93. package/dest/test/mock_l2_block_source.d.ts +39 -17
  94. package/dest/test/mock_l2_block_source.d.ts.map +1 -1
  95. package/dest/test/mock_l2_block_source.js +232 -83
  96. package/dest/test/mock_structs.d.ts +78 -3
  97. package/dest/test/mock_structs.d.ts.map +1 -1
  98. package/dest/test/mock_structs.js +140 -7
  99. package/dest/test/noop_l1_archiver.d.ts +23 -0
  100. package/dest/test/noop_l1_archiver.d.ts.map +1 -0
  101. package/dest/test/noop_l1_archiver.js +68 -0
  102. package/package.json +16 -17
  103. package/src/archiver.ts +443 -0
  104. package/src/{archiver/config.ts → config.ts} +13 -2
  105. package/src/{archiver/errors.ts → errors.ts} +12 -0
  106. package/src/factory.ts +140 -11
  107. package/src/index.ts +11 -3
  108. package/src/interfaces.ts +9 -0
  109. package/src/{archiver/l1 → l1}/bin/retrieve-calldata.ts +16 -17
  110. package/src/{archiver/l1 → l1}/calldata_retriever.ts +28 -6
  111. package/src/{archiver/l1 → l1}/data_retrieval.ts +60 -74
  112. package/src/{archiver/l1 → l1}/validate_trace.ts +25 -7
  113. package/src/modules/data_source_base.ts +328 -0
  114. package/src/modules/data_store_updater.ts +448 -0
  115. package/src/{archiver → modules}/instrumentation.ts +33 -70
  116. package/src/modules/l1_synchronizer.ts +932 -0
  117. package/src/{archiver → modules}/validation.ts +11 -6
  118. package/src/{archiver/kv_archiver_store → store}/block_store.ts +293 -100
  119. package/src/{archiver/kv_archiver_store → store}/contract_class_store.ts +12 -8
  120. package/src/{archiver/kv_archiver_store → store}/contract_instance_store.ts +1 -1
  121. package/src/{archiver/kv_archiver_store → store}/kv_archiver_store.ts +273 -40
  122. package/src/store/l2_tips_cache.ts +89 -0
  123. package/src/{archiver/kv_archiver_store → store}/log_store.ts +242 -121
  124. package/src/test/fake_l1_state.ts +607 -0
  125. package/src/test/index.ts +4 -0
  126. package/src/test/mock_archiver.ts +4 -3
  127. package/src/test/mock_l1_to_l2_message_source.ts +10 -4
  128. package/src/test/mock_l2_block_source.ts +276 -90
  129. package/src/test/mock_structs.ts +269 -8
  130. package/src/test/noop_l1_archiver.ts +109 -0
  131. package/dest/archiver/archiver.d.ts +0 -304
  132. package/dest/archiver/archiver.d.ts.map +0 -1
  133. package/dest/archiver/archiver.js +0 -1645
  134. package/dest/archiver/archiver_store.d.ts +0 -308
  135. package/dest/archiver/archiver_store.d.ts.map +0 -1
  136. package/dest/archiver/archiver_store.js +0 -4
  137. package/dest/archiver/archiver_store_test_suite.d.ts +0 -8
  138. package/dest/archiver/archiver_store_test_suite.d.ts.map +0 -1
  139. package/dest/archiver/archiver_store_test_suite.js +0 -2770
  140. package/dest/archiver/config.d.ts.map +0 -1
  141. package/dest/archiver/errors.d.ts +0 -36
  142. package/dest/archiver/errors.d.ts.map +0 -1
  143. package/dest/archiver/index.d.ts +0 -7
  144. package/dest/archiver/index.d.ts.map +0 -1
  145. package/dest/archiver/index.js +0 -4
  146. package/dest/archiver/instrumentation.d.ts +0 -37
  147. package/dest/archiver/instrumentation.d.ts.map +0 -1
  148. package/dest/archiver/kv_archiver_store/block_store.d.ts +0 -157
  149. package/dest/archiver/kv_archiver_store/block_store.d.ts.map +0 -1
  150. package/dest/archiver/kv_archiver_store/contract_class_store.d.ts +0 -18
  151. package/dest/archiver/kv_archiver_store/contract_class_store.d.ts.map +0 -1
  152. package/dest/archiver/kv_archiver_store/contract_instance_store.d.ts +0 -24
  153. package/dest/archiver/kv_archiver_store/contract_instance_store.d.ts.map +0 -1
  154. package/dest/archiver/kv_archiver_store/kv_archiver_store.d.ts +0 -158
  155. package/dest/archiver/kv_archiver_store/kv_archiver_store.d.ts.map +0 -1
  156. package/dest/archiver/kv_archiver_store/kv_archiver_store.js +0 -313
  157. package/dest/archiver/kv_archiver_store/log_store.d.ts +0 -45
  158. package/dest/archiver/kv_archiver_store/log_store.d.ts.map +0 -1
  159. package/dest/archiver/kv_archiver_store/message_store.d.ts.map +0 -1
  160. package/dest/archiver/l1/bin/retrieve-calldata.d.ts.map +0 -1
  161. package/dest/archiver/l1/calldata_retriever.d.ts.map +0 -1
  162. package/dest/archiver/l1/data_retrieval.d.ts +0 -90
  163. package/dest/archiver/l1/data_retrieval.d.ts.map +0 -1
  164. package/dest/archiver/l1/debug_tx.d.ts.map +0 -1
  165. package/dest/archiver/l1/spire_proposer.d.ts.map +0 -1
  166. package/dest/archiver/l1/trace_tx.d.ts.map +0 -1
  167. package/dest/archiver/l1/types.d.ts +0 -12
  168. package/dest/archiver/l1/types.d.ts.map +0 -1
  169. package/dest/archiver/l1/validate_trace.d.ts.map +0 -1
  170. package/dest/archiver/structs/data_retrieval.d.ts.map +0 -1
  171. package/dest/archiver/structs/inbox_message.d.ts +0 -15
  172. package/dest/archiver/structs/inbox_message.d.ts.map +0 -1
  173. package/dest/archiver/structs/published.d.ts.map +0 -1
  174. package/dest/archiver/validation.d.ts +0 -17
  175. package/dest/archiver/validation.d.ts.map +0 -1
  176. package/dest/rpc/index.d.ts +0 -9
  177. package/dest/rpc/index.d.ts.map +0 -1
  178. package/dest/rpc/index.js +0 -15
  179. package/src/archiver/archiver.ts +0 -2157
  180. package/src/archiver/archiver_store.ts +0 -372
  181. package/src/archiver/archiver_store_test_suite.ts +0 -2843
  182. package/src/archiver/index.ts +0 -6
  183. package/src/rpc/index.ts +0 -16
  184. /package/dest/{archiver/l1 → l1}/debug_tx.js +0 -0
  185. /package/dest/{archiver/l1 → l1}/spire_proposer.js +0 -0
  186. /package/dest/{archiver/l1 → l1}/trace_tx.js +0 -0
  187. /package/dest/{archiver/l1 → l1}/types.js +0 -0
  188. /package/dest/{archiver/kv_archiver_store → store}/message_store.js +0 -0
  189. /package/dest/{archiver/structs → structs}/data_retrieval.js +0 -0
  190. /package/dest/{archiver/structs → structs}/inbox_message.js +0 -0
  191. /package/dest/{archiver/structs → structs}/published.js +0 -0
  192. /package/src/{archiver/l1 → l1}/README.md +0 -0
  193. /package/src/{archiver/l1 → l1}/debug_tx.ts +0 -0
  194. /package/src/{archiver/l1 → l1}/spire_proposer.ts +0 -0
  195. /package/src/{archiver/l1 → l1}/trace_tx.ts +0 -0
  196. /package/src/{archiver/l1 → l1}/types.ts +0 -0
  197. /package/src/{archiver/kv_archiver_store → store}/message_store.ts +0 -0
  198. /package/src/{archiver/structs → structs}/data_retrieval.ts +0 -0
  199. /package/src/{archiver/structs → structs}/inbox_message.ts +0 -0
  200. /package/src/{archiver/structs → structs}/published.ts +0 -0
@@ -8,18 +8,20 @@ import { createLogger } from '@aztec/foundation/log';
8
8
  import type { FunctionSelector } from '@aztec/stdlib/abi';
9
9
  import type { AztecAddress } from '@aztec/stdlib/aztec-address';
10
10
  import {
11
+ type BlockData,
12
+ BlockHash,
13
+ CheckpointedL2Block,
11
14
  L2Block,
12
- L2BlockHash,
13
- L2BlockNew,
14
15
  type L2BlockSource,
15
16
  type L2Tips,
16
- PublishedL2Block,
17
- type ValidateBlockResult,
17
+ type ValidateCheckpointResult,
18
18
  } from '@aztec/stdlib/block';
19
- import { type Checkpoint, L1PublishedData } from '@aztec/stdlib/checkpoint';
19
+ import { Checkpoint, type CheckpointData, L1PublishedData, PublishedCheckpoint } from '@aztec/stdlib/checkpoint';
20
20
  import type { ContractClassPublic, ContractDataSource, ContractInstanceWithAddress } from '@aztec/stdlib/contract';
21
21
  import { EmptyL1RollupConstants, type L1RollupConstants, getSlotRangeForEpoch } from '@aztec/stdlib/epoch-helpers';
22
- import { type BlockHeader, TxHash, TxReceipt, TxStatus } from '@aztec/stdlib/tx';
22
+ import { computeCheckpointOutHash } from '@aztec/stdlib/messaging';
23
+ import { CheckpointHeader } from '@aztec/stdlib/rollup';
24
+ import { type BlockHeader, TxExecutionResult, TxHash, TxReceipt, TxStatus } from '@aztec/stdlib/tx';
23
25
  import type { UInt64 } from '@aztec/stdlib/types';
24
26
 
25
27
  /**
@@ -27,29 +29,57 @@ import type { UInt64 } from '@aztec/stdlib/types';
27
29
  */
28
30
  export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
29
31
  protected l2Blocks: L2Block[] = [];
32
+ protected checkpointList: Checkpoint[] = [];
30
33
 
31
34
  private provenBlockNumber: number = 0;
32
35
  private finalizedBlockNumber: number = 0;
36
+ private checkpointedBlockNumber: number = 0;
33
37
 
34
38
  private log = createLogger('archiver:mock_l2_block_source');
35
39
 
40
+ /** Creates blocks grouped into single-block checkpoints. */
36
41
  public async createBlocks(numBlocks: number) {
37
- for (let i = 0; i < numBlocks; i++) {
38
- const blockNum = this.l2Blocks.length + 1;
39
- const block = await L2Block.random(BlockNumber(blockNum));
40
- this.l2Blocks.push(block);
42
+ await this.createCheckpoints(numBlocks, 1);
43
+ }
44
+
45
+ /** Creates checkpoints, each containing `blocksPerCheckpoint` blocks. */
46
+ public async createCheckpoints(numCheckpoints: number, blocksPerCheckpoint: number = 1) {
47
+ for (let c = 0; c < numCheckpoints; c++) {
48
+ const checkpointNum = CheckpointNumber(this.checkpointList.length + 1);
49
+ const startBlockNum = this.l2Blocks.length + 1;
50
+ const slotNumber = SlotNumber(Number(checkpointNum));
51
+ const checkpoint = await Checkpoint.random(checkpointNum, {
52
+ numBlocks: blocksPerCheckpoint,
53
+ startBlockNumber: startBlockNum,
54
+ slotNumber,
55
+ checkpointNumber: checkpointNum,
56
+ });
57
+ this.checkpointList.push(checkpoint);
58
+ this.l2Blocks.push(...checkpoint.blocks);
41
59
  }
42
60
 
43
- this.log.verbose(`Created ${numBlocks} blocks in the mock L2 block source`);
61
+ this.log.verbose(
62
+ `Created ${numCheckpoints} checkpoints with ${blocksPerCheckpoint} blocks each in the mock L2 block source`,
63
+ );
44
64
  }
45
65
 
46
- public addBlocks(blocks: L2Block[]) {
66
+ public addProposedBlocks(blocks: L2Block[]) {
47
67
  this.l2Blocks.push(...blocks);
48
- this.log.verbose(`Added ${blocks.length} blocks to the mock L2 block source`);
68
+ this.log.verbose(`Added ${blocks.length} proposed blocks to the mock L2 block source`);
49
69
  }
50
70
 
51
71
  public removeBlocks(numBlocks: number) {
52
72
  this.l2Blocks = this.l2Blocks.slice(0, -numBlocks);
73
+ const maxBlockNum = this.l2Blocks.length;
74
+ // Remove any checkpoint whose last block is beyond the remaining blocks.
75
+ this.checkpointList = this.checkpointList.filter(c => {
76
+ const lastBlockNum = c.blocks[0].number + c.blocks.length - 1;
77
+ return lastBlockNum <= maxBlockNum;
78
+ });
79
+ // Keep tip numbers consistent with remaining blocks.
80
+ this.checkpointedBlockNumber = Math.min(this.checkpointedBlockNumber, maxBlockNum);
81
+ this.provenBlockNumber = Math.min(this.provenBlockNumber, maxBlockNum);
82
+ this.finalizedBlockNumber = Math.min(this.finalizedBlockNumber, maxBlockNum);
53
83
  this.log.verbose(`Removed ${numBlocks} blocks from the mock L2 block source`);
54
84
  }
55
85
 
@@ -64,6 +94,36 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
64
94
  this.finalizedBlockNumber = finalizedBlockNumber;
65
95
  }
66
96
 
97
+ public setCheckpointedBlockNumber(checkpointedBlockNumber: number) {
98
+ const prevCheckpointed = this.checkpointedBlockNumber;
99
+ this.checkpointedBlockNumber = checkpointedBlockNumber;
100
+ // Auto-create single-block checkpoints for newly checkpointed blocks that don't have one yet.
101
+ // This handles blocks added via addProposedBlocks that are now being marked as checkpointed.
102
+ const newCheckpoints: Checkpoint[] = [];
103
+ for (let blockNum = prevCheckpointed + 1; blockNum <= checkpointedBlockNumber; blockNum++) {
104
+ const block = this.l2Blocks[blockNum - 1];
105
+ if (!block) {
106
+ continue;
107
+ }
108
+ if (this.checkpointList.some(c => c.blocks.some(b => b.number === block.number))) {
109
+ continue;
110
+ }
111
+ const checkpointNum = CheckpointNumber(this.checkpointList.length + newCheckpoints.length + 1);
112
+ const checkpoint = new Checkpoint(
113
+ block.archive,
114
+ CheckpointHeader.random({ slotNumber: block.header.globalVariables.slotNumber }),
115
+ [block],
116
+ checkpointNum,
117
+ );
118
+ newCheckpoints.push(checkpoint);
119
+ }
120
+ // Insert new checkpoints in order by number.
121
+ if (newCheckpoints.length > 0) {
122
+ this.checkpointList.push(...newCheckpoints);
123
+ this.checkpointList.sort((a, b) => a.number - b.number);
124
+ }
125
+ }
126
+
67
127
  /**
68
128
  * Method to fetch the rollup contract address at the base-layer.
69
129
  * @returns The rollup address.
@@ -92,9 +152,38 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
92
152
  return Promise.resolve(BlockNumber(this.provenBlockNumber));
93
153
  }
94
154
 
95
- public getCheckpointedBlock(_number: BlockNumber) {
96
- // In this mock, we don't track checkpointed blocks separately
97
- return Promise.resolve(undefined);
155
+ public getCheckpointedL2BlockNumber() {
156
+ return Promise.resolve(BlockNumber(this.checkpointedBlockNumber));
157
+ }
158
+
159
+ public getFinalizedL2BlockNumber() {
160
+ return Promise.resolve(BlockNumber(this.finalizedBlockNumber));
161
+ }
162
+
163
+ public getCheckpointedBlock(number: BlockNumber): Promise<CheckpointedL2Block | undefined> {
164
+ if (number > this.checkpointedBlockNumber) {
165
+ return Promise.resolve(undefined);
166
+ }
167
+ const block = this.l2Blocks[number - 1];
168
+ if (!block) {
169
+ return Promise.resolve(undefined);
170
+ }
171
+ return Promise.resolve(this.toCheckpointedBlock(block));
172
+ }
173
+
174
+ public async getCheckpointedBlocks(from: BlockNumber, limit: number): Promise<CheckpointedL2Block[]> {
175
+ const result: CheckpointedL2Block[] = [];
176
+ for (let i = 0; i < limit; i++) {
177
+ const blockNum = from + i;
178
+ if (blockNum > this.checkpointedBlockNumber) {
179
+ break;
180
+ }
181
+ const block = await this.getCheckpointedBlock(BlockNumber(blockNum));
182
+ if (block) {
183
+ result.push(block);
184
+ }
185
+ }
186
+ return result;
98
187
  }
99
188
 
100
189
  /**
@@ -102,8 +191,9 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
102
191
  * @param number - The block number to return (inclusive).
103
192
  * @returns The requested L2 block.
104
193
  */
105
- public getBlock(number: number) {
106
- return Promise.resolve(this.l2Blocks[number - 1]);
194
+ public getBlock(number: number): Promise<L2Block | undefined> {
195
+ const block = this.l2Blocks[number - 1];
196
+ return Promise.resolve(block);
107
197
  }
108
198
 
109
199
  /**
@@ -111,9 +201,9 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
111
201
  * @param number - The block number to return.
112
202
  * @returns The requested L2 block.
113
203
  */
114
- public getL2BlockNew(number: BlockNumber): Promise<L2BlockNew | undefined> {
204
+ public getL2Block(number: BlockNumber): Promise<L2Block | undefined> {
115
205
  const block = this.l2Blocks[number - 1];
116
- return Promise.resolve(block?.toL2Block());
206
+ return Promise.resolve(block);
117
207
  }
118
208
 
119
209
  /**
@@ -122,68 +212,60 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
122
212
  * @param limit - The maximum number of blocks to return.
123
213
  * @returns The requested mocked L2 blocks.
124
214
  */
125
- public getBlocks(from: number, limit: number, proven?: boolean) {
215
+ public getBlocks(from: number, limit: number): Promise<L2Block[]> {
216
+ return Promise.resolve(this.l2Blocks.slice(from - 1, from - 1 + limit));
217
+ }
218
+
219
+ public getCheckpoints(from: CheckpointNumber, limit: number) {
220
+ const checkpoints = this.checkpointList.slice(from - 1, from - 1 + limit);
126
221
  return Promise.resolve(
127
- this.l2Blocks
128
- .slice(from - 1, from - 1 + limit)
129
- .filter(b => !proven || this.provenBlockNumber === undefined || b.number <= this.provenBlockNumber),
222
+ checkpoints.map(checkpoint => new PublishedCheckpoint(checkpoint, this.mockL1DataForCheckpoint(checkpoint), [])),
130
223
  );
131
224
  }
132
225
 
133
- public async getPublishedCheckpoints(from: CheckpointNumber, limit: number) {
134
- // TODO: Implement this properly. This only works when we have one block per checkpoint.
135
- return (await this.getPublishedBlocks(from, limit)).map(block => block.toPublishedCheckpoint());
226
+ public getCheckpointByArchive(archive: Fr): Promise<Checkpoint | undefined> {
227
+ const checkpoint = this.checkpointList.find(c => c.archive.root.equals(archive));
228
+ return Promise.resolve(checkpoint);
136
229
  }
137
230
 
138
- public async getCheckpointByArchive(archive: Fr): Promise<Checkpoint | undefined> {
139
- // TODO: Implement this properly. This only works when we have one block per checkpoint.
140
- return (await this.getPublishedBlockByArchive(archive))?.block.toCheckpoint();
231
+ public async getCheckpointedBlockByHash(blockHash: BlockHash): Promise<CheckpointedL2Block | undefined> {
232
+ for (const block of this.l2Blocks) {
233
+ const hash = await block.hash();
234
+ if (hash.equals(blockHash)) {
235
+ return this.toCheckpointedBlock(block);
236
+ }
237
+ }
238
+ return undefined;
141
239
  }
142
240
 
143
- public async getPublishedBlocks(from: number, limit: number, proven?: boolean) {
144
- const blocks = await this.getBlocks(from, limit, proven);
145
- return blocks.map(block =>
146
- PublishedL2Block.fromFields({
147
- block,
148
- l1: new L1PublishedData(BigInt(block.number), BigInt(block.number), Buffer32.random().toString()),
149
- attestations: [],
150
- }),
151
- );
241
+ public getCheckpointedBlockByArchive(archive: Fr): Promise<CheckpointedL2Block | undefined> {
242
+ const block = this.l2Blocks.find(b => b.archive.root.equals(archive));
243
+ if (!block) {
244
+ return Promise.resolve(undefined);
245
+ }
246
+ return Promise.resolve(this.toCheckpointedBlock(block));
152
247
  }
153
248
 
154
- public async getPublishedBlockByHash(blockHash: Fr): Promise<PublishedL2Block | undefined> {
249
+ public async getL2BlockByHash(blockHash: BlockHash): Promise<L2Block | undefined> {
155
250
  for (const block of this.l2Blocks) {
156
251
  const hash = await block.hash();
157
252
  if (hash.equals(blockHash)) {
158
- return PublishedL2Block.fromFields({
159
- block,
160
- l1: new L1PublishedData(BigInt(block.number), BigInt(block.number), Buffer32.random().toString()),
161
- attestations: [],
162
- });
253
+ return block;
163
254
  }
164
255
  }
165
256
  return undefined;
166
257
  }
167
258
 
168
- public getPublishedBlockByArchive(archive: Fr): Promise<PublishedL2Block | undefined> {
259
+ public getL2BlockByArchive(archive: Fr): Promise<L2Block | undefined> {
169
260
  const block = this.l2Blocks.find(b => b.archive.root.equals(archive));
170
- if (!block) {
171
- return Promise.resolve(undefined);
172
- }
173
- return Promise.resolve(
174
- PublishedL2Block.fromFields({
175
- block,
176
- l1: new L1PublishedData(BigInt(block.number), BigInt(block.number), Buffer32.random().toString()),
177
- attestations: [],
178
- }),
179
- );
261
+ return Promise.resolve(block);
180
262
  }
181
263
 
182
- public async getBlockHeaderByHash(blockHash: Fr): Promise<BlockHeader | undefined> {
264
+ public async getBlockHeaderByHash(blockHash: BlockHash): Promise<BlockHeader | undefined> {
183
265
  for (const block of this.l2Blocks) {
184
266
  const hash = await block.hash();
185
267
  if (hash.equals(blockHash)) {
186
- return block.getBlockHeader();
268
+ return block.header;
187
269
  }
188
270
  }
189
271
  return undefined;
@@ -191,31 +273,80 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
191
273
 
192
274
  public getBlockHeaderByArchive(archive: Fr): Promise<BlockHeader | undefined> {
193
275
  const block = this.l2Blocks.find(b => b.archive.root.equals(archive));
194
- return Promise.resolve(block?.getBlockHeader());
276
+ return Promise.resolve(block?.header);
277
+ }
278
+
279
+ public async getBlockData(number: BlockNumber): Promise<BlockData | undefined> {
280
+ const block = this.l2Blocks[number - 1];
281
+ if (!block) {
282
+ return undefined;
283
+ }
284
+ return {
285
+ header: block.header,
286
+ archive: block.archive,
287
+ blockHash: await block.hash(),
288
+ checkpointNumber: block.checkpointNumber,
289
+ indexWithinCheckpoint: block.indexWithinCheckpoint,
290
+ };
291
+ }
292
+
293
+ public async getBlockDataByArchive(archive: Fr): Promise<BlockData | undefined> {
294
+ const block = this.l2Blocks.find(b => b.archive.root.equals(archive));
295
+ if (!block) {
296
+ return undefined;
297
+ }
298
+ return {
299
+ header: block.header,
300
+ archive: block.archive,
301
+ blockHash: await block.hash(),
302
+ checkpointNumber: block.checkpointNumber,
303
+ indexWithinCheckpoint: block.indexWithinCheckpoint,
304
+ };
195
305
  }
196
306
 
197
307
  getBlockHeader(number: number | 'latest'): Promise<BlockHeader | undefined> {
198
- return Promise.resolve(this.l2Blocks.at(typeof number === 'number' ? number - 1 : -1)?.getBlockHeader());
308
+ return Promise.resolve(this.l2Blocks.at(typeof number === 'number' ? number - 1 : -1)?.header);
199
309
  }
200
310
 
201
311
  getCheckpointsForEpoch(epochNumber: EpochNumber): Promise<Checkpoint[]> {
202
- // TODO: Implement this properly. This only works when we have one block per checkpoint.
203
- return this.getBlocksForEpoch(epochNumber).then(blocks => blocks.map(b => b.toCheckpoint()));
312
+ return Promise.resolve(this.getCheckpointsInEpoch(epochNumber));
204
313
  }
205
314
 
206
- getBlocksForEpoch(epochNumber: EpochNumber): Promise<L2Block[]> {
207
- const epochDuration = DefaultL1ContractsConfig.aztecEpochDuration;
208
- const [start, end] = getSlotRangeForEpoch(epochNumber, { epochDuration });
209
- const blocks = this.l2Blocks.filter(b => {
210
- const slot = b.header.globalVariables.slotNumber;
211
- return slot >= start && slot <= end;
212
- });
315
+ getCheckpointsDataForEpoch(epochNumber: EpochNumber): Promise<CheckpointData[]> {
316
+ const checkpoints = this.getCheckpointsInEpoch(epochNumber);
317
+ return Promise.resolve(
318
+ checkpoints.map(
319
+ (checkpoint): CheckpointData => ({
320
+ checkpointNumber: checkpoint.number,
321
+ header: checkpoint.header,
322
+ archive: checkpoint.archive,
323
+ checkpointOutHash: computeCheckpointOutHash(
324
+ checkpoint.blocks.map(b => b.body.txEffects.map(tx => tx.l2ToL1Msgs)),
325
+ ),
326
+ startBlock: checkpoint.blocks[0].number,
327
+ blockCount: checkpoint.blocks.length,
328
+ attestations: [],
329
+ l1: this.mockL1DataForCheckpoint(checkpoint),
330
+ }),
331
+ ),
332
+ );
333
+ }
334
+
335
+ getCheckpointedBlocksForEpoch(epochNumber: EpochNumber): Promise<CheckpointedL2Block[]> {
336
+ const checkpoints = this.getCheckpointsInEpoch(epochNumber);
337
+ return Promise.resolve(
338
+ checkpoints.flatMap(checkpoint => checkpoint.blocks.map(block => this.toCheckpointedBlock(block))),
339
+ );
340
+ }
341
+
342
+ getBlocksForSlot(slotNumber: SlotNumber): Promise<L2Block[]> {
343
+ const blocks = this.l2Blocks.filter(b => b.header.globalVariables.slotNumber === slotNumber);
213
344
  return Promise.resolve(blocks);
214
345
  }
215
346
 
216
- async getBlockHeadersForEpoch(epochNumber: EpochNumber): Promise<BlockHeader[]> {
217
- const blocks = await this.getBlocksForEpoch(epochNumber);
218
- return blocks.map(b => b.getBlockHeader());
347
+ async getCheckpointedBlockHeadersForEpoch(epochNumber: EpochNumber): Promise<BlockHeader[]> {
348
+ const checkpointedBlocks = await this.getCheckpointedBlocksForEpoch(epochNumber);
349
+ return checkpointedBlocks.map(b => b.block.header);
219
350
  }
220
351
 
221
352
  /**
@@ -234,7 +365,7 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
234
365
  return {
235
366
  data: txEffect,
236
367
  l2BlockNumber: block.number,
237
- l2BlockHash: L2BlockHash.fromField(await block.hash()),
368
+ l2BlockHash: await block.hash(),
238
369
  txIndexInBlock: block.body.txEffects.indexOf(txEffect),
239
370
  };
240
371
  }
@@ -248,12 +379,14 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
248
379
  for (const block of this.l2Blocks) {
249
380
  for (const txEffect of block.body.txEffects) {
250
381
  if (txEffect.txHash.equals(txHash)) {
382
+ // In mock, assume all txs are checkpointed with successful execution
251
383
  return new TxReceipt(
252
384
  txHash,
253
- TxStatus.SUCCESS,
254
- '',
385
+ TxStatus.CHECKPOINTED,
386
+ TxExecutionResult.SUCCESS,
387
+ undefined,
255
388
  txEffect.transactionFee.toBigInt(),
256
- L2BlockHash.fromField(await block.hash()),
389
+ await block.hash(),
257
390
  block.number,
258
391
  );
259
392
  }
@@ -263,29 +396,48 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
263
396
  }
264
397
 
265
398
  async getL2Tips(): Promise<L2Tips> {
266
- const [latest, proven, finalized] = [
399
+ const [latest, proven, finalized, checkpointed] = [
267
400
  await this.getBlockNumber(),
268
401
  await this.getProvenBlockNumber(),
269
402
  this.finalizedBlockNumber,
403
+ this.checkpointedBlockNumber,
270
404
  ] as const;
271
405
 
272
406
  const latestBlock = this.l2Blocks[latest - 1];
273
407
  const provenBlock = this.l2Blocks[proven - 1];
274
408
  const finalizedBlock = this.l2Blocks[finalized - 1];
409
+ const checkpointedBlock = this.l2Blocks[checkpointed - 1];
275
410
 
276
- return {
277
- latest: {
278
- number: BlockNumber(latest),
279
- hash: (await latestBlock?.hash())?.toString(),
280
- },
281
- proven: {
282
- number: BlockNumber(proven),
283
- hash: (await provenBlock?.hash())?.toString(),
284
- },
285
- finalized: {
286
- number: BlockNumber(finalized),
287
- hash: (await finalizedBlock?.hash())?.toString(),
411
+ const latestBlockId = {
412
+ number: BlockNumber(latest),
413
+ hash: (await latestBlock?.hash())?.toString(),
414
+ };
415
+ const provenBlockId = {
416
+ number: BlockNumber(proven),
417
+ hash: (await provenBlock?.hash())?.toString(),
418
+ };
419
+ const finalizedBlockId = {
420
+ number: BlockNumber(finalized),
421
+ hash: (await finalizedBlock?.hash())?.toString(),
422
+ };
423
+ const checkpointedBlockId = {
424
+ number: BlockNumber(checkpointed),
425
+ hash: (await checkpointedBlock?.hash())?.toString(),
426
+ };
427
+
428
+ const makeTipId = (blockId: typeof latestBlockId) => ({
429
+ block: blockId,
430
+ checkpoint: {
431
+ number: this.findCheckpointNumberForBlock(blockId.number) ?? CheckpointNumber(0),
432
+ hash: blockId.hash,
288
433
  },
434
+ });
435
+
436
+ return {
437
+ proposed: latestBlockId,
438
+ checkpointed: makeTipId(checkpointedBlockId),
439
+ proven: makeTipId(provenBlockId),
440
+ finalized: makeTipId(finalizedBlockId),
289
441
  };
290
442
  }
291
443
 
@@ -363,7 +515,41 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
363
515
  return Promise.resolve(false);
364
516
  }
365
517
 
366
- getPendingChainValidationStatus(): Promise<ValidateBlockResult> {
518
+ getPendingChainValidationStatus(): Promise<ValidateCheckpointResult> {
367
519
  return Promise.resolve({ valid: true });
368
520
  }
521
+
522
+ /** Returns checkpoints whose slot falls within the given epoch. */
523
+ private getCheckpointsInEpoch(epochNumber: EpochNumber): Checkpoint[] {
524
+ const epochDuration = DefaultL1ContractsConfig.aztecEpochDuration;
525
+ const [start, end] = getSlotRangeForEpoch(epochNumber, { epochDuration });
526
+ return this.checkpointList.filter(c => c.header.slotNumber >= start && c.header.slotNumber <= end);
527
+ }
528
+
529
+ /** Creates a mock L1PublishedData for a checkpoint. */
530
+ private mockL1DataForCheckpoint(checkpoint: Checkpoint): L1PublishedData {
531
+ return new L1PublishedData(BigInt(checkpoint.number), BigInt(checkpoint.number), Buffer32.random().toString());
532
+ }
533
+
534
+ /** Creates a CheckpointedL2Block from a block using stored checkpoint info. */
535
+ private toCheckpointedBlock(block: L2Block): CheckpointedL2Block {
536
+ const checkpoint = this.checkpointList.find(c => c.blocks.some(b => b.number === block.number));
537
+ const checkpointNumber = checkpoint?.number ?? block.checkpointNumber;
538
+ return new CheckpointedL2Block(
539
+ checkpointNumber,
540
+ block,
541
+ new L1PublishedData(
542
+ BigInt(block.number),
543
+ BigInt(block.number),
544
+ `0x${block.number.toString(16).padStart(64, '0')}`,
545
+ ),
546
+ [],
547
+ );
548
+ }
549
+
550
+ /** Finds the checkpoint number for a block, or undefined if the block is not in any checkpoint. */
551
+ private findCheckpointNumberForBlock(blockNumber: BlockNumber): CheckpointNumber | undefined {
552
+ const checkpoint = this.checkpointList.find(c => c.blocks.some(b => b.number === blockNumber));
553
+ return checkpoint?.number;
554
+ }
369
555
  }