@aztec/archiver 0.0.1-commit.03f7ef2 → 0.0.1-commit.04d373f

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 (235) hide show
  1. package/README.md +164 -22
  2. package/dest/archiver.d.ts +158 -0
  3. package/dest/archiver.d.ts.map +1 -0
  4. package/dest/archiver.js +881 -0
  5. package/dest/config.d.ts +33 -0
  6. package/dest/config.d.ts.map +1 -0
  7. package/dest/config.js +79 -0
  8. package/dest/errors.d.ts +87 -0
  9. package/dest/errors.d.ts.map +1 -0
  10. package/dest/errors.js +129 -0
  11. package/dest/factory.d.ts +16 -10
  12. package/dest/factory.d.ts.map +1 -1
  13. package/dest/factory.js +112 -20
  14. package/dest/index.d.ts +19 -4
  15. package/dest/index.d.ts.map +1 -1
  16. package/dest/index.js +17 -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 +35 -32
  23. package/dest/l1/calldata_retriever.d.ts +136 -0
  24. package/dest/l1/calldata_retriever.d.ts.map +1 -0
  25. package/dest/l1/calldata_retriever.js +412 -0
  26. package/dest/l1/data_retrieval.d.ts +97 -0
  27. package/dest/l1/data_retrieval.d.ts.map +1 -0
  28. package/dest/{archiver/l1 → l1}/data_retrieval.js +65 -89
  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 +5 -5
  32. package/dest/l1/spire_proposer.d.ts.map +1 -0
  33. package/dest/{archiver/l1 → l1}/spire_proposer.js +9 -17
  34. package/dest/l1/trace_tx.d.ts +43 -0
  35. package/dest/l1/trace_tx.d.ts.map +1 -0
  36. package/dest/l1/types.d.ts +12 -0
  37. package/dest/l1/types.d.ts.map +1 -0
  38. package/dest/l1/validate_historical_logs.d.ts +23 -0
  39. package/dest/l1/validate_historical_logs.d.ts.map +1 -0
  40. package/dest/l1/validate_historical_logs.js +108 -0
  41. package/dest/{archiver/l1 → l1}/validate_trace.d.ts +6 -3
  42. package/dest/l1/validate_trace.d.ts.map +1 -0
  43. package/dest/{archiver/l1 → l1}/validate_trace.js +14 -10
  44. package/dest/modules/contract_data_source_adapter.d.ts +25 -0
  45. package/dest/modules/contract_data_source_adapter.d.ts.map +1 -0
  46. package/dest/modules/contract_data_source_adapter.js +40 -0
  47. package/dest/modules/data_source_base.d.ts +113 -0
  48. package/dest/modules/data_source_base.d.ts.map +1 -0
  49. package/dest/modules/data_source_base.js +351 -0
  50. package/dest/modules/data_store_updater.d.ts +105 -0
  51. package/dest/modules/data_store_updater.d.ts.map +1 -0
  52. package/dest/modules/data_store_updater.js +392 -0
  53. package/dest/modules/instrumentation.d.ts +55 -0
  54. package/dest/modules/instrumentation.d.ts.map +1 -0
  55. package/dest/modules/instrumentation.js +145 -0
  56. package/dest/modules/l1_synchronizer.d.ts +77 -0
  57. package/dest/modules/l1_synchronizer.d.ts.map +1 -0
  58. package/dest/modules/l1_synchronizer.js +1344 -0
  59. package/dest/modules/validation.d.ts +18 -0
  60. package/dest/modules/validation.d.ts.map +1 -0
  61. package/dest/{archiver → modules}/validation.js +13 -7
  62. package/dest/store/block_store.d.ts +300 -0
  63. package/dest/store/block_store.d.ts.map +1 -0
  64. package/dest/store/block_store.js +1219 -0
  65. package/dest/store/contract_class_store.d.ts +31 -0
  66. package/dest/store/contract_class_store.d.ts.map +1 -0
  67. package/dest/store/contract_class_store.js +80 -0
  68. package/dest/store/contract_instance_store.d.ts +51 -0
  69. package/dest/store/contract_instance_store.d.ts.map +1 -0
  70. package/dest/{archiver/kv_archiver_store → store}/contract_instance_store.js +38 -3
  71. package/dest/store/data_stores.d.ts +68 -0
  72. package/dest/store/data_stores.d.ts.map +1 -0
  73. package/dest/store/data_stores.js +54 -0
  74. package/dest/store/function_names_cache.d.ts +17 -0
  75. package/dest/store/function_names_cache.d.ts.map +1 -0
  76. package/dest/store/function_names_cache.js +30 -0
  77. package/dest/store/l2_tips_cache.d.ts +25 -0
  78. package/dest/store/l2_tips_cache.d.ts.map +1 -0
  79. package/dest/store/l2_tips_cache.js +26 -0
  80. package/dest/store/log_store.d.ts +59 -0
  81. package/dest/store/log_store.d.ts.map +1 -0
  82. package/dest/store/log_store.js +310 -0
  83. package/dest/store/log_store_codec.d.ts +70 -0
  84. package/dest/store/log_store_codec.d.ts.map +1 -0
  85. package/dest/store/log_store_codec.js +101 -0
  86. package/dest/store/message_store.d.ts +50 -0
  87. package/dest/store/message_store.d.ts.map +1 -0
  88. package/dest/{archiver/kv_archiver_store → store}/message_store.js +51 -9
  89. package/dest/{archiver/structs → structs}/data_retrieval.d.ts +1 -1
  90. package/dest/structs/data_retrieval.d.ts.map +1 -0
  91. package/dest/structs/inbox_message.d.ts +15 -0
  92. package/dest/structs/inbox_message.d.ts.map +1 -0
  93. package/dest/{archiver/structs → structs}/published.d.ts +1 -1
  94. package/dest/structs/published.d.ts.map +1 -0
  95. package/dest/test/fake_l1_state.d.ts +214 -0
  96. package/dest/test/fake_l1_state.d.ts.map +1 -0
  97. package/dest/test/fake_l1_state.js +517 -0
  98. package/dest/test/index.d.ts +2 -1
  99. package/dest/test/index.d.ts.map +1 -1
  100. package/dest/test/index.js +4 -1
  101. package/dest/test/mock_archiver.d.ts +2 -2
  102. package/dest/test/mock_archiver.d.ts.map +1 -1
  103. package/dest/test/mock_archiver.js +3 -3
  104. package/dest/test/mock_l1_to_l2_message_source.d.ts +2 -2
  105. package/dest/test/mock_l1_to_l2_message_source.d.ts.map +1 -1
  106. package/dest/test/mock_l1_to_l2_message_source.js +14 -4
  107. package/dest/test/mock_l2_block_source.d.ts +68 -40
  108. package/dest/test/mock_l2_block_source.d.ts.map +1 -1
  109. package/dest/test/mock_l2_block_source.js +356 -131
  110. package/dest/test/mock_structs.d.ts +81 -3
  111. package/dest/test/mock_structs.d.ts.map +1 -1
  112. package/dest/test/mock_structs.js +152 -7
  113. package/dest/test/noop_l1_archiver.d.ts +29 -0
  114. package/dest/test/noop_l1_archiver.d.ts.map +1 -0
  115. package/dest/test/noop_l1_archiver.js +85 -0
  116. package/package.json +17 -18
  117. package/src/archiver.ts +681 -0
  118. package/src/{archiver/config.ts → config.ts} +45 -14
  119. package/src/errors.ts +203 -0
  120. package/src/factory.ts +175 -22
  121. package/src/index.ts +27 -3
  122. package/src/interfaces.ts +9 -0
  123. package/src/l1/README.md +55 -0
  124. package/src/{archiver/l1 → l1}/bin/retrieve-calldata.ts +45 -33
  125. package/src/l1/calldata_retriever.ts +522 -0
  126. package/src/{archiver/l1 → l1}/data_retrieval.ts +106 -134
  127. package/src/{archiver/l1 → l1}/spire_proposer.ts +7 -15
  128. package/src/l1/validate_historical_logs.ts +140 -0
  129. package/src/{archiver/l1 → l1}/validate_trace.ts +25 -7
  130. package/src/modules/contract_data_source_adapter.ts +55 -0
  131. package/src/modules/data_source_base.ts +493 -0
  132. package/src/modules/data_store_updater.ts +518 -0
  133. package/src/modules/instrumentation.ts +204 -0
  134. package/src/modules/l1_synchronizer.ts +1257 -0
  135. package/src/{archiver → modules}/validation.ts +21 -15
  136. package/src/store/block_store.ts +1590 -0
  137. package/src/store/contract_class_store.ts +108 -0
  138. package/src/{archiver/kv_archiver_store → store}/contract_instance_store.ts +52 -6
  139. package/src/store/data_stores.ts +104 -0
  140. package/src/store/function_names_cache.ts +37 -0
  141. package/src/store/l2_tips_cache.ts +35 -0
  142. package/src/store/log_store.ts +379 -0
  143. package/src/store/log_store_codec.ts +132 -0
  144. package/src/{archiver/kv_archiver_store → store}/message_store.ts +60 -10
  145. package/src/{archiver/structs → structs}/inbox_message.ts +1 -1
  146. package/src/test/fake_l1_state.ts +770 -0
  147. package/src/test/index.ts +4 -0
  148. package/src/test/mock_archiver.ts +4 -3
  149. package/src/test/mock_l1_to_l2_message_source.ts +11 -4
  150. package/src/test/mock_l2_block_source.ts +434 -144
  151. package/src/test/mock_structs.ts +283 -8
  152. package/src/test/noop_l1_archiver.ts +139 -0
  153. package/dest/archiver/archiver.d.ts +0 -304
  154. package/dest/archiver/archiver.d.ts.map +0 -1
  155. package/dest/archiver/archiver.js +0 -1645
  156. package/dest/archiver/archiver_store.d.ts +0 -308
  157. package/dest/archiver/archiver_store.d.ts.map +0 -1
  158. package/dest/archiver/archiver_store.js +0 -4
  159. package/dest/archiver/archiver_store_test_suite.d.ts +0 -8
  160. package/dest/archiver/archiver_store_test_suite.d.ts.map +0 -1
  161. package/dest/archiver/archiver_store_test_suite.js +0 -2790
  162. package/dest/archiver/config.d.ts +0 -22
  163. package/dest/archiver/config.d.ts.map +0 -1
  164. package/dest/archiver/config.js +0 -62
  165. package/dest/archiver/errors.d.ts +0 -36
  166. package/dest/archiver/errors.d.ts.map +0 -1
  167. package/dest/archiver/errors.js +0 -54
  168. package/dest/archiver/index.d.ts +0 -7
  169. package/dest/archiver/index.d.ts.map +0 -1
  170. package/dest/archiver/index.js +0 -4
  171. package/dest/archiver/instrumentation.d.ts +0 -37
  172. package/dest/archiver/instrumentation.d.ts.map +0 -1
  173. package/dest/archiver/instrumentation.js +0 -151
  174. package/dest/archiver/kv_archiver_store/block_store.d.ts +0 -157
  175. package/dest/archiver/kv_archiver_store/block_store.d.ts.map +0 -1
  176. package/dest/archiver/kv_archiver_store/block_store.js +0 -607
  177. package/dest/archiver/kv_archiver_store/contract_class_store.d.ts +0 -18
  178. package/dest/archiver/kv_archiver_store/contract_class_store.d.ts.map +0 -1
  179. package/dest/archiver/kv_archiver_store/contract_class_store.js +0 -120
  180. package/dest/archiver/kv_archiver_store/contract_instance_store.d.ts +0 -24
  181. package/dest/archiver/kv_archiver_store/contract_instance_store.d.ts.map +0 -1
  182. package/dest/archiver/kv_archiver_store/kv_archiver_store.d.ts +0 -158
  183. package/dest/archiver/kv_archiver_store/kv_archiver_store.d.ts.map +0 -1
  184. package/dest/archiver/kv_archiver_store/kv_archiver_store.js +0 -313
  185. package/dest/archiver/kv_archiver_store/log_store.d.ts +0 -45
  186. package/dest/archiver/kv_archiver_store/log_store.d.ts.map +0 -1
  187. package/dest/archiver/kv_archiver_store/log_store.js +0 -404
  188. package/dest/archiver/kv_archiver_store/message_store.d.ts +0 -40
  189. package/dest/archiver/kv_archiver_store/message_store.d.ts.map +0 -1
  190. package/dest/archiver/l1/bin/retrieve-calldata.d.ts.map +0 -1
  191. package/dest/archiver/l1/calldata_retriever.d.ts +0 -112
  192. package/dest/archiver/l1/calldata_retriever.d.ts.map +0 -1
  193. package/dest/archiver/l1/calldata_retriever.js +0 -471
  194. package/dest/archiver/l1/data_retrieval.d.ts +0 -90
  195. package/dest/archiver/l1/data_retrieval.d.ts.map +0 -1
  196. package/dest/archiver/l1/debug_tx.d.ts.map +0 -1
  197. package/dest/archiver/l1/spire_proposer.d.ts.map +0 -1
  198. package/dest/archiver/l1/trace_tx.d.ts +0 -97
  199. package/dest/archiver/l1/trace_tx.d.ts.map +0 -1
  200. package/dest/archiver/l1/types.d.ts +0 -12
  201. package/dest/archiver/l1/types.d.ts.map +0 -1
  202. package/dest/archiver/l1/validate_trace.d.ts.map +0 -1
  203. package/dest/archiver/structs/data_retrieval.d.ts.map +0 -1
  204. package/dest/archiver/structs/inbox_message.d.ts +0 -15
  205. package/dest/archiver/structs/inbox_message.d.ts.map +0 -1
  206. package/dest/archiver/structs/published.d.ts.map +0 -1
  207. package/dest/archiver/validation.d.ts +0 -17
  208. package/dest/archiver/validation.d.ts.map +0 -1
  209. package/dest/rpc/index.d.ts +0 -9
  210. package/dest/rpc/index.d.ts.map +0 -1
  211. package/dest/rpc/index.js +0 -15
  212. package/src/archiver/archiver.ts +0 -2157
  213. package/src/archiver/archiver_store.ts +0 -372
  214. package/src/archiver/archiver_store_test_suite.ts +0 -2863
  215. package/src/archiver/errors.ts +0 -90
  216. package/src/archiver/index.ts +0 -6
  217. package/src/archiver/instrumentation.ts +0 -201
  218. package/src/archiver/kv_archiver_store/block_store.ts +0 -822
  219. package/src/archiver/kv_archiver_store/contract_class_store.ts +0 -176
  220. package/src/archiver/kv_archiver_store/kv_archiver_store.ts +0 -438
  221. package/src/archiver/kv_archiver_store/log_store.ts +0 -522
  222. package/src/archiver/l1/README.md +0 -98
  223. package/src/archiver/l1/calldata_retriever.ts +0 -641
  224. package/src/rpc/index.ts +0 -16
  225. /package/dest/{archiver/l1 → l1}/debug_tx.js +0 -0
  226. /package/dest/{archiver/l1 → l1}/trace_tx.js +0 -0
  227. /package/dest/{archiver/l1 → l1}/types.js +0 -0
  228. /package/dest/{archiver/structs → structs}/data_retrieval.js +0 -0
  229. /package/dest/{archiver/structs → structs}/inbox_message.js +0 -0
  230. /package/dest/{archiver/structs → structs}/published.js +0 -0
  231. /package/src/{archiver/l1 → l1}/debug_tx.ts +0 -0
  232. /package/src/{archiver/l1 → l1}/trace_tx.ts +0 -0
  233. /package/src/{archiver/l1 → l1}/types.ts +0 -0
  234. /package/src/{archiver/structs → structs}/data_retrieval.ts +0 -0
  235. /package/src/{archiver/structs → structs}/published.ts +0 -0
@@ -1,10 +1,28 @@
1
- import { CheckpointNumber } from '@aztec/foundation/branded-types';
1
+ import {
2
+ MAX_NOTE_HASHES_PER_TX,
3
+ NUMBER_OF_L1_L2_MESSAGES_PER_ROLLUP,
4
+ PRIVATE_LOG_SIZE_IN_FIELDS,
5
+ } from '@aztec/constants';
6
+ import { makeTuple } from '@aztec/foundation/array';
7
+ import { BlockNumber, CheckpointNumber, IndexWithinCheckpoint } from '@aztec/foundation/branded-types';
2
8
  import { Buffer16, Buffer32 } from '@aztec/foundation/buffer';
9
+ import { times, timesParallel } from '@aztec/foundation/collection';
3
10
  import { randomBigInt, randomInt } from '@aztec/foundation/crypto/random';
11
+ import type { Secp256k1Signer } from '@aztec/foundation/crypto/secp256k1-signer';
4
12
  import { Fr } from '@aztec/foundation/curves/bn254';
13
+ import { EthAddress } from '@aztec/foundation/eth-address';
14
+ import { AztecAddress } from '@aztec/stdlib/aztec-address';
15
+ import { CommitteeAttestation, L2Block } from '@aztec/stdlib/block';
16
+ import { Checkpoint, L1PublishedData, PublishedCheckpoint } from '@aztec/stdlib/checkpoint';
17
+ import { PrivateLog, PublicLog, SiloedTag, Tag } from '@aztec/stdlib/logs';
5
18
  import { InboxLeaf } from '@aztec/stdlib/messaging';
19
+ import { orderAttestations } from '@aztec/stdlib/p2p';
20
+ import { CheckpointHeader } from '@aztec/stdlib/rollup';
21
+ import { makeCheckpointAttestationFromCheckpoint } from '@aztec/stdlib/testing';
22
+ import { AppendOnlyTreeSnapshot } from '@aztec/stdlib/trees';
23
+ import { PartialStateReference, StateReference, TxEffect } from '@aztec/stdlib/tx';
6
24
 
7
- import { type InboxMessage, updateRollingHash } from '../archiver/structs/inbox_message.js';
25
+ import { type InboxMessage, updateRollingHash } from '../structs/inbox_message.js';
8
26
 
9
27
  export function makeInboxMessage(
10
28
  previousRollingHash = Buffer16.ZERO,
@@ -28,23 +46,280 @@ export function makeInboxMessage(
28
46
  }
29
47
 
30
48
  export function makeInboxMessages(
31
- count: number,
49
+ totalCount: number,
32
50
  opts: {
33
51
  initialHash?: Buffer16;
34
52
  initialCheckpointNumber?: CheckpointNumber;
53
+ messagesPerCheckpoint?: number;
35
54
  overrideFn?: (msg: InboxMessage, index: number) => InboxMessage;
36
55
  } = {},
37
56
  ): InboxMessage[] {
38
- const { initialHash = Buffer16.ZERO, overrideFn = msg => msg, initialCheckpointNumber = 1 } = opts;
57
+ const {
58
+ initialHash = Buffer16.ZERO,
59
+ overrideFn = msg => msg,
60
+ initialCheckpointNumber = CheckpointNumber(1),
61
+ messagesPerCheckpoint = 1,
62
+ } = opts;
63
+
39
64
  const messages: InboxMessage[] = [];
40
65
  let rollingHash = initialHash;
41
- for (let i = 0; i < count; i++) {
66
+ for (let i = 0; i < totalCount; i++) {
67
+ const msgIndex = i % messagesPerCheckpoint;
68
+ const checkpointNumber = CheckpointNumber.fromBigInt(
69
+ BigInt(initialCheckpointNumber) + BigInt(i) / BigInt(messagesPerCheckpoint),
70
+ );
42
71
  const leaf = Fr.random();
43
- const checkpointNumber = CheckpointNumber(i + initialCheckpointNumber);
44
- const message = overrideFn(makeInboxMessage(rollingHash, { leaf, checkpointNumber }), i);
72
+ const message = overrideFn(
73
+ makeInboxMessage(rollingHash, {
74
+ leaf,
75
+ checkpointNumber,
76
+ index: InboxLeaf.smallestIndexForCheckpoint(checkpointNumber) + BigInt(msgIndex),
77
+ }),
78
+ i,
79
+ );
45
80
  rollingHash = message.rollingHash;
46
81
  messages.push(message);
47
82
  }
48
-
49
83
  return messages;
50
84
  }
85
+
86
+ /** Creates inbox messages distributed across multiple blocks with proper checkpoint numbering. */
87
+ export function makeInboxMessagesWithFullBlocks(
88
+ blockCount: number,
89
+ opts: { initialCheckpointNumber?: CheckpointNumber } = {},
90
+ ): InboxMessage[] {
91
+ const { initialCheckpointNumber = CheckpointNumber(13) } = opts;
92
+ return makeInboxMessages(NUMBER_OF_L1_L2_MESSAGES_PER_ROLLUP * blockCount, {
93
+ overrideFn: (msg, i) => {
94
+ const checkpointNumber = CheckpointNumber(
95
+ initialCheckpointNumber + Math.floor(i / NUMBER_OF_L1_L2_MESSAGES_PER_ROLLUP),
96
+ );
97
+ const index =
98
+ InboxLeaf.smallestIndexForCheckpoint(checkpointNumber) + BigInt(i % NUMBER_OF_L1_L2_MESSAGES_PER_ROLLUP);
99
+ return { ...msg, checkpointNumber, index };
100
+ },
101
+ });
102
+ }
103
+
104
+ /** Creates a deterministic block hash from a block number. */
105
+ export function makeBlockHash(blockNumber: number): `0x${string}` {
106
+ return `0x${blockNumber.toString(16).padStart(64, '0')}`;
107
+ }
108
+
109
+ /**
110
+ * Creates a StateReference with properly calculated noteHashTree.nextAvailableLeafIndex.
111
+ * This ensures LogStore's dataStartIndexForBlock calculation doesn't produce negative values.
112
+ */
113
+ export function makeStateForBlock(blockNumber: number, txsPerBlock: number): StateReference {
114
+ const noteHashIndex = blockNumber * txsPerBlock * MAX_NOTE_HASHES_PER_TX;
115
+ return new StateReference(
116
+ AppendOnlyTreeSnapshot.random(),
117
+ new PartialStateReference(
118
+ new AppendOnlyTreeSnapshot(Fr.random(), noteHashIndex),
119
+ AppendOnlyTreeSnapshot.random(),
120
+ AppendOnlyTreeSnapshot.random(),
121
+ ),
122
+ );
123
+ }
124
+
125
+ /** Creates L1PublishedData with deterministic values based on l1BlockNumber. */
126
+ export function makeL1PublishedData(l1BlockNumber: number): L1PublishedData {
127
+ return new L1PublishedData(BigInt(l1BlockNumber), BigInt(l1BlockNumber * 1000), makeBlockHash(l1BlockNumber));
128
+ }
129
+
130
+ /** Creates a Checkpoint from a list of blocks with a header that matches the blocks' structure. */
131
+ export function makeCheckpoint(blocks: L2Block[], checkpointNumber = CheckpointNumber(1)): Checkpoint {
132
+ const firstBlock = blocks[0];
133
+ const { slotNumber, timestamp, coinbase, feeRecipient, gasFees } = firstBlock.header.globalVariables;
134
+ return new Checkpoint(
135
+ blocks.at(-1)!.archive,
136
+ CheckpointHeader.random({
137
+ lastArchiveRoot: firstBlock.header.lastArchive.root,
138
+ slotNumber,
139
+ timestamp,
140
+ coinbase,
141
+ feeRecipient,
142
+ gasFees,
143
+ }),
144
+ blocks,
145
+ checkpointNumber,
146
+ );
147
+ }
148
+
149
+ /** Wraps a Checkpoint with L1 published data and random attestations. */
150
+ export function makePublishedCheckpoint(
151
+ checkpoint: Checkpoint,
152
+ l1BlockNumber: number,
153
+ attestationCount = 3,
154
+ ): PublishedCheckpoint {
155
+ return new PublishedCheckpoint(
156
+ checkpoint,
157
+ makeL1PublishedData(l1BlockNumber),
158
+ times(attestationCount, CommitteeAttestation.random),
159
+ );
160
+ }
161
+
162
+ export interface MakeChainedCheckpointsOptions {
163
+ /** Number of L2 blocks per checkpoint. Default: 1 */
164
+ blocksPerCheckpoint?: number;
165
+ /** Number of transactions per block. Default: 4 */
166
+ txsPerBlock?: number;
167
+ /** Starting checkpoint number. Default: CheckpointNumber(1) */
168
+ startCheckpointNumber?: CheckpointNumber;
169
+ /** Starting block number. Default: 1 */
170
+ startBlockNumber?: number;
171
+ /** Starting L1 block number. Default: 10 */
172
+ startL1BlockNumber?: number;
173
+ /** Previous archive to chain from. Default: undefined */
174
+ previousArchive?: AppendOnlyTreeSnapshot;
175
+ /** Optional function to provide per-checkpoint overrides */
176
+ makeCheckpointOptions?: (cpNumber: CheckpointNumber) => Partial<Parameters<typeof Checkpoint.random>[1]>;
177
+ }
178
+
179
+ /**
180
+ * Creates multiple checkpoints with properly chained archives.
181
+ * Each checkpoint's blocks have their lastArchive set to the previous block's archive,
182
+ * ensuring archive chain continuity for testing.
183
+ */
184
+ export async function makeChainedCheckpoints(
185
+ count: number,
186
+ options: MakeChainedCheckpointsOptions = {},
187
+ ): Promise<PublishedCheckpoint[]> {
188
+ const {
189
+ blocksPerCheckpoint = 1,
190
+ txsPerBlock = 4,
191
+ startCheckpointNumber = CheckpointNumber(1),
192
+ startBlockNumber = 1,
193
+ startL1BlockNumber = 10,
194
+ makeCheckpointOptions,
195
+ } = options;
196
+
197
+ let previousArchive = options.previousArchive;
198
+ const checkpoints: PublishedCheckpoint[] = [];
199
+
200
+ for (let i = 0; i < count; i++) {
201
+ const cpNumber = CheckpointNumber(startCheckpointNumber + i);
202
+ const blockStart = startBlockNumber + i * blocksPerCheckpoint;
203
+ const customOptions = makeCheckpointOptions?.(cpNumber) ?? {};
204
+
205
+ const checkpoint = await Checkpoint.random(cpNumber, {
206
+ numBlocks: blocksPerCheckpoint,
207
+ startBlockNumber: blockStart,
208
+ previousArchive,
209
+ txsPerBlock,
210
+ state: makeStateForBlock(blockStart, txsPerBlock),
211
+ txOptions: { numPublicCallsPerTx: 2, numPublicLogsPerCall: 2 },
212
+ ...customOptions,
213
+ });
214
+
215
+ previousArchive = checkpoint.blocks.at(-1)!.archive;
216
+ checkpoints.push(makePublishedCheckpoint(checkpoint, startL1BlockNumber + i * 10));
217
+ }
218
+
219
+ return checkpoints;
220
+ }
221
+
222
+ /**
223
+ * Creates a PublishedCheckpoint with attestations signed by the provided signers.
224
+ * Useful for testing attestation validation.
225
+ */
226
+ export function makeSignedPublishedCheckpoint(
227
+ checkpoint: Checkpoint,
228
+ signers: Secp256k1Signer[],
229
+ committee: EthAddress[],
230
+ l1BlockNumber = 1,
231
+ ): PublishedCheckpoint {
232
+ const attestations = signers.map(signer => makeCheckpointAttestationFromCheckpoint(checkpoint, signer));
233
+ const committeeAttestations = orderAttestations(attestations, committee);
234
+ return new PublishedCheckpoint(checkpoint, makeL1PublishedData(l1BlockNumber), committeeAttestations);
235
+ }
236
+
237
+ /** Creates a deterministic SiloedTag for private log testing. */
238
+ export function makePrivateLogTag(blockNumber: number, txIndex: number, logIndex: number): SiloedTag {
239
+ return new SiloedTag(
240
+ blockNumber === 1 && txIndex === 0 && logIndex === 0
241
+ ? Fr.ZERO
242
+ : new Fr(blockNumber * 100 + txIndex * 10 + logIndex),
243
+ );
244
+ }
245
+
246
+ /** Creates a PrivateLog with fields derived from the tag. */
247
+ export function makePrivateLog(tag: SiloedTag): PrivateLog {
248
+ return PrivateLog.from({
249
+ fields: makeTuple(PRIVATE_LOG_SIZE_IN_FIELDS, i => (!i ? tag.value : new Fr(tag.value.toBigInt() + BigInt(i)))),
250
+ emittedLength: PRIVATE_LOG_SIZE_IN_FIELDS,
251
+ });
252
+ }
253
+
254
+ /** Creates multiple private logs for a transaction. */
255
+ export function mockPrivateLogs(blockNumber: number, txIndex: number, numLogsPerTx: number): PrivateLog[] {
256
+ return times(numLogsPerTx, logIndex => {
257
+ const tag = makePrivateLogTag(blockNumber, txIndex, logIndex);
258
+ return makePrivateLog(tag);
259
+ });
260
+ }
261
+
262
+ /** Creates a deterministic Tag for public log testing. */
263
+ export function makePublicLogTag(blockNumber: number, txIndex: number, logIndex: number): Tag {
264
+ return new Tag(
265
+ blockNumber === 1 && txIndex === 0 && logIndex === 0
266
+ ? Fr.ZERO
267
+ : new Fr((blockNumber * 100 + txIndex * 10 + logIndex) * 123),
268
+ );
269
+ }
270
+
271
+ /** Creates a PublicLog with fields derived from the tag. */
272
+ export function makePublicLog(tag: Tag, contractAddress: AztecAddress = AztecAddress.fromNumber(543254)): PublicLog {
273
+ return PublicLog.from({
274
+ contractAddress,
275
+ fields: new Array(10).fill(null).map((_, i) => (!i ? tag.value : new Fr(tag.value.toBigInt() + BigInt(i)))),
276
+ });
277
+ }
278
+
279
+ /** Creates multiple public logs for a transaction. */
280
+ export function makePublicLogs(
281
+ blockNumber: number,
282
+ txIndex: number,
283
+ numLogsPerTx: number,
284
+ contractAddress: AztecAddress = AztecAddress.fromNumber(543254),
285
+ ): PublicLog[] {
286
+ return times(numLogsPerTx, logIndex => {
287
+ const tag = makePublicLogTag(blockNumber, txIndex, logIndex);
288
+ return makePublicLog(tag, contractAddress);
289
+ });
290
+ }
291
+
292
+ export interface MockCheckpointWithLogsOptions {
293
+ previousArchive?: AppendOnlyTreeSnapshot;
294
+ numTxsPerBlock?: number;
295
+ privateLogs?: { numLogsPerTx: number };
296
+ publicLogs?: { numLogsPerTx: number; contractAddress?: AztecAddress };
297
+ }
298
+
299
+ /** Creates a checkpoint with specified logs on each tx effect. */
300
+ export async function makeCheckpointWithLogs(
301
+ blockNumber: number,
302
+ options: MockCheckpointWithLogsOptions = {},
303
+ ): Promise<PublishedCheckpoint> {
304
+ const { previousArchive, numTxsPerBlock = 4, privateLogs, publicLogs } = options;
305
+
306
+ const block = await L2Block.random(BlockNumber(blockNumber), {
307
+ checkpointNumber: CheckpointNumber.fromBlockNumber(BlockNumber(blockNumber)),
308
+ indexWithinCheckpoint: IndexWithinCheckpoint(0),
309
+ state: makeStateForBlock(blockNumber, numTxsPerBlock),
310
+ ...(previousArchive ? { lastArchive: previousArchive } : {}),
311
+ });
312
+ block.header.globalVariables.blockNumber = BlockNumber(blockNumber);
313
+
314
+ block.body.txEffects = await timesParallel(numTxsPerBlock, async (txIndex: number) => {
315
+ const txEffect = await TxEffect.random();
316
+ txEffect.privateLogs = privateLogs ? mockPrivateLogs(blockNumber, txIndex, privateLogs.numLogsPerTx) : [];
317
+ txEffect.publicLogs = publicLogs
318
+ ? makePublicLogs(blockNumber, txIndex, publicLogs.numLogsPerTx, publicLogs.contractAddress)
319
+ : [];
320
+ return txEffect;
321
+ });
322
+
323
+ const checkpoint = makeCheckpoint([block], CheckpointNumber.fromBlockNumber(BlockNumber(blockNumber)));
324
+ return makePublishedCheckpoint(checkpoint, blockNumber);
325
+ }
@@ -0,0 +1,139 @@
1
+ import type { BlobClientInterface } from '@aztec/blob-client/client';
2
+ import { EpochCache } from '@aztec/epoch-cache';
3
+ import type { RollupContract } from '@aztec/ethereum/contracts';
4
+ import type { ViemPublicClient, ViemPublicDebugClient } from '@aztec/ethereum/types';
5
+ import { SlotNumber } from '@aztec/foundation/branded-types';
6
+ import { Buffer32 } from '@aztec/foundation/buffer';
7
+ import { Fr } from '@aztec/foundation/curves/bn254';
8
+ import { EthAddress } from '@aztec/foundation/eth-address';
9
+ import { DateProvider } from '@aztec/foundation/timer';
10
+ import type { FunctionsOf } from '@aztec/foundation/types';
11
+ import type { ArchiverEmitter, BlockHash } from '@aztec/stdlib/block';
12
+ import type { L1RollupConstants } from '@aztec/stdlib/epoch-helpers';
13
+ import type { BlockHeader } from '@aztec/stdlib/tx';
14
+ import { type TelemetryClient, type Tracer, getTelemetryClient } from '@aztec/telemetry-client';
15
+
16
+ import { mock } from 'jest-mock-extended';
17
+ import { EventEmitter } from 'node:events';
18
+
19
+ import { Archiver } from '../archiver.js';
20
+ import { ArchiverInstrumentation } from '../modules/instrumentation.js';
21
+ import type { ArchiverL1Synchronizer } from '../modules/l1_synchronizer.js';
22
+ import type { ArchiverDataStores } from '../store/data_stores.js';
23
+ import { L2TipsCache } from '../store/l2_tips_cache.js';
24
+
25
+ /** Noop L1 synchronizer for testing without L1 connectivity. */
26
+ class NoopL1Synchronizer implements FunctionsOf<ArchiverL1Synchronizer> {
27
+ public readonly tracer: Tracer;
28
+
29
+ constructor(tracer: Tracer) {
30
+ this.tracer = tracer;
31
+ }
32
+
33
+ setConfig(_config: unknown) {}
34
+ getL1BlockNumber(): bigint | undefined {
35
+ return 0n;
36
+ }
37
+ getL1Timestamp(): bigint | undefined {
38
+ return undefined;
39
+ }
40
+ testEthereumNodeSynced(): Promise<void> {
41
+ return Promise.resolve();
42
+ }
43
+ syncFromL1(_initialSyncComplete: boolean): Promise<void> {
44
+ return Promise.resolve();
45
+ }
46
+ }
47
+
48
+ /**
49
+ * Archiver with mocked L1 connectivity for testing.
50
+ * Uses mock L1 clients and a noop synchronizer, enabling tests that
51
+ * don't require real Ethereum connectivity.
52
+ */
53
+ export class NoopL1Archiver extends Archiver {
54
+ constructor(
55
+ dataStores: ArchiverDataStores,
56
+ l1Constants: L1RollupConstants & { genesisArchiveRoot: Fr },
57
+ instrumentation: ArchiverInstrumentation,
58
+ initialHeader: BlockHeader,
59
+ initialBlockHash: BlockHash,
60
+ l2TipsCache: L2TipsCache,
61
+ ) {
62
+ // Create mocks for L1 clients
63
+ const publicClient = mock<ViemPublicClient>();
64
+ const debugClient = mock<ViemPublicDebugClient>();
65
+ const rollup = mock<RollupContract>();
66
+ const blobClient = mock<BlobClientInterface>();
67
+
68
+ // Mock methods called during start()
69
+ blobClient.testSources.mockResolvedValue();
70
+ publicClient.getBlockNumber.mockResolvedValue(1n);
71
+
72
+ const events = new EventEmitter() as ArchiverEmitter;
73
+ const synchronizer = new NoopL1Synchronizer(instrumentation.tracer);
74
+ const epochCache = mock<EpochCache>();
75
+ epochCache.pipeliningOffset.mockReturnValue(0);
76
+
77
+ super(
78
+ publicClient,
79
+ debugClient,
80
+ rollup,
81
+ {
82
+ rollupAddress: EthAddress.ZERO,
83
+ registryAddress: EthAddress.ZERO,
84
+ inboxAddress: EthAddress.ZERO,
85
+ governanceProposerAddress: EthAddress.ZERO,
86
+ slashingProposerAddress: EthAddress.ZERO,
87
+ },
88
+ dataStores,
89
+ {
90
+ pollingIntervalMs: 1000,
91
+ batchSize: 100,
92
+ skipValidateCheckpointAttestations: true,
93
+ maxAllowedEthClientDriftSeconds: 300,
94
+ ethereumAllowNoDebugHosts: true, // Skip trace validation
95
+ skipHistoricalLogsCheck: true, // Skip historical logs validation
96
+ orphanProposedBlockPruneGraceSeconds: 2,
97
+ },
98
+ blobClient,
99
+ instrumentation,
100
+ { ...l1Constants, l1StartBlockHash: Buffer32.random() },
101
+ synchronizer as ArchiverL1Synchronizer,
102
+ events,
103
+ initialHeader,
104
+ initialBlockHash,
105
+ l2TipsCache,
106
+ epochCache,
107
+ new DateProvider(),
108
+ );
109
+ }
110
+
111
+ /** Override start to skip L1 validation checks. */
112
+ public override start(_blockUntilSynced?: boolean): Promise<void> {
113
+ // Just start the running promise without L1 checks
114
+ this.runningPromise.start();
115
+ return Promise.resolve();
116
+ }
117
+
118
+ /** Always reports as fully synced since there is no real L1 to sync from. */
119
+ public override getSyncedL2SlotNumber(): Promise<SlotNumber | undefined> {
120
+ return Promise.resolve(SlotNumber(Number.MAX_SAFE_INTEGER));
121
+ }
122
+ }
123
+
124
+ /** Creates an archiver with mocked L1 connectivity for testing. */
125
+ export async function createNoopL1Archiver(
126
+ dataStores: ArchiverDataStores,
127
+ l1Constants: L1RollupConstants & { genesisArchiveRoot: Fr },
128
+ telemetry: TelemetryClient = getTelemetryClient(),
129
+ initialHeader: BlockHeader,
130
+ ): Promise<NoopL1Archiver> {
131
+ const instrumentation = await ArchiverInstrumentation.new(telemetry, () => dataStores.db.estimateSize());
132
+ // Mirror the production factory: precompute the dynamic genesis block hash from the injected
133
+ // initial header so `L2TipsCache` reports the correct tip hash at block 0. Without this, the
134
+ // cache falls back to the static `GENESIS_BLOCK_HEADER_HASH`, which only matches deployments
135
+ // with default empty genesis.
136
+ const initialBlockHash = await initialHeader.hash();
137
+ const l2TipsCache = new L2TipsCache(dataStores.blocks, initialBlockHash);
138
+ return new NoopL1Archiver(dataStores, l1Constants, instrumentation, initialHeader, initialBlockHash, l2TipsCache);
139
+ }