@aztec/archiver 0.0.1-commit.96dac018d → 0.0.1-commit.993d240

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 (121) hide show
  1. package/README.md +19 -11
  2. package/dest/archiver.d.ts +36 -17
  3. package/dest/archiver.d.ts.map +1 -1
  4. package/dest/archiver.js +257 -75
  5. package/dest/config.d.ts +6 -3
  6. package/dest/config.d.ts.map +1 -1
  7. package/dest/config.js +23 -15
  8. package/dest/errors.d.ts +55 -9
  9. package/dest/errors.d.ts.map +1 -1
  10. package/dest/errors.js +81 -14
  11. package/dest/factory.d.ts +13 -9
  12. package/dest/factory.d.ts.map +1 -1
  13. package/dest/factory.js +47 -35
  14. package/dest/index.d.ts +11 -3
  15. package/dest/index.d.ts.map +1 -1
  16. package/dest/index.js +10 -2
  17. package/dest/l1/calldata_retriever.d.ts +2 -1
  18. package/dest/l1/calldata_retriever.d.ts.map +1 -1
  19. package/dest/l1/calldata_retriever.js +15 -5
  20. package/dest/l1/data_retrieval.d.ts +24 -12
  21. package/dest/l1/data_retrieval.d.ts.map +1 -1
  22. package/dest/l1/data_retrieval.js +36 -37
  23. package/dest/l1/trace_tx.d.ts +12 -66
  24. package/dest/l1/trace_tx.d.ts.map +1 -1
  25. package/dest/l1/validate_historical_logs.d.ts +23 -0
  26. package/dest/l1/validate_historical_logs.d.ts.map +1 -0
  27. package/dest/l1/validate_historical_logs.js +108 -0
  28. package/dest/modules/contract_data_source_adapter.d.ts +25 -0
  29. package/dest/modules/contract_data_source_adapter.d.ts.map +1 -0
  30. package/dest/modules/contract_data_source_adapter.js +40 -0
  31. package/dest/modules/data_source_base.d.ts +70 -46
  32. package/dest/modules/data_source_base.d.ts.map +1 -1
  33. package/dest/modules/data_source_base.js +270 -135
  34. package/dest/modules/data_store_updater.d.ts +42 -17
  35. package/dest/modules/data_store_updater.d.ts.map +1 -1
  36. package/dest/modules/data_store_updater.js +191 -122
  37. package/dest/modules/instrumentation.d.ts +7 -2
  38. package/dest/modules/instrumentation.d.ts.map +1 -1
  39. package/dest/modules/instrumentation.js +25 -7
  40. package/dest/modules/l1_synchronizer.d.ts +12 -6
  41. package/dest/modules/l1_synchronizer.d.ts.map +1 -1
  42. package/dest/modules/l1_synchronizer.js +432 -205
  43. package/dest/modules/validation.d.ts +4 -3
  44. package/dest/modules/validation.d.ts.map +1 -1
  45. package/dest/modules/validation.js +6 -6
  46. package/dest/store/block_store.d.ts +174 -70
  47. package/dest/store/block_store.d.ts.map +1 -1
  48. package/dest/store/block_store.js +696 -250
  49. package/dest/store/contract_class_store.d.ts +17 -4
  50. package/dest/store/contract_class_store.d.ts.map +1 -1
  51. package/dest/store/contract_class_store.js +24 -68
  52. package/dest/store/contract_instance_store.d.ts +28 -1
  53. package/dest/store/contract_instance_store.d.ts.map +1 -1
  54. package/dest/store/contract_instance_store.js +37 -2
  55. package/dest/store/data_stores.d.ts +68 -0
  56. package/dest/store/data_stores.d.ts.map +1 -0
  57. package/dest/store/data_stores.js +54 -0
  58. package/dest/store/function_names_cache.d.ts +17 -0
  59. package/dest/store/function_names_cache.d.ts.map +1 -0
  60. package/dest/store/function_names_cache.js +30 -0
  61. package/dest/store/l2_tips_cache.d.ts +13 -7
  62. package/dest/store/l2_tips_cache.d.ts.map +1 -1
  63. package/dest/store/l2_tips_cache.js +13 -76
  64. package/dest/store/log_store.d.ts +42 -37
  65. package/dest/store/log_store.d.ts.map +1 -1
  66. package/dest/store/log_store.js +262 -408
  67. package/dest/store/log_store_codec.d.ts +70 -0
  68. package/dest/store/log_store_codec.d.ts.map +1 -0
  69. package/dest/store/log_store_codec.js +101 -0
  70. package/dest/store/message_store.d.ts +11 -1
  71. package/dest/store/message_store.d.ts.map +1 -1
  72. package/dest/store/message_store.js +51 -9
  73. package/dest/test/fake_l1_state.d.ts +20 -1
  74. package/dest/test/fake_l1_state.d.ts.map +1 -1
  75. package/dest/test/fake_l1_state.js +114 -18
  76. package/dest/test/mock_l1_to_l2_message_source.d.ts +1 -1
  77. package/dest/test/mock_l1_to_l2_message_source.d.ts.map +1 -1
  78. package/dest/test/mock_l1_to_l2_message_source.js +2 -1
  79. package/dest/test/mock_l2_block_source.d.ts +52 -46
  80. package/dest/test/mock_l2_block_source.d.ts.map +1 -1
  81. package/dest/test/mock_l2_block_source.js +246 -170
  82. package/dest/test/mock_structs.d.ts +4 -1
  83. package/dest/test/mock_structs.d.ts.map +1 -1
  84. package/dest/test/mock_structs.js +13 -1
  85. package/dest/test/noop_l1_archiver.d.ts +12 -6
  86. package/dest/test/noop_l1_archiver.d.ts.map +1 -1
  87. package/dest/test/noop_l1_archiver.js +26 -9
  88. package/package.json +14 -14
  89. package/src/archiver.ts +313 -75
  90. package/src/config.ts +32 -12
  91. package/src/errors.ts +122 -21
  92. package/src/factory.ts +54 -29
  93. package/src/index.ts +18 -2
  94. package/src/l1/calldata_retriever.ts +16 -5
  95. package/src/l1/data_retrieval.ts +52 -53
  96. package/src/l1/validate_historical_logs.ts +140 -0
  97. package/src/modules/contract_data_source_adapter.ts +55 -0
  98. package/src/modules/data_source_base.ts +336 -171
  99. package/src/modules/data_store_updater.ts +224 -154
  100. package/src/modules/instrumentation.ts +28 -8
  101. package/src/modules/l1_synchronizer.ts +572 -248
  102. package/src/modules/validation.ts +10 -9
  103. package/src/store/block_store.ts +865 -290
  104. package/src/store/contract_class_store.ts +31 -103
  105. package/src/store/contract_instance_store.ts +51 -5
  106. package/src/store/data_stores.ts +104 -0
  107. package/src/store/function_names_cache.ts +37 -0
  108. package/src/store/l2_tips_cache.ts +16 -70
  109. package/src/store/log_store.ts +301 -559
  110. package/src/store/log_store_codec.ts +132 -0
  111. package/src/store/message_store.ts +60 -10
  112. package/src/structs/inbox_message.ts +1 -1
  113. package/src/test/fake_l1_state.ts +142 -29
  114. package/src/test/mock_l1_to_l2_message_source.ts +1 -0
  115. package/src/test/mock_l2_block_source.ts +309 -205
  116. package/src/test/mock_structs.ts +20 -6
  117. package/src/test/noop_l1_archiver.ts +39 -9
  118. package/dest/store/kv_archiver_store.d.ts +0 -354
  119. package/dest/store/kv_archiver_store.d.ts.map +0 -1
  120. package/dest/store/kv_archiver_store.js +0 -464
  121. package/src/store/kv_archiver_store.ts +0 -671
@@ -1,637 +1,379 @@
1
1
  import { INITIAL_L2_BLOCK_NUM } from '@aztec/constants';
2
- import { BlockNumber } from '@aztec/foundation/branded-types';
3
- import { filterAsync } from '@aztec/foundation/collection';
4
2
  import { Fr } from '@aztec/foundation/curves/bn254';
5
3
  import { createLogger } from '@aztec/foundation/log';
6
- import { BufferReader, numToUInt32BE } from '@aztec/foundation/serialize';
7
4
  import type { AztecAsyncKVStore, AztecAsyncMap } from '@aztec/kv-store';
8
- import type { AztecAddress } from '@aztec/stdlib/aztec-address';
9
- import { BlockHash, L2Block } from '@aztec/stdlib/block';
5
+ import type { BlockHash, L2Block } from '@aztec/stdlib/block';
10
6
  import { MAX_LOGS_PER_TAG } from '@aztec/stdlib/interfaces/api-limit';
11
- import type { GetContractClassLogsResponse, GetPublicLogsResponse } from '@aztec/stdlib/interfaces/client';
12
- import {
13
- ContractClassLog,
14
- ExtendedContractClassLog,
15
- ExtendedPublicLog,
16
- type LogFilter,
17
- LogId,
18
- PublicLog,
19
- type SiloedTag,
7
+ import type {
8
+ LogCursor,
9
+ LogResult,
10
+ PrivateLogsQuery,
11
+ PublicLogsQuery,
12
+ SiloedTag,
20
13
  Tag,
21
- TxScopedL2Log,
14
+ TagQuery,
22
15
  } from '@aztec/stdlib/logs';
23
16
  import { TxHash } from '@aztec/stdlib/tx';
24
17
 
25
18
  import type { BlockStore } from './block_store.js';
19
+ import {
20
+ decodeKeyTail,
21
+ decodeValue,
22
+ encodeKey,
23
+ encodePublicPrefix,
24
+ encodeValue,
25
+ endOfTagRange,
26
+ endOfTxRange,
27
+ fieldHex,
28
+ incKey,
29
+ } from './log_store_codec.js';
26
30
 
27
31
  /**
28
- * A store for logs
32
+ * Indexes every emitted private and public log under a composite hex-string key
33
+ * `[contractAddress (public only)]-tag-blockNumber-txIndexWithinBlock-logIndexWithinTx`,
34
+ * where each numeric segment is zero-padded to 8 lowercase hex digits (4 bytes BE) and
35
+ * `contractAddress` / `tag` are the bare 64-hex-char field representations (no `0x` prefix). The
36
+ * fixed-width zero-padded hex segments sort lexicographically in the same order as the canonical
37
+ * `(contract, tag, blockNumber, txIndexWithinBlock, logIndexWithinTx)` tuple, so a single ordered
38
+ * range scan answers every {@link PrivateLogsQuery} / {@link PublicLogsQuery}.
39
+ *
40
+ * Per-block secondary indices (`#privateKeysByBlock`, `#publicKeysByBlock`) record the exact primary
41
+ * keys written for each block so {@link deleteLogs} can drop them on reorg without having to range
42
+ * scan by block (block isn't the leading key segment).
43
+ *
44
+ * Contract-class logs are no longer stored or served by the log store.
29
45
  */
30
46
  export class LogStore {
31
- // `tag` --> private logs
32
- #privateLogsByTag: AztecAsyncMap<string, Buffer[]>;
33
- // `{contractAddress}_${tag}` --> public logs
34
- #publicLogsByContractAndTag: AztecAsyncMap<string, Buffer[]>;
35
- #privateLogKeysByBlock: AztecAsyncMap<number, string[]>;
36
- #publicLogKeysByBlock: AztecAsyncMap<number, string[]>;
37
- #publicLogsByBlock: AztecAsyncMap<number, Buffer>;
38
- #contractClassLogsByBlock: AztecAsyncMap<number, Buffer>;
39
- #logsMaxPageSize: number;
47
+ /** Primary map: composite private key (tag + tail = 96 hex chars + separators) -> serialized {@link StoredLogValue}. */
48
+ #privateLogs: AztecAsyncMap<string, Buffer>;
49
+ /** Primary map: composite public key (contract + tag + tail) -> serialized {@link StoredLogValue}. */
50
+ #publicLogs: AztecAsyncMap<string, Buffer>;
51
+
52
+ /** Secondary deletion index: blockNumber -> the exact primary keys written for that block. */
53
+ #privateKeysByBlock: AztecAsyncMap<number, string[]>;
54
+ #publicKeysByBlock: AztecAsyncMap<number, string[]>;
55
+
40
56
  #log = createLogger('archiver:log_store');
41
57
 
58
+ /**
59
+ * @param genesisBlockHash - Hash of the synthetic genesis block. During early sync the PXE anchors to
60
+ * genesis and passes its hash as a query `referenceBlock`; since the archiver never indexes the
61
+ * genesis block, the store recognizes this hash directly and resolves it to the genesis block number
62
+ * rather than mistaking it for a reorg.
63
+ */
42
64
  constructor(
43
65
  private db: AztecAsyncKVStore,
44
66
  private blockStore: BlockStore,
45
- logsMaxPageSize: number = 1000,
67
+ private readonly genesisBlockHash: BlockHash,
46
68
  ) {
47
- this.#privateLogsByTag = db.openMap('archiver_private_tagged_logs_by_tag');
48
- this.#publicLogsByContractAndTag = db.openMap('archiver_public_tagged_logs_by_tag');
49
- this.#privateLogKeysByBlock = db.openMap('archiver_private_log_keys_by_block');
50
- this.#publicLogKeysByBlock = db.openMap('archiver_public_log_keys_by_block');
51
- this.#publicLogsByBlock = db.openMap('archiver_public_logs_by_block');
52
- this.#contractClassLogsByBlock = db.openMap('archiver_contract_class_logs_by_block');
53
-
54
- this.#logsMaxPageSize = logsMaxPageSize;
69
+ this.#privateLogs = db.openMap('archiver_private_logs');
70
+ this.#publicLogs = db.openMap('archiver_public_logs');
71
+ this.#privateKeysByBlock = db.openMap('archiver_private_log_keys_by_block');
72
+ this.#publicKeysByBlock = db.openMap('archiver_public_log_keys_by_block');
55
73
  }
56
74
 
57
75
  /**
58
- * Extracts tagged logs from a single block, grouping them into private and public maps.
76
+ * Indexes every emitted private and public log from the given blocks. Wraps the write in a single
77
+ * `db.transactionAsync` so the primary entries and the per-block secondary indices stay consistent.
59
78
  *
60
- * @param block - The L2 block to extract logs from.
61
- * @returns An object containing the private and public tagged logs for the block.
62
- */
63
- #extractTaggedLogsFromBlock(block: L2Block) {
64
- // SiloedTag (as string) -> array of log buffers.
65
- const privateTaggedLogs = new Map<string, Buffer[]>();
66
- // "{contractAddress}_{tag}" (as string) -> array of log buffers.
67
- const publicTaggedLogs = new Map<string, Buffer[]>();
68
-
69
- block.body.txEffects.forEach(txEffect => {
70
- const txHash = txEffect.txHash;
71
-
72
- txEffect.privateLogs.forEach(log => {
73
- // Private logs use SiloedTag (already siloed by kernel)
74
- const tag = log.fields[0];
75
- this.#log.debug(`Found private log with tag ${tag.toString()} in block ${block.number}`);
76
-
77
- const currentLogs = privateTaggedLogs.get(tag.toString()) ?? [];
78
- currentLogs.push(
79
- new TxScopedL2Log(
80
- txHash,
81
- block.number,
82
- block.timestamp,
83
- log.getEmittedFields(),
84
- txEffect.noteHashes,
85
- txEffect.nullifiers[0],
86
- ).toBuffer(),
87
- );
88
- privateTaggedLogs.set(tag.toString(), currentLogs);
89
- });
90
-
91
- txEffect.publicLogs.forEach(log => {
92
- // Public logs use Tag directly (not siloed) and are stored with contract address
93
- const tag = log.fields[0];
94
- const contractAddress = log.contractAddress;
95
- const key = `${contractAddress.toString()}_${tag.toString()}`;
96
- this.#log.debug(
97
- `Found public log with tag ${tag.toString()} from contract ${contractAddress.toString()} in block ${block.number}`,
98
- );
99
-
100
- const currentLogs = publicTaggedLogs.get(key) ?? [];
101
- currentLogs.push(
102
- new TxScopedL2Log(
103
- txHash,
104
- block.number,
105
- block.timestamp,
106
- log.getEmittedFields(),
107
- txEffect.noteHashes,
108
- txEffect.nullifiers[0],
109
- ).toBuffer(),
110
- );
111
- publicTaggedLogs.set(key, currentLogs);
112
- });
113
- });
114
-
115
- return { privateTaggedLogs, publicTaggedLogs };
116
- }
117
-
118
- /**
119
- * Extracts and aggregates tagged logs from a list of blocks.
120
- * @param blocks - The blocks to extract logs from.
121
- * @returns A map from tag (as string) to an array of serialized private logs belonging to that tag, and a map from
122
- * "{contractAddress}_{tag}" (as string) to an array of serialized public logs belonging to that key.
79
+ * A block is only ever added once; on reorg the archiver calls {@link deleteLogs} first, so we write
80
+ * the secondary index entries with a plain `set` (overwrite) rather than read-modify-append.
123
81
  */
124
- #extractTaggedLogs(blocks: L2Block[]): {
125
- privateTaggedLogs: Map<string, Buffer[]>;
126
- publicTaggedLogs: Map<string, Buffer[]>;
127
- } {
128
- const taggedLogsInBlocks = blocks.map(block => this.#extractTaggedLogsFromBlock(block));
129
-
130
- // Now we merge the maps from each block into a single map.
131
- const privateTaggedLogs = taggedLogsInBlocks.reduce((acc, { privateTaggedLogs }) => {
132
- for (const [tag, logs] of privateTaggedLogs.entries()) {
133
- const currentLogs = acc.get(tag) ?? [];
134
- acc.set(tag, currentLogs.concat(logs));
135
- }
136
- return acc;
137
- }, new Map<string, Buffer[]>());
138
-
139
- const publicTaggedLogs = taggedLogsInBlocks.reduce((acc, { publicTaggedLogs }) => {
140
- for (const [key, logs] of publicTaggedLogs.entries()) {
141
- const currentLogs = acc.get(key) ?? [];
142
- acc.set(key, currentLogs.concat(logs));
143
- }
144
- return acc;
145
- }, new Map<string, Buffer[]>());
146
-
147
- return { privateTaggedLogs, publicTaggedLogs };
148
- }
149
-
150
- async #addPrivateLogs(blocks: L2Block[]): Promise<void> {
151
- const newBlocks = await filterAsync(
152
- blocks,
153
- async block => !(await this.#privateLogKeysByBlock.hasAsync(block.number)),
154
- );
155
-
156
- const { privateTaggedLogs } = this.#extractTaggedLogs(newBlocks);
157
- const keysOfPrivateLogsToUpdate = Array.from(privateTaggedLogs.keys());
158
-
159
- const currentPrivateTaggedLogs = await Promise.all(
160
- keysOfPrivateLogsToUpdate.map(async key => ({
161
- tag: key,
162
- logBuffers: await this.#privateLogsByTag.getAsync(key),
163
- })),
164
- );
165
-
166
- for (const taggedLogBuffer of currentPrivateTaggedLogs) {
167
- if (taggedLogBuffer.logBuffers && taggedLogBuffer.logBuffers.length > 0) {
168
- privateTaggedLogs.set(
169
- taggedLogBuffer.tag,
170
- taggedLogBuffer.logBuffers!.concat(privateTaggedLogs.get(taggedLogBuffer.tag)!),
171
- );
172
- }
173
- }
174
-
175
- for (const block of newBlocks) {
176
- const privateTagsInBlock: string[] = [];
177
- for (const [tag, logs] of privateTaggedLogs.entries()) {
178
- await this.#privateLogsByTag.set(tag, logs);
179
- privateTagsInBlock.push(tag);
180
- }
181
- await this.#privateLogKeysByBlock.set(block.number, privateTagsInBlock);
182
- }
183
- }
184
-
185
- async #addPublicLogs(blocks: L2Block[]): Promise<void> {
186
- const newBlocks = await filterAsync(
187
- blocks,
188
- async block => !(await this.#publicLogKeysByBlock.hasAsync(block.number)),
189
- );
190
-
191
- const { publicTaggedLogs } = this.#extractTaggedLogs(newBlocks);
192
- const keysOfPublicLogsToUpdate = Array.from(publicTaggedLogs.keys());
82
+ addLogs(blocks: L2Block[]): Promise<boolean> {
83
+ return this.db.transactionAsync(async () => {
84
+ for (const block of blocks) {
85
+ const blockHash = await block.hash();
86
+ const blockNumber = block.number;
87
+ const blockTimestamp = block.timestamp;
88
+
89
+ const privateKeys: string[] = [];
90
+ const privateValues: Buffer[] = [];
91
+ const publicKeys: string[] = [];
92
+ const publicValues: Buffer[] = [];
93
+
94
+ for (let txIndexWithinBlock = 0; txIndexWithinBlock < block.body.txEffects.length; txIndexWithinBlock++) {
95
+ const txEffect = block.body.txEffects[txIndexWithinBlock];
96
+ const txHash = txEffect.txHash;
97
+
98
+ // Private and public log indices are counted independently per tx, each starting at 0.
99
+ let privateLogIndexWithinTx = 0;
100
+ let publicLogIndexWithinTx = 0;
101
+
102
+ for (const log of txEffect.privateLogs) {
103
+ const tagHex = fieldHex(log.fields[0]);
104
+ const key = encodeKey(tagHex, blockNumber, txIndexWithinBlock, privateLogIndexWithinTx);
105
+ const value = encodeValue({
106
+ txHash,
107
+ blockHash,
108
+ blockTimestamp,
109
+ logData: log.getEmittedFields(),
110
+ });
111
+ privateKeys.push(key);
112
+ privateValues.push(value);
113
+ privateLogIndexWithinTx++;
114
+ }
115
+
116
+ for (const log of txEffect.publicLogs) {
117
+ const contractHex = fieldHex(log.contractAddress);
118
+ const tagHex = fieldHex(log.fields[0]);
119
+ const key = encodeKey(
120
+ encodePublicPrefix(contractHex, tagHex),
121
+ blockNumber,
122
+ txIndexWithinBlock,
123
+ publicLogIndexWithinTx,
124
+ );
125
+ const value = encodeValue({
126
+ txHash,
127
+ blockHash,
128
+ blockTimestamp,
129
+ logData: log.getEmittedFields(),
130
+ });
131
+ publicKeys.push(key);
132
+ publicValues.push(value);
133
+ publicLogIndexWithinTx++;
134
+ }
135
+ }
193
136
 
194
- const currentPublicTaggedLogs = await Promise.all(
195
- keysOfPublicLogsToUpdate.map(async key => ({
196
- tag: key,
197
- logBuffers: await this.#publicLogsByContractAndTag.getAsync(key),
198
- })),
199
- );
137
+ for (let i = 0; i < privateKeys.length; i++) {
138
+ await this.#privateLogs.set(privateKeys[i], privateValues[i]);
139
+ }
140
+ for (let i = 0; i < publicKeys.length; i++) {
141
+ await this.#publicLogs.set(publicKeys[i], publicValues[i]);
142
+ }
200
143
 
201
- for (const taggedLogBuffer of currentPublicTaggedLogs) {
202
- if (taggedLogBuffer.logBuffers && taggedLogBuffer.logBuffers.length > 0) {
203
- publicTaggedLogs.set(
204
- taggedLogBuffer.tag,
205
- taggedLogBuffer.logBuffers!.concat(publicTaggedLogs.get(taggedLogBuffer.tag)!),
206
- );
207
- }
208
- }
144
+ await this.#privateKeysByBlock.set(blockNumber, privateKeys);
145
+ await this.#publicKeysByBlock.set(blockNumber, publicKeys);
209
146
 
210
- for (const block of newBlocks) {
211
- const blockHash = await block.hash();
212
- const publicTagsInBlock: string[] = [];
213
- for (const [tag, logs] of publicTaggedLogs.entries()) {
214
- await this.#publicLogsByContractAndTag.set(tag, logs);
215
- publicTagsInBlock.push(tag);
147
+ this.#log.debug(`Indexed logs for block ${blockNumber}`, {
148
+ blockNumber,
149
+ privateCount: privateKeys.length,
150
+ publicCount: publicKeys.length,
151
+ });
216
152
  }
217
- await this.#publicLogKeysByBlock.set(block.number, publicTagsInBlock);
218
-
219
- const publicLogsInBlock = block.body.txEffects
220
- .map((txEffect, txIndex) =>
221
- [
222
- numToUInt32BE(txIndex),
223
- txEffect.txHash.toBuffer(),
224
- numToUInt32BE(txEffect.publicLogs.length),
225
- txEffect.publicLogs.map(log => log.toBuffer()),
226
- ].flat(),
227
- )
228
- .flat();
229
-
230
- await this.#publicLogsByBlock.set(block.number, this.#packWithBlockHash(blockHash, publicLogsInBlock));
231
- }
232
- }
233
-
234
- async #addContractClassLogs(blocks: L2Block[]): Promise<void> {
235
- const newBlocks = await filterAsync(
236
- blocks,
237
- async block => !(await this.#contractClassLogsByBlock.hasAsync(block.number)),
238
- );
239
-
240
- for (const block of newBlocks) {
241
- const blockHash = await block.hash();
242
-
243
- const contractClassLogsInBlock = block.body.txEffects
244
- .map((txEffect, txIndex) =>
245
- [
246
- numToUInt32BE(txIndex),
247
- txEffect.txHash.toBuffer(),
248
- numToUInt32BE(txEffect.contractClassLogs.length),
249
- txEffect.contractClassLogs.map(log => log.toBuffer()),
250
- ].flat(),
251
- )
252
- .flat();
253
-
254
- await this.#contractClassLogsByBlock.set(
255
- block.number,
256
- this.#packWithBlockHash(blockHash, contractClassLogsInBlock),
257
- );
258
- }
259
- }
260
-
261
- /**
262
- * Append new logs to the store's list.
263
- * @param blocks - The blocks for which to add the logs.
264
- * @returns True if the operation is successful.
265
- */
266
- addLogs(blocks: L2Block[]): Promise<boolean> {
267
- return this.db.transactionAsync(async () => {
268
- await Promise.all([
269
- this.#addPrivateLogs(blocks),
270
- this.#addPublicLogs(blocks),
271
- this.#addContractClassLogs(blocks),
272
- ]);
273
153
  return true;
274
154
  });
275
155
  }
276
156
 
277
- #packWithBlockHash(blockHash: BlockHash, data: Buffer<ArrayBufferLike>[]): Buffer<ArrayBufferLike> {
278
- return Buffer.concat([blockHash.toBuffer(), ...data]);
279
- }
280
-
281
- #unpackBlockHash(reader: BufferReader): BlockHash {
282
- const blockHash = reader.remainingBytes() > 0 ? reader.readObject(Fr) : undefined;
283
-
284
- if (!blockHash) {
285
- throw new Error('Failed to read block hash from log entry buffer');
286
- }
287
-
288
- return new BlockHash(blockHash);
289
- }
290
-
157
+ /**
158
+ * Deletes every log indexed under any of the given blocks. Secondary-index driven, so it doesn't
159
+ * have to range-scan the primary maps.
160
+ */
291
161
  deleteLogs(blocks: L2Block[]): Promise<boolean> {
292
162
  return this.db.transactionAsync(async () => {
293
- await Promise.all(
294
- blocks.map(async block => {
295
- // Delete private logs
296
- const privateKeys = (await this.#privateLogKeysByBlock.getAsync(block.number)) ?? [];
297
- await Promise.all(privateKeys.map(tag => this.#privateLogsByTag.delete(tag)));
298
-
299
- // Delete public logs
300
- const publicKeys = (await this.#publicLogKeysByBlock.getAsync(block.number)) ?? [];
301
- await Promise.all(publicKeys.map(key => this.#publicLogsByContractAndTag.delete(key)));
302
- }),
303
- );
304
-
305
- await Promise.all(
306
- blocks.map(block =>
307
- Promise.all([
308
- this.#publicLogsByBlock.delete(block.number),
309
- this.#privateLogKeysByBlock.delete(block.number),
310
- this.#publicLogKeysByBlock.delete(block.number),
311
- this.#contractClassLogsByBlock.delete(block.number),
312
- ]),
313
- ),
314
- );
315
-
163
+ for (const block of blocks) {
164
+ const blockNumber = block.number;
165
+
166
+ const [privateKeys, publicKeys] = await Promise.all([
167
+ this.#privateKeysByBlock.getAsync(blockNumber),
168
+ this.#publicKeysByBlock.getAsync(blockNumber),
169
+ ]);
170
+
171
+ if (privateKeys) {
172
+ for (const key of privateKeys) {
173
+ await this.#privateLogs.delete(key);
174
+ }
175
+ await this.#privateKeysByBlock.delete(blockNumber);
176
+ }
177
+ if (publicKeys) {
178
+ for (const key of publicKeys) {
179
+ await this.#publicLogs.delete(key);
180
+ }
181
+ await this.#publicKeysByBlock.delete(blockNumber);
182
+ }
183
+ }
316
184
  return true;
317
185
  });
318
186
  }
319
187
 
320
- /**
321
- * Gets private logs that match any of the `tags`. For each tag, an array of matching logs is returned. An empty
322
- * array implies no logs match that tag.
323
- * @param tags - The tags to search for.
324
- * @param page - The page number (0-indexed) for pagination.
325
- * @returns An array of log arrays, one per tag. Returns at most MAX_LOGS_PER_TAG logs per tag per page. If
326
- * MAX_LOGS_PER_TAG logs are returned for a tag, the caller should fetch the next page to check for more logs.
327
- */
328
- async getPrivateLogsByTags(tags: SiloedTag[], page: number = 0): Promise<TxScopedL2Log[][]> {
329
- const logs = await Promise.all(tags.map(tag => this.#privateLogsByTag.getAsync(tag.toString())));
330
- const start = page * MAX_LOGS_PER_TAG;
331
- const end = start + MAX_LOGS_PER_TAG;
332
-
333
- return logs.map(
334
- logBuffers => logBuffers?.slice(start, end).map(logBuffer => TxScopedL2Log.fromBuffer(logBuffer)) ?? [],
335
- );
188
+ /** Returns one inner array per element of `query.tags`, in input order. */
189
+ getPrivateLogsByTags(query: PrivateLogsQuery): Promise<LogResult[][]> {
190
+ LogStore.#validateQuery(query);
191
+ return this.db.transactionAsync(() => this.#runQuery(query, /* contractHex */ undefined));
336
192
  }
337
193
 
338
- /**
339
- * Gets public logs that match any of the `tags` from the specified contract. For each tag, an array of matching
340
- * logs is returned. An empty array implies no logs match that tag.
341
- * @param contractAddress - The contract address to search logs for.
342
- * @param tags - The tags to search for.
343
- * @param page - The page number (0-indexed) for pagination.
344
- * @returns An array of log arrays, one per tag. Returns at most MAX_LOGS_PER_TAG logs per tag per page. If
345
- * MAX_LOGS_PER_TAG logs are returned for a tag, the caller should fetch the next page to check for more logs.
346
- */
347
- async getPublicLogsByTagsFromContract(
348
- contractAddress: AztecAddress,
349
- tags: Tag[],
350
- page: number = 0,
351
- ): Promise<TxScopedL2Log[][]> {
352
- const logs = await Promise.all(
353
- tags.map(tag => {
354
- const key = `${contractAddress.toString()}_${tag.value.toString()}`;
355
- return this.#publicLogsByContractAndTag.getAsync(key);
356
- }),
357
- );
358
- const start = page * MAX_LOGS_PER_TAG;
359
- const end = start + MAX_LOGS_PER_TAG;
360
-
361
- return logs.map(
362
- logBuffers => logBuffers?.slice(start, end).map(logBuffer => TxScopedL2Log.fromBuffer(logBuffer)) ?? [],
363
- );
194
+ /** Returns one inner array per element of `query.tags`, in input order. */
195
+ getPublicLogsByTags(query: PublicLogsQuery): Promise<LogResult[][]> {
196
+ LogStore.#validateQuery(query);
197
+ return this.db.transactionAsync(() => this.#runQuery(query, fieldHex(query.contractAddress)));
364
198
  }
365
199
 
366
- /**
367
- * Gets public logs based on the provided filter.
368
- * @param filter - The filter to apply to the logs.
369
- * @returns The requested logs.
370
- */
371
- getPublicLogs(filter: LogFilter): Promise<GetPublicLogsResponse> {
372
- if (filter.afterLog) {
373
- return this.#filterPublicLogsBetweenBlocks(filter);
374
- } else if (filter.txHash) {
375
- return this.#filterPublicLogsOfTx(filter);
376
- } else {
377
- return this.#filterPublicLogsBetweenBlocks(filter);
200
+ static #validateQuery(query: { txHash?: TxHash; fromBlock?: unknown; toBlock?: unknown }): void {
201
+ if (query.txHash !== undefined && (query.fromBlock !== undefined || query.toBlock !== undefined)) {
202
+ throw new Error('`txHash` is mutually exclusive with `fromBlock`/`toBlock`');
378
203
  }
379
204
  }
380
205
 
381
- async #filterPublicLogsOfTx(filter: LogFilter): Promise<GetPublicLogsResponse> {
382
- if (!filter.txHash) {
383
- throw new Error('Missing txHash');
206
+ async #runQuery(query: PrivateLogsQuery | PublicLogsQuery, contractHex: string | undefined): Promise<LogResult[][]> {
207
+ const isPublic = contractHex !== undefined;
208
+ const tags = (query.tags as ReadonlyArray<TagQuery<Tag | SiloedTag>>) ?? [];
209
+ const primaryMap = isPublic ? this.#publicLogs : this.#privateLogs;
210
+
211
+ // referenceBlock reorg check, in-transaction, against the same db the log primary maps live on. The
212
+ // genesis block is a valid anchor during early sync but is synthetic and never indexed in the block
213
+ // store, so resolve it directly to the genesis block number rather than mistaking it for a reorg.
214
+ let referenceBlockNumber: number | undefined;
215
+ if (query.referenceBlock) {
216
+ if (query.referenceBlock.equals(this.genesisBlockHash)) {
217
+ referenceBlockNumber = INITIAL_L2_BLOCK_NUM - 1;
218
+ } else {
219
+ const refBlk = await this.blockStore.getBlockData({ hash: query.referenceBlock });
220
+ if (!refBlk) {
221
+ throw new Error(
222
+ `Reference block ${query.referenceBlock.toString()} not found in the node. This might indicate a reorg has occurred.`,
223
+ );
224
+ }
225
+ referenceBlockNumber = refBlk.header.globalVariables.blockNumber;
226
+ }
384
227
  }
385
228
 
386
- const [blockNumber, txIndex] = (await this.blockStore.getTxLocation(filter.txHash)) ?? [];
387
- if (typeof blockNumber !== 'number' || typeof txIndex !== 'number') {
388
- return { logs: [], maxLogsHit: false };
229
+ // Compute the exclusive upper-block bound across `toBlock` and `referenceBlock`.
230
+ // `toBlock` is already exclusive; `referenceBlock` caps inclusively, so its exclusive form is +1.
231
+ let upperExclusive: number | undefined;
232
+ if (query.toBlock !== undefined) {
233
+ upperExclusive = query.toBlock;
234
+ }
235
+ if (referenceBlockNumber !== undefined) {
236
+ const refExclusive = referenceBlockNumber + 1;
237
+ upperExclusive = upperExclusive === undefined ? refExclusive : Math.min(upperExclusive, refExclusive);
389
238
  }
390
239
 
391
- const buffer = (await this.#publicLogsByBlock.getAsync(blockNumber)) ?? Buffer.alloc(0);
392
- const publicLogsInBlock: { txHash: TxHash; logs: PublicLog[] }[] = [];
393
- const reader = new BufferReader(buffer);
394
-
395
- const blockHash = this.#unpackBlockHash(reader);
396
-
397
- while (reader.remainingBytes() > 0) {
398
- const indexOfTx = reader.readNumber();
399
- const txHash = reader.readObject(TxHash);
400
- const numLogsInTx = reader.readNumber();
401
- publicLogsInBlock[indexOfTx] = { txHash, logs: [] };
402
- for (let i = 0; i < numLogsInTx; i++) {
403
- publicLogsInBlock[indexOfTx].logs.push(reader.readObject(PublicLog));
240
+ // Resolve txHash -> (blockNumber, txIndexInBlock) once for the whole query.
241
+ let txLocation: [number, number] | undefined;
242
+ if (query.txHash) {
243
+ const loc = await this.blockStore.getTxLocation(query.txHash);
244
+ if (!loc) {
245
+ return tags.map(() => []);
246
+ }
247
+ txLocation = loc;
248
+ if (upperExclusive !== undefined && txLocation[0] >= upperExclusive) {
249
+ return tags.map(() => []);
404
250
  }
405
251
  }
406
252
 
407
- const txData = publicLogsInBlock[txIndex];
408
-
409
- const logs: ExtendedPublicLog[] = [];
410
- const maxLogsHit = this.#accumulatePublicLogs(
411
- logs,
412
- blockNumber,
413
- blockHash,
414
- txIndex,
415
- txData.txHash,
416
- txData.logs,
417
- filter,
418
- );
419
-
420
- return { logs, maxLogsHit };
421
- }
422
-
423
- async #filterPublicLogsBetweenBlocks(filter: LogFilter): Promise<GetPublicLogsResponse> {
424
- const start =
425
- filter.afterLog?.blockNumber ?? Math.max(filter.fromBlock ?? INITIAL_L2_BLOCK_NUM, INITIAL_L2_BLOCK_NUM);
426
- const end = filter.toBlock;
253
+ const fromBlock = query.fromBlock ?? INITIAL_L2_BLOCK_NUM;
254
+ const includeEffects = query.includeEffects === true;
255
+
256
+ const perTagResults: LogResult[][] = [];
257
+ for (const tagEntry of tags) {
258
+ const { tagHex, afterLog } = normalizeTagEntry(tagEntry);
259
+ const prefix = contractHex !== undefined ? encodePublicPrefix(contractHex, tagHex) : tagHex;
260
+
261
+ const end = txLocation
262
+ ? endOfTxRange(prefix, txLocation[0], txLocation[1])
263
+ : endOfTagRange(prefix, upperExclusive);
264
+
265
+ let start: string;
266
+ if (afterLog) {
267
+ // Cursor wins as the start; `fromBlock` is ignored (fine if the cursor sits below it). The cursor
268
+ // carries `(blockNumber, txIndexWithinBlock, logIndexWithinTx)`, which slot directly into the
269
+ // composite key no tx-hash lookup needed.
270
+ start = incKey(encodeKey(prefix, afterLog.blockNumber, afterLog.txIndexWithinBlock, afterLog.logIndexWithinTx));
271
+ } else if (txLocation) {
272
+ start = encodeKey(prefix, txLocation[0], txLocation[1], 0);
273
+ } else {
274
+ start = encodeKey(prefix, fromBlock, 0, 0);
275
+ }
427
276
 
428
- if (typeof end === 'number' && end < start) {
429
- return {
430
- logs: [],
431
- maxLogsHit: true,
432
- };
277
+ const limit = query.limitPerTag ?? MAX_LOGS_PER_TAG;
278
+ const out: LogResult[] = [];
279
+ for await (const [rawKey, rawVal] of primaryMap.entriesAsync({ start, end, limit })) {
280
+ const tail = decodeKeyTail(rawKey);
281
+ const value = decodeValue(rawVal);
282
+ out.push({
283
+ logData: value.logData,
284
+ blockNumber: tail.blockNumber,
285
+ blockHash: value.blockHash,
286
+ blockTimestamp: value.blockTimestamp,
287
+ txHash: value.txHash,
288
+ txIndexWithinBlock: tail.txIndexWithinBlock,
289
+ logIndexWithinTx: tail.logIndexWithinTx,
290
+ });
291
+ }
292
+ perTagResults.push(out);
433
293
  }
434
294
 
435
- const logs: ExtendedPublicLog[] = [];
436
-
437
- let maxLogsHit = false;
438
- loopOverBlocks: for await (const [blockNumber, logBuffer] of this.#publicLogsByBlock.entriesAsync({ start, end })) {
439
- const publicLogsInBlock: { txHash: TxHash; logs: PublicLog[] }[] = [];
440
- const reader = new BufferReader(logBuffer);
441
-
442
- const blockHash = this.#unpackBlockHash(reader);
443
-
444
- while (reader.remainingBytes() > 0) {
445
- const indexOfTx = reader.readNumber();
446
- const txHash = reader.readObject(TxHash);
447
- const numLogsInTx = reader.readNumber();
448
- publicLogsInBlock[indexOfTx] = { txHash, logs: [] };
449
- for (let i = 0; i < numLogsInTx; i++) {
450
- publicLogsInBlock[indexOfTx].logs.push(reader.readObject(PublicLog));
295
+ if (includeEffects) {
296
+ // Dedupe by txHash across the entire page so a tx with many tagged logs costs one fetch.
297
+ const txHashByKey = new Map<string, TxHash>();
298
+ for (const arr of perTagResults) {
299
+ for (const log of arr) {
300
+ txHashByKey.set(log.txHash.toString(), log.txHash);
451
301
  }
452
302
  }
453
- for (let txIndex = filter.afterLog?.txIndex ?? 0; txIndex < publicLogsInBlock.length; txIndex++) {
454
- const txData = publicLogsInBlock[txIndex];
455
- maxLogsHit = this.#accumulatePublicLogs(
456
- logs,
457
- blockNumber,
458
- blockHash,
459
- txIndex,
460
- txData.txHash,
461
- txData.logs,
462
- filter,
463
- );
464
- if (maxLogsHit) {
465
- this.#log.debug(`Max logs hit at block ${blockNumber}`);
466
- break loopOverBlocks;
303
+ const uniqueTxs = Array.from(txHashByKey.values());
304
+ if (uniqueTxs.length > 0) {
305
+ const effects = await this.blockStore.getNoteHashesAndNullifiers(uniqueTxs);
306
+ const byTxHash = new Map<string, [Fr[], Fr[]]>();
307
+ uniqueTxs.forEach((tx, i) => byTxHash.set(tx.toString(), effects[i]));
308
+ for (let i = 0; i < perTagResults.length; i++) {
309
+ perTagResults[i] = perTagResults[i].map(log => {
310
+ const [noteHashes, nullifiers] = byTxHash.get(log.txHash.toString()) ?? [[], []];
311
+ return { ...log, noteHashes, nullifiers };
312
+ });
467
313
  }
468
314
  }
469
315
  }
470
316
 
471
- return { logs, maxLogsHit };
317
+ return perTagResults;
472
318
  }
473
319
 
474
320
  /**
475
- * Gets contract class logs based on the provided filter.
476
- * @param filter - The filter to apply to the logs.
477
- * @returns The requested logs.
321
+ * Reads back every private log indexed for the given block via the per-block secondary index. Order
322
+ * matches the canonical composite-key order (`tag`, `blockNumber`, `txIndexWithinBlock`,
323
+ * `logIndexWithinTx`). Used by the data-store-updater test suite to verify the indexed-vs-block-body
324
+ * counts without depending on the removed `getPublicLogs(LogFilter)` API.
478
325
  */
479
- getContractClassLogs(filter: LogFilter): Promise<GetContractClassLogsResponse> {
480
- if (filter.afterLog) {
481
- return this.#filterContractClassLogsBetweenBlocks(filter);
482
- } else if (filter.txHash) {
483
- return this.#filterContractClassLogsOfTx(filter);
484
- } else {
485
- return this.#filterContractClassLogsBetweenBlocks(filter);
486
- }
487
- }
488
-
489
- async #filterContractClassLogsOfTx(filter: LogFilter): Promise<GetContractClassLogsResponse> {
490
- if (!filter.txHash) {
491
- throw new Error('Missing txHash');
492
- }
493
-
494
- const [blockNumber, txIndex] = (await this.blockStore.getTxLocation(filter.txHash)) ?? [];
495
- if (typeof blockNumber !== 'number' || typeof txIndex !== 'number') {
496
- return { logs: [], maxLogsHit: false };
497
- }
498
- const contractClassLogsBuffer = (await this.#contractClassLogsByBlock.getAsync(blockNumber)) ?? Buffer.alloc(0);
499
- const contractClassLogsInBlock: { txHash: TxHash; logs: ContractClassLog[] }[] = [];
500
-
501
- const reader = new BufferReader(contractClassLogsBuffer);
502
- const blockHash = this.#unpackBlockHash(reader);
503
-
504
- while (reader.remainingBytes() > 0) {
505
- const indexOfTx = reader.readNumber();
506
- const txHash = reader.readObject(TxHash);
507
- const numLogsInTx = reader.readNumber();
508
- contractClassLogsInBlock[indexOfTx] = { txHash, logs: [] };
509
- for (let i = 0; i < numLogsInTx; i++) {
510
- contractClassLogsInBlock[indexOfTx].logs.push(reader.readObject(ContractClassLog));
511
- }
512
- }
513
-
514
- const txData = contractClassLogsInBlock[txIndex];
515
-
516
- const logs: ExtendedContractClassLog[] = [];
517
- const maxLogsHit = this.#accumulateContractClassLogs(
518
- logs,
519
- blockNumber,
520
- blockHash,
521
- txIndex,
522
- txData.txHash,
523
- txData.logs,
524
- filter,
326
+ getPrivateLogsForBlock(blockNumber: number): Promise<LogResult[]> {
327
+ return this.db.transactionAsync(() =>
328
+ this.#readBlockLogs(this.#privateKeysByBlock, this.#privateLogs, blockNumber),
525
329
  );
526
-
527
- return { logs, maxLogsHit };
528
330
  }
529
331
 
530
- async #filterContractClassLogsBetweenBlocks(filter: LogFilter): Promise<GetContractClassLogsResponse> {
531
- const start =
532
- filter.afterLog?.blockNumber ?? Math.max(filter.fromBlock ?? INITIAL_L2_BLOCK_NUM, INITIAL_L2_BLOCK_NUM);
533
- const end = filter.toBlock;
534
-
535
- if (typeof end === 'number' && end < start) {
536
- return {
537
- logs: [],
538
- maxLogsHit: true,
539
- };
540
- }
541
-
542
- const logs: ExtendedContractClassLog[] = [];
543
-
544
- let maxLogsHit = false;
545
- loopOverBlocks: for await (const [blockNumber, logBuffer] of this.#contractClassLogsByBlock.entriesAsync({
546
- start,
547
- end,
548
- })) {
549
- const contractClassLogsInBlock: { txHash: TxHash; logs: ContractClassLog[] }[] = [];
550
- const reader = new BufferReader(logBuffer);
551
- const blockHash = this.#unpackBlockHash(reader);
552
- while (reader.remainingBytes() > 0) {
553
- const indexOfTx = reader.readNumber();
554
- const txHash = reader.readObject(TxHash);
555
- const numLogsInTx = reader.readNumber();
556
- contractClassLogsInBlock[indexOfTx] = { txHash, logs: [] };
557
- for (let i = 0; i < numLogsInTx; i++) {
558
- contractClassLogsInBlock[indexOfTx].logs.push(reader.readObject(ContractClassLog));
559
- }
560
- }
561
- for (let txIndex = filter.afterLog?.txIndex ?? 0; txIndex < contractClassLogsInBlock.length; txIndex++) {
562
- const txData = contractClassLogsInBlock[txIndex];
563
- maxLogsHit = this.#accumulateContractClassLogs(
564
- logs,
565
- blockNumber,
566
- blockHash,
567
- txIndex,
568
- txData.txHash,
569
- txData.logs,
570
- filter,
571
- );
572
- if (maxLogsHit) {
573
- this.#log.debug(`Max logs hit at block ${blockNumber}`);
574
- break loopOverBlocks;
575
- }
576
- }
577
- }
578
-
579
- return { logs, maxLogsHit };
332
+ /** {@inheritDoc LogStore.getPrivateLogsForBlock} */
333
+ getPublicLogsForBlock(blockNumber: number): Promise<LogResult[]> {
334
+ return this.db.transactionAsync(() => this.#readBlockLogs(this.#publicKeysByBlock, this.#publicLogs, blockNumber));
580
335
  }
581
336
 
582
- #accumulatePublicLogs(
583
- results: ExtendedPublicLog[],
337
+ async #readBlockLogs(
338
+ keysByBlock: AztecAsyncMap<number, string[]>,
339
+ primaryMap: AztecAsyncMap<string, Buffer>,
584
340
  blockNumber: number,
585
- blockHash: BlockHash,
586
- txIndex: number,
587
- txHash: TxHash,
588
- txLogs: PublicLog[],
589
- filter: LogFilter = {},
590
- ): boolean {
591
- let maxLogsHit = false;
592
- let logIndex = typeof filter.afterLog?.logIndex === 'number' ? filter.afterLog.logIndex + 1 : 0;
593
- for (; logIndex < txLogs.length; logIndex++) {
594
- const log = txLogs[logIndex];
595
- if (!filter.contractAddress || log.contractAddress.equals(filter.contractAddress)) {
596
- results.push(
597
- new ExtendedPublicLog(new LogId(BlockNumber(blockNumber), blockHash, txHash, txIndex, logIndex), log),
598
- );
599
-
600
- if (results.length >= this.#logsMaxPageSize) {
601
- maxLogsHit = true;
602
- break;
603
- }
604
- }
341
+ ): Promise<LogResult[]> {
342
+ const keys = await keysByBlock.getAsync(blockNumber);
343
+ if (!keys || keys.length === 0) {
344
+ return [];
605
345
  }
606
-
607
- return maxLogsHit;
608
- }
609
-
610
- #accumulateContractClassLogs(
611
- results: ExtendedContractClassLog[],
612
- blockNumber: number,
613
- blockHash: BlockHash,
614
- txIndex: number,
615
- txHash: TxHash,
616
- txLogs: ContractClassLog[],
617
- filter: LogFilter = {},
618
- ): boolean {
619
- let maxLogsHit = false;
620
- let logIndex = typeof filter.afterLog?.logIndex === 'number' ? filter.afterLog.logIndex + 1 : 0;
621
- for (; logIndex < txLogs.length; logIndex++) {
622
- const log = txLogs[logIndex];
623
- if (!filter.contractAddress || log.contractAddress.equals(filter.contractAddress)) {
624
- results.push(
625
- new ExtendedContractClassLog(new LogId(BlockNumber(blockNumber), blockHash, txHash, txIndex, logIndex), log),
626
- );
627
-
628
- if (results.length >= this.#logsMaxPageSize) {
629
- maxLogsHit = true;
630
- break;
631
- }
346
+ const results: LogResult[] = [];
347
+ for (const key of keys) {
348
+ const raw = await primaryMap.getAsync(key);
349
+ if (!raw) {
350
+ continue;
632
351
  }
352
+ const tail = decodeKeyTail(key);
353
+ const value = decodeValue(raw);
354
+ results.push({
355
+ logData: value.logData,
356
+ blockNumber: tail.blockNumber,
357
+ blockHash: value.blockHash,
358
+ blockTimestamp: value.blockTimestamp,
359
+ txHash: value.txHash,
360
+ txIndexWithinBlock: tail.txIndexWithinBlock,
361
+ logIndexWithinTx: tail.logIndexWithinTx,
362
+ });
633
363
  }
364
+ return results;
365
+ }
366
+ }
634
367
 
635
- return maxLogsHit;
368
+ /** Pulls `{ tagHex, afterLog }` out of a {@link TagQuery}, normalizing the bare-tag form. */
369
+ function normalizeTagEntry<T extends Tag | SiloedTag>(
370
+ entry: TagQuery<T>,
371
+ ): {
372
+ tagHex: string;
373
+ afterLog: LogCursor | undefined;
374
+ } {
375
+ if (typeof entry === 'object' && entry !== null && 'tag' in entry) {
376
+ return { tagHex: fieldHex(entry.tag.value), afterLog: entry.afterLog };
636
377
  }
378
+ return { tagHex: fieldHex((entry as T).value), afterLog: undefined };
637
379
  }