@aztec/pxe 0.0.1-commit.e61ad554 → 0.0.1-commit.ec5f612

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 (205) hide show
  1. package/dest/access_scopes.d.ts +9 -0
  2. package/dest/access_scopes.d.ts.map +1 -0
  3. package/dest/access_scopes.js +6 -0
  4. package/dest/block_synchronizer/block_synchronizer.d.ts +5 -3
  5. package/dest/block_synchronizer/block_synchronizer.d.ts.map +1 -1
  6. package/dest/block_synchronizer/block_synchronizer.js +11 -5
  7. package/dest/config/package_info.js +1 -1
  8. package/dest/contract_function_simulator/contract_function_simulator.d.ts +54 -30
  9. package/dest/contract_function_simulator/contract_function_simulator.d.ts.map +1 -1
  10. package/dest/contract_function_simulator/contract_function_simulator.js +174 -70
  11. package/dest/contract_function_simulator/execution_tagging_index_cache.d.ts +5 -5
  12. package/dest/contract_function_simulator/execution_tagging_index_cache.d.ts.map +1 -1
  13. package/dest/contract_function_simulator/execution_tagging_index_cache.js +3 -3
  14. package/dest/contract_function_simulator/noir-structs/event_validation_request.js +1 -1
  15. package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts +2 -2
  16. package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts.map +1 -1
  17. package/dest/contract_function_simulator/noir-structs/note_validation_request.js +1 -1
  18. package/dest/contract_function_simulator/oracle/interfaces.d.ts +10 -10
  19. package/dest/contract_function_simulator/oracle/interfaces.d.ts.map +1 -1
  20. package/dest/contract_function_simulator/oracle/oracle.d.ts +5 -5
  21. package/dest/contract_function_simulator/oracle/oracle.d.ts.map +1 -1
  22. package/dest/contract_function_simulator/oracle/oracle.js +38 -26
  23. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts +36 -36
  24. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts.map +1 -1
  25. package/dest/contract_function_simulator/oracle/private_execution_oracle.js +74 -29
  26. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts +53 -29
  27. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts.map +1 -1
  28. package/dest/contract_function_simulator/oracle/utility_execution_oracle.js +84 -63
  29. package/dest/contract_logging.d.ts +22 -0
  30. package/dest/contract_logging.d.ts.map +1 -0
  31. package/dest/contract_logging.js +23 -0
  32. package/dest/contract_sync/contract_sync_service.d.ts +43 -0
  33. package/dest/contract_sync/contract_sync_service.d.ts.map +1 -0
  34. package/dest/contract_sync/contract_sync_service.js +97 -0
  35. package/dest/contract_sync/helpers.d.ts +29 -0
  36. package/dest/contract_sync/helpers.d.ts.map +1 -0
  37. package/dest/contract_sync/{index.js → helpers.js} +13 -12
  38. package/dest/debug/pxe_debug_utils.d.ts +24 -10
  39. package/dest/debug/pxe_debug_utils.d.ts.map +1 -1
  40. package/dest/debug/pxe_debug_utils.js +28 -18
  41. package/dest/entrypoints/client/bundle/index.d.ts +4 -1
  42. package/dest/entrypoints/client/bundle/index.d.ts.map +1 -1
  43. package/dest/entrypoints/client/bundle/index.js +3 -0
  44. package/dest/entrypoints/client/bundle/utils.d.ts +1 -1
  45. package/dest/entrypoints/client/bundle/utils.d.ts.map +1 -1
  46. package/dest/entrypoints/client/bundle/utils.js +21 -7
  47. package/dest/entrypoints/client/lazy/index.d.ts +4 -1
  48. package/dest/entrypoints/client/lazy/index.d.ts.map +1 -1
  49. package/dest/entrypoints/client/lazy/index.js +3 -0
  50. package/dest/entrypoints/client/lazy/utils.d.ts +2 -2
  51. package/dest/entrypoints/client/lazy/utils.d.ts.map +1 -1
  52. package/dest/entrypoints/client/lazy/utils.js +22 -8
  53. package/dest/entrypoints/pxe_creation_options.d.ts +3 -2
  54. package/dest/entrypoints/pxe_creation_options.d.ts.map +1 -1
  55. package/dest/entrypoints/server/index.d.ts +4 -2
  56. package/dest/entrypoints/server/index.d.ts.map +1 -1
  57. package/dest/entrypoints/server/index.js +3 -1
  58. package/dest/entrypoints/server/utils.d.ts +1 -1
  59. package/dest/entrypoints/server/utils.d.ts.map +1 -1
  60. package/dest/entrypoints/server/utils.js +28 -9
  61. package/dest/events/event_service.d.ts +4 -5
  62. package/dest/events/event_service.d.ts.map +1 -1
  63. package/dest/events/event_service.js +5 -6
  64. package/dest/job_coordinator/job_coordinator.d.ts +3 -2
  65. package/dest/job_coordinator/job_coordinator.d.ts.map +1 -1
  66. package/dest/job_coordinator/job_coordinator.js +3 -2
  67. package/dest/logs/log_service.d.ts +7 -5
  68. package/dest/logs/log_service.d.ts.map +1 -1
  69. package/dest/logs/log_service.js +19 -29
  70. package/dest/notes/note_service.d.ts +7 -7
  71. package/dest/notes/note_service.d.ts.map +1 -1
  72. package/dest/notes/note_service.js +9 -9
  73. package/dest/notes_filter.d.ts +25 -0
  74. package/dest/notes_filter.d.ts.map +1 -0
  75. package/dest/notes_filter.js +4 -0
  76. package/dest/oracle_version.d.ts +3 -3
  77. package/dest/oracle_version.d.ts.map +1 -1
  78. package/dest/oracle_version.js +2 -2
  79. package/dest/private_kernel/hints/compute_tx_expiration_timestamp.d.ts +4 -0
  80. package/dest/private_kernel/hints/compute_tx_expiration_timestamp.d.ts.map +1 -0
  81. package/dest/private_kernel/hints/{compute_tx_include_by_timestamp.js → compute_tx_expiration_timestamp.js} +12 -12
  82. package/dest/private_kernel/hints/index.d.ts +1 -1
  83. package/dest/private_kernel/hints/index.js +1 -1
  84. package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.d.ts +4 -3
  85. package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.d.ts.map +1 -1
  86. package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.js +129 -68
  87. package/dest/private_kernel/hints/test_utils.d.ts +122 -0
  88. package/dest/private_kernel/hints/test_utils.d.ts.map +1 -0
  89. package/dest/private_kernel/hints/test_utils.js +203 -0
  90. package/dest/private_kernel/private_kernel_execution_prover.d.ts +3 -2
  91. package/dest/private_kernel/private_kernel_execution_prover.d.ts.map +1 -1
  92. package/dest/private_kernel/private_kernel_execution_prover.js +21 -13
  93. package/dest/private_kernel/private_kernel_oracle.d.ts +8 -4
  94. package/dest/private_kernel/private_kernel_oracle.d.ts.map +1 -1
  95. package/dest/private_kernel/private_kernel_oracle.js +7 -3
  96. package/dest/pxe.d.ts +69 -23
  97. package/dest/pxe.d.ts.map +1 -1
  98. package/dest/pxe.js +98 -63
  99. package/dest/storage/address_store/address_store.d.ts +1 -1
  100. package/dest/storage/address_store/address_store.d.ts.map +1 -1
  101. package/dest/storage/address_store/address_store.js +12 -11
  102. package/dest/storage/anchor_block_store/anchor_block_store.d.ts +9 -1
  103. package/dest/storage/anchor_block_store/anchor_block_store.d.ts.map +1 -1
  104. package/dest/storage/anchor_block_store/anchor_block_store.js +8 -1
  105. package/dest/storage/capsule_store/capsule_store.js +6 -8
  106. package/dest/storage/contract_store/contract_store.d.ts +42 -15
  107. package/dest/storage/contract_store/contract_store.d.ts.map +1 -1
  108. package/dest/storage/contract_store/contract_store.js +157 -72
  109. package/dest/storage/metadata.d.ts +1 -1
  110. package/dest/storage/metadata.js +1 -1
  111. package/dest/storage/note_store/note_store.d.ts +13 -3
  112. package/dest/storage/note_store/note_store.d.ts.map +1 -1
  113. package/dest/storage/note_store/note_store.js +147 -107
  114. package/dest/storage/private_event_store/private_event_store.d.ts +1 -1
  115. package/dest/storage/private_event_store/private_event_store.d.ts.map +1 -1
  116. package/dest/storage/private_event_store/private_event_store.js +84 -61
  117. package/dest/storage/private_event_store/stored_private_event.d.ts +4 -4
  118. package/dest/storage/private_event_store/stored_private_event.d.ts.map +1 -1
  119. package/dest/storage/private_event_store/stored_private_event.js +2 -2
  120. package/dest/storage/tagging_store/recipient_tagging_store.d.ts +6 -6
  121. package/dest/storage/tagging_store/recipient_tagging_store.d.ts.map +1 -1
  122. package/dest/storage/tagging_store/recipient_tagging_store.js +31 -19
  123. package/dest/storage/tagging_store/sender_address_book_store.d.ts +1 -1
  124. package/dest/storage/tagging_store/sender_address_book_store.d.ts.map +1 -1
  125. package/dest/storage/tagging_store/sender_address_book_store.js +20 -14
  126. package/dest/storage/tagging_store/sender_tagging_store.d.ts +5 -5
  127. package/dest/storage/tagging_store/sender_tagging_store.d.ts.map +1 -1
  128. package/dest/storage/tagging_store/sender_tagging_store.js +184 -114
  129. package/dest/tagging/get_all_logs_by_tags.d.ts +4 -4
  130. package/dest/tagging/get_all_logs_by_tags.d.ts.map +1 -1
  131. package/dest/tagging/get_all_logs_by_tags.js +17 -3
  132. package/dest/tagging/index.d.ts +2 -2
  133. package/dest/tagging/index.d.ts.map +1 -1
  134. package/dest/tagging/index.js +1 -1
  135. package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.d.ts +5 -6
  136. package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.d.ts.map +1 -1
  137. package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.js +7 -7
  138. package/dest/tagging/recipient_sync/utils/find_highest_indexes.js +2 -2
  139. package/dest/tagging/recipient_sync/utils/load_logs_for_range.d.ts +7 -8
  140. package/dest/tagging/recipient_sync/utils/load_logs_for_range.d.ts.map +1 -1
  141. package/dest/tagging/recipient_sync/utils/load_logs_for_range.js +12 -11
  142. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts +5 -9
  143. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts.map +1 -1
  144. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.js +3 -6
  145. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts +5 -8
  146. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts.map +1 -1
  147. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.js +14 -15
  148. package/package.json +25 -16
  149. package/src/access_scopes.ts +9 -0
  150. package/src/block_synchronizer/block_synchronizer.ts +23 -19
  151. package/src/config/package_info.ts +1 -1
  152. package/src/contract_function_simulator/contract_function_simulator.ts +323 -128
  153. package/src/contract_function_simulator/execution_tagging_index_cache.ts +5 -5
  154. package/src/contract_function_simulator/noir-structs/event_validation_request.ts +1 -1
  155. package/src/contract_function_simulator/noir-structs/note_validation_request.ts +1 -1
  156. package/src/contract_function_simulator/oracle/interfaces.ts +12 -12
  157. package/src/contract_function_simulator/oracle/oracle.ts +41 -24
  158. package/src/contract_function_simulator/oracle/private_execution_oracle.ts +100 -110
  159. package/src/contract_function_simulator/oracle/utility_execution_oracle.ts +139 -70
  160. package/src/contract_logging.ts +39 -0
  161. package/src/contract_sync/contract_sync_service.ts +152 -0
  162. package/src/contract_sync/{index.ts → helpers.ts} +21 -21
  163. package/src/debug/pxe_debug_utils.ts +63 -19
  164. package/src/entrypoints/client/bundle/index.ts +3 -0
  165. package/src/entrypoints/client/bundle/utils.ts +16 -15
  166. package/src/entrypoints/client/lazy/index.ts +3 -0
  167. package/src/entrypoints/client/lazy/utils.ts +17 -15
  168. package/src/entrypoints/pxe_creation_options.ts +2 -1
  169. package/src/entrypoints/server/index.ts +3 -1
  170. package/src/entrypoints/server/utils.ts +22 -26
  171. package/src/events/event_service.ts +4 -6
  172. package/src/job_coordinator/job_coordinator.ts +4 -3
  173. package/src/logs/log_service.ts +31 -38
  174. package/src/notes/note_service.ts +9 -10
  175. package/src/notes_filter.ts +26 -0
  176. package/src/oracle_version.ts +2 -2
  177. package/src/private_kernel/hints/{compute_tx_include_by_timestamp.ts → compute_tx_expiration_timestamp.ts} +13 -13
  178. package/src/private_kernel/hints/index.ts +1 -1
  179. package/src/private_kernel/hints/private_kernel_reset_private_inputs_builder.ts +164 -117
  180. package/src/private_kernel/hints/test_utils.ts +325 -0
  181. package/src/private_kernel/private_kernel_execution_prover.ts +25 -15
  182. package/src/private_kernel/private_kernel_oracle.ts +9 -9
  183. package/src/pxe.ts +191 -115
  184. package/src/storage/address_store/address_store.ts +15 -15
  185. package/src/storage/anchor_block_store/anchor_block_store.ts +8 -0
  186. package/src/storage/capsule_store/capsule_store.ts +8 -8
  187. package/src/storage/contract_store/contract_store.ts +186 -76
  188. package/src/storage/metadata.ts +1 -1
  189. package/src/storage/note_store/note_store.ts +169 -132
  190. package/src/storage/private_event_store/private_event_store.ts +102 -81
  191. package/src/storage/private_event_store/stored_private_event.ts +3 -3
  192. package/src/storage/tagging_store/recipient_tagging_store.ts +38 -24
  193. package/src/storage/tagging_store/sender_address_book_store.ts +20 -14
  194. package/src/storage/tagging_store/sender_tagging_store.ts +214 -130
  195. package/src/tagging/get_all_logs_by_tags.ts +31 -7
  196. package/src/tagging/index.ts +1 -1
  197. package/src/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.ts +9 -12
  198. package/src/tagging/recipient_sync/utils/find_highest_indexes.ts +2 -2
  199. package/src/tagging/recipient_sync/utils/load_logs_for_range.ts +12 -17
  200. package/src/tagging/sender_sync/sync_sender_tagging_indexes.ts +6 -11
  201. package/src/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.ts +14 -23
  202. package/dest/contract_sync/index.d.ts +0 -23
  203. package/dest/contract_sync/index.d.ts.map +0 -1
  204. package/dest/private_kernel/hints/compute_tx_include_by_timestamp.d.ts +0 -4
  205. package/dest/private_kernel/hints/compute_tx_include_by_timestamp.d.ts.map +0 -1
@@ -1,4 +1,3 @@
1
- import { toArray } from '@aztec/foundation/iterable';
2
1
  import { Semaphore } from '@aztec/foundation/queue';
3
2
  import { NoteStatus } from '@aztec/stdlib/note';
4
3
  import { StoredNote } from './stored_note.js';
@@ -49,26 +48,18 @@ import { StoredNote } from './stored_note.js';
49
48
  * @param scope - The scope (user/account) under which to store the notes
50
49
  * @param jobId - The job context for staged writes
51
50
  */ addNotes(notes, scope, jobId) {
52
- return this.#withJobLock(jobId, ()=>Promise.all(notes.map((noteDao)=>this.#addNote(noteDao, scope, jobId))));
53
- }
54
- async #addNote(note, scope, jobId) {
55
- const noteForJob = await this.#readNote(note.siloedNullifier.toString(), jobId) ?? new StoredNote(note, new Set());
56
- // Make sure the note is linked to the scope and staged for this job
57
- noteForJob.addScope(scope.toString());
58
- this.#writeNote(noteForJob, jobId);
51
+ return this.#withJobLock(jobId, ()=>this.#store.transactionAsync(()=>Promise.all(notes.map(async (note)=>{
52
+ const noteForJob = await this.#readNote(note.siloedNullifier.toString(), jobId) ?? new StoredNote(note, new Set());
53
+ noteForJob.addScope(scope.toString());
54
+ this.#writeNote(noteForJob, jobId);
55
+ }))));
59
56
  }
60
57
  async #readNote(nullifier, jobId) {
61
- // First check staged notes for this job
62
- const noteForJob = this.#getNotesForJob(jobId).get(nullifier);
63
- if (noteForJob) {
64
- return noteForJob;
65
- }
66
- // Then check persistent storage
58
+ // Always issue DB read to keep IndexedDB transaction alive (they auto-commit when a new micro-task starts and there
59
+ // are no pending read requests). The staged value still takes precedence if it exists.
67
60
  const noteBuffer = await this.#notes.getAsync(nullifier);
68
- if (noteBuffer) {
69
- return StoredNote.fromBuffer(noteBuffer);
70
- }
71
- return undefined;
61
+ const noteForJob = this.#getNotesForJob(jobId).get(nullifier);
62
+ return noteForJob ?? (noteBuffer ? StoredNote.fromBuffer(noteBuffer) : undefined);
72
63
  }
73
64
  #writeNote(note, jobId) {
74
65
  this.#getNotesForJob(jobId).set(note.noteDao.siloedNullifier.toString(), note);
@@ -83,49 +74,84 @@ import { StoredNote } from './stored_note.js';
83
74
  * @params jobId - the job context to read from.
84
75
  * @returns Filtered and deduplicated notes (a note might be present in multiple scopes - we ensure it is only
85
76
  * returned once if this is the case)
86
- * @throws If filtering by an empty scopes array. Scopes have to be set to undefined or to a non-empty array.
87
- */ async getNotes(filter, jobId) {
88
- if (filter.scopes !== undefined && filter.scopes.length === 0) {
89
- throw new Error('Trying to get notes with an empty scopes array');
77
+ */ getNotes(filter, jobId) {
78
+ if (filter.scopes !== 'ALL_SCOPES' && filter.scopes.length === 0) {
79
+ return Promise.resolve([]);
90
80
  }
91
- const targetStatus = filter.status ?? NoteStatus.ACTIVE;
92
- const foundNotes = new Map();
93
- const nullifiersOfContract = await this.#nullifiersOfContract(filter.contractAddress, jobId);
94
- for (const nullifier of nullifiersOfContract){
95
- const note = await this.#readNote(nullifier, jobId);
96
- // Defensive: hitting this case means we're mishandling contract indices or in-memory job data
97
- if (!note) {
98
- throw new Error('PXE note database is corrupted.');
99
- }
100
- // Apply filters
101
- if (targetStatus === NoteStatus.ACTIVE && note.isNullified()) {
102
- continue;
103
- }
104
- if (filter.owner && !note.noteDao.owner.equals(filter.owner)) {
105
- continue;
106
- }
107
- if (filter.storageSlot && !note.noteDao.storageSlot.equals(filter.storageSlot)) {
108
- continue;
109
- }
110
- if (filter.siloedNullifier && !note.noteDao.siloedNullifier.equals(filter.siloedNullifier)) {
111
- continue;
112
- }
113
- if (filter.scopes && note.scopes.intersection(new Set(filter.scopes.map((s)=>s.toString()))).size === 0) {
114
- continue;
81
+ return this.#store.transactionAsync(async ()=>{
82
+ const targetStatus = filter.status ?? NoteStatus.ACTIVE;
83
+ // The code below might read a bit unnatural, the reason is that we need to be careful in how we use `await` inside
84
+ // `transactionAsync`, otherwise browsers might choose to auto-commit the IndexedDB transaction forcing us to
85
+ // explicitly handle that condition. The rule we need to honor is: do not await unless you generate a database
86
+ // read or write or you're done using the DB for the remainder of the transaction. The following sequence is
87
+ // unsafe in IndexedDB:
88
+ //
89
+ // 1. start transactionAsync()
90
+ // 2. await readDb() <-- OK, transaction alive because we issued DB ops
91
+ // 3. run a bunch of computations (no await involved) <-- OK, tx alive because we are in the same microtask
92
+ // 4. await doSthNotInDb() <-- no DB ops issued in this task, browser's free to decide to commit the tx
93
+ // 5. await readDb() <-- BOOM, TransactionInactiveError
94
+ //
95
+ // Note that the real issue is in step number 5: we try to continue using a transaction that the browser might
96
+ // have already committed.
97
+ //
98
+ // We need to read candidate notes which are either indexed by contract address in the DB (in
99
+ // #nullifiersByContractAddress), or lie in memory for the not yet committed `jobId`.
100
+ // So we collect promises based on both sources without awaiting for them.
101
+ const noteReadPromises = new Map();
102
+ // Awaiting the getValuesAsync iterator is fine because it's reading from the DB
103
+ for await (const nullifier of this.#nullifiersByContractAddress.getValuesAsync(filter.contractAddress.toString())){
104
+ // Each #readNote will perform a DB read
105
+ noteReadPromises.set(nullifier, this.#readNote(nullifier, jobId));
115
106
  }
116
- foundNotes.set(note.noteDao.siloedNullifier.toString(), note.noteDao);
117
- }
118
- // Sort by block number, then by tx index within block, then by note index within tx
119
- return [
120
- ...foundNotes.values()
121
- ].sort((a, b)=>{
122
- if (a.l2BlockNumber !== b.l2BlockNumber) {
123
- return a.l2BlockNumber - b.l2BlockNumber;
107
+ // Add staged nullifiers from job, no awaits involved, so we are fine
108
+ for (const storedNote of this.#getNotesForJob(jobId).values()){
109
+ if (storedNote.noteDao.contractAddress.equals(filter.contractAddress)) {
110
+ const nullifier = storedNote.noteDao.siloedNullifier.toString();
111
+ if (!noteReadPromises.has(nullifier)) {
112
+ noteReadPromises.set(nullifier, Promise.resolve(storedNote));
113
+ }
114
+ }
124
115
  }
125
- if (a.txIndexInBlock !== b.txIndexInBlock) {
126
- return a.txIndexInBlock - b.txIndexInBlock;
116
+ // By now we have pending DB requests from all the #readNote calls. Await them all together.
117
+ const notes = await Promise.all(noteReadPromises.values());
118
+ // The rest of the function is await-free, and just deals with filtering and sorting our findings.
119
+ const foundNotes = new Map();
120
+ for (const note of notes){
121
+ // Defensive: hitting this case means we're mishandling contract indices or in-memory job data
122
+ if (!note) {
123
+ throw new Error('PXE note database is corrupted.');
124
+ }
125
+ // Apply filters
126
+ if (targetStatus === NoteStatus.ACTIVE && note.isNullified()) {
127
+ continue;
128
+ }
129
+ if (filter.owner && !note.noteDao.owner.equals(filter.owner)) {
130
+ continue;
131
+ }
132
+ if (filter.storageSlot && !note.noteDao.storageSlot.equals(filter.storageSlot)) {
133
+ continue;
134
+ }
135
+ if (filter.siloedNullifier && !note.noteDao.siloedNullifier.equals(filter.siloedNullifier)) {
136
+ continue;
137
+ }
138
+ if (filter.scopes !== 'ALL_SCOPES' && note.scopes.intersection(new Set(filter.scopes.map((s)=>s.toString()))).size === 0) {
139
+ continue;
140
+ }
141
+ foundNotes.set(note.noteDao.siloedNullifier.toString(), note.noteDao);
127
142
  }
128
- return a.noteIndexInTx - b.noteIndexInTx;
143
+ // Sort by block number, then by tx index within block, then by note index within tx
144
+ return [
145
+ ...foundNotes.values()
146
+ ].sort((a, b)=>{
147
+ if (a.l2BlockNumber !== b.l2BlockNumber) {
148
+ return a.l2BlockNumber - b.l2BlockNumber;
149
+ }
150
+ if (a.txIndexInBlock !== b.txIndexInBlock) {
151
+ return a.txIndexInBlock - b.txIndexInBlock;
152
+ }
153
+ return a.noteIndexInTx - b.noteIndexInTx;
154
+ });
129
155
  });
130
156
  }
131
157
  /**
@@ -143,10 +169,13 @@ import { StoredNote } from './stored_note.js';
143
169
  * @returns Array of NoteDao objects that were nullified
144
170
  * @throws Error if any nullifier is not found in this notes store
145
171
  */ applyNullifiers(nullifiers, jobId) {
172
+ if (nullifiers.length === 0) {
173
+ return Promise.resolve([]);
174
+ }
175
+ if (nullifiers.some((n)=>n.l2BlockNumber === 0)) {
176
+ return Promise.reject(new Error('applyNullifiers: nullifiers cannot have been emitted at block 0'));
177
+ }
146
178
  return this.#withJobLock(jobId, ()=>this.#store.transactionAsync(async ()=>{
147
- if (nullifiers.length === 0) {
148
- return [];
149
- }
150
179
  const notesToNullify = await Promise.all(nullifiers.map(async (nullifierInBlock)=>{
151
180
  const nullifier = nullifierInBlock.data.toString();
152
181
  const storedNote = await this.#readNote(nullifier, jobId);
@@ -154,13 +183,12 @@ import { StoredNote } from './stored_note.js';
154
183
  throw new Error(`Attempted to mark a note as nullified which does not exist in PXE DB`);
155
184
  }
156
185
  return {
157
- storedNote: await this.#readNote(nullifier, jobId),
186
+ storedNote,
158
187
  blockNumber: nullifierInBlock.l2BlockNumber
159
188
  };
160
189
  }));
161
190
  const notesNullifiedInThisCall = new Map();
162
191
  for (const noteToNullify of notesToNullify){
163
- // Safe to coerce (!) because we throw if we find any undefined above
164
192
  const note = noteToNullify.storedNote;
165
193
  // Skip already nullified notes
166
194
  if (note.isNullified()) {
@@ -200,15 +228,22 @@ import { StoredNote } from './stored_note.js';
200
228
  *
201
229
  * @param blockNumber - Notes created after this block number will be deleted
202
230
  */ async #deleteActiveNotesAfterBlock(blockNumber) {
203
- const notes = await toArray(this.#notes.valuesAsync());
204
- for (const noteBuffer of notes){
231
+ // Collect notes to delete during iteration to keep IndexedDB transaction alive.
232
+ const notesToDelete = [];
233
+ for await (const noteBuffer of this.#notes.valuesAsync()){
205
234
  const storedNote = StoredNote.fromBuffer(noteBuffer);
206
235
  if (storedNote.noteDao.l2BlockNumber > blockNumber) {
207
- const noteNullifier = storedNote.noteDao.siloedNullifier.toString();
208
- await this.#notes.delete(noteNullifier);
209
- await this.#nullifiersByContractAddress.deleteValue(storedNote.noteDao.contractAddress.toString(), noteNullifier);
236
+ notesToDelete.push({
237
+ nullifier: storedNote.noteDao.siloedNullifier.toString(),
238
+ contractAddress: storedNote.noteDao.contractAddress.toString()
239
+ });
210
240
  }
211
241
  }
242
+ // Delete all collected notes. Each delete is a DB operation that keeps the transaction alive.
243
+ for (const { nullifier, contractAddress } of notesToDelete){
244
+ await this.#notes.delete(nullifier);
245
+ await this.#nullifiersByContractAddress.deleteValue(contractAddress, nullifier);
246
+ }
212
247
  }
213
248
  /**
214
249
  * Rewinds nullifications after a given block number.
@@ -219,47 +254,62 @@ import { StoredNote } from './stored_note.js';
219
254
  * @param blockNumber - Revert nullifications that occurred after this block
220
255
  * @param anchorBlockNumber - Upper bound for the block range to process
221
256
  */ async #rewindNullifiedNotesAfterBlock(blockNumber, anchorBlockNumber) {
222
- const currentBlockNumber = blockNumber + 1;
223
- for(let i = currentBlockNumber; i <= anchorBlockNumber; i++){
224
- const noteNullifiersToReinsert = await toArray(this.#nullifiersByNullificationBlockNumber.getValuesAsync(i));
225
- const nullifiedNoteBuffers = await Promise.all(noteNullifiersToReinsert.map(async (noteNullifier)=>{
226
- const note = await this.#notes.getAsync(noteNullifier);
227
- if (!note) {
228
- throw new Error(`PXE DB integrity error: no note found with nullifier ${noteNullifier}`);
257
+ // First pass: collect all nullifiers for all blocks, starting reads during iteration to keep tx alive.
258
+ const nullifiersByBlock = new Map();
259
+ for(let i = blockNumber + 1; i <= anchorBlockNumber; i++){
260
+ const blockNullifiers = [];
261
+ for await (const nullifier of this.#nullifiersByNullificationBlockNumber.getValuesAsync(i)){
262
+ // Start read immediately during iteration to keep IndexedDB transaction alive
263
+ blockNullifiers.push({
264
+ nullifier,
265
+ noteReadPromise: this.#notes.getAsync(nullifier)
266
+ });
267
+ }
268
+ if (blockNullifiers.length > 0) {
269
+ nullifiersByBlock.set(i, blockNullifiers);
270
+ }
271
+ }
272
+ // Second pass: await reads and perform writes
273
+ for (const [block, nullifiers] of nullifiersByBlock){
274
+ for (const { nullifier, noteReadPromise } of nullifiers){
275
+ const noteBuffer = await noteReadPromise;
276
+ if (!noteBuffer) {
277
+ throw new Error(`PXE DB integrity error: no note found with nullifier ${nullifier}`);
229
278
  }
230
- return note;
231
- }));
232
- const storedNotes = nullifiedNoteBuffers.map((buffer)=>StoredNote.fromBuffer(buffer));
233
- for (const storedNote of storedNotes){
234
- const noteNullifier = storedNote.noteDao.siloedNullifier.toString();
235
- const scopes = storedNote.scopes;
236
- if (scopes.size === 0) {
237
- // We should never run into this error because notes always have a scope assigned to them - either on initial
238
- // insertion via `addNotes` or when removing their nullifiers.
239
- throw new Error(`No scopes found for nullified note with nullifier ${noteNullifier}`);
279
+ const storedNote = StoredNote.fromBuffer(noteBuffer);
280
+ if (storedNote.scopes.size === 0) {
281
+ throw new Error(`No scopes found for nullified note with nullifier ${nullifier}`);
240
282
  }
241
283
  storedNote.markAsActive();
242
284
  await Promise.all([
243
- this.#notes.set(noteNullifier, storedNote.toBuffer()),
244
- this.#nullifiersByNullificationBlockNumber.deleteValue(i, noteNullifier)
285
+ this.#notes.set(nullifier, storedNote.toBuffer()),
286
+ this.#nullifiersByNullificationBlockNumber.deleteValue(block, nullifier)
245
287
  ]);
246
288
  }
247
289
  }
248
290
  }
249
- commit(jobId) {
250
- return this.#withJobLock(jobId, async ()=>{
251
- for (const [nullifier, storedNote] of this.#getNotesForJob(jobId)){
252
- await this.#notes.set(nullifier, storedNote.toBuffer());
253
- await this.#nullifiersByContractAddress.set(storedNote.noteDao.contractAddress.toString(), nullifier);
254
- if (storedNote.nullifiedAt !== undefined) {
255
- await this.#nullifiersByNullificationBlockNumber.set(storedNote.nullifiedAt, nullifier);
256
- }
291
+ /**
292
+ * Commits in memory job data to persistent storage.
293
+ *
294
+ * Called by JobCoordinator when a job completes successfully.
295
+ *
296
+ * Note: JobCoordinator wraps all commits in a single transaction, so we don't need our own transactionAsync here
297
+ * (and using one would throw on IndexedDB as it does not support nested txs).
298
+ *
299
+ * @param jobId - The jobId identifying which staged data to commit
300
+ */ async commit(jobId) {
301
+ for (const [nullifier, storedNote] of this.#getNotesForJob(jobId)){
302
+ await this.#notes.set(nullifier, storedNote.toBuffer());
303
+ await this.#nullifiersByContractAddress.set(storedNote.noteDao.contractAddress.toString(), nullifier);
304
+ if (storedNote.nullifiedAt !== undefined) {
305
+ await this.#nullifiersByNullificationBlockNumber.set(storedNote.nullifiedAt, nullifier);
257
306
  }
258
- this.#clearJobData(jobId);
259
- });
307
+ }
308
+ this.#clearJobData(jobId);
260
309
  }
261
310
  discardStaged(jobId) {
262
- return this.#withJobLock(jobId, ()=>Promise.resolve(this.#clearJobData(jobId)));
311
+ this.#clearJobData(jobId);
312
+ return Promise.resolve();
263
313
  }
264
314
  #clearJobData(jobId) {
265
315
  this.#notesForJob.delete(jobId);
@@ -290,14 +340,4 @@ import { StoredNote } from './stored_note.js';
290
340
  }
291
341
  return notesForJob;
292
342
  }
293
- async #nullifiersOfContract(contractAddress, jobId) {
294
- // Collect persisted nullifiers for this contract
295
- const persistedNullifiers = await toArray(this.#nullifiersByContractAddress.getValuesAsync(contractAddress.toString()));
296
- // Collect staged nullifiers from the job where the note's contract matches
297
- const stagedNullifiers = this.#getNotesForJob(jobId).values().filter((storedNote)=>storedNote.noteDao.contractAddress.equals(contractAddress)).map((storedNote)=>storedNote.noteDao.siloedNullifier.toString());
298
- return new Set([
299
- ...persistedNullifiers,
300
- ...stagedNullifiers
301
- ]);
302
- }
303
343
  }
@@ -88,4 +88,4 @@ export declare class PrivateEventStore implements StagedStore {
88
88
  discardStaged(jobId: string): Promise<void>;
89
89
  }
90
90
  export {};
91
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpdmF0ZV9ldmVudF9zdG9yZS5kLnRzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3N0b3JhZ2UvcHJpdmF0ZV9ldmVudF9zdG9yZS9wcml2YXRlX2V2ZW50X3N0b3JlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxFQUFFLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUlwRCxPQUFPLEtBQUssRUFBRSxpQkFBaUIsRUFBcUMsTUFBTSxpQkFBaUIsQ0FBQztBQUM1RixPQUFPLEtBQUssRUFBRSxhQUFhLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUN2RCxPQUFPLEtBQUssRUFBRSxZQUFZLEVBQUUsTUFBTSw2QkFBNkIsQ0FBQztBQUNoRSxPQUFPLEtBQUssRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFLE1BQU0sa0JBQWtCLENBQUM7QUFFckQsT0FBTyxLQUFLLEVBQUUsV0FBVyxFQUFFLE1BQU0sMENBQTBDLENBQUM7QUFDNUUsT0FBTyxLQUFLLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFHdkQsTUFBTSxNQUFNLHVCQUF1QixHQUFHO0lBQ3BDLGVBQWUsRUFBRSxZQUFZLENBQUM7SUFDOUIsU0FBUyxFQUFFLE1BQU0sQ0FBQztJQUNsQixPQUFPLEVBQUUsTUFBTSxDQUFDO0lBQ2hCLE1BQU0sRUFBRSxZQUFZLEVBQUUsQ0FBQztJQUN2QixNQUFNLENBQUMsRUFBRSxNQUFNLENBQUM7Q0FDakIsQ0FBQztBQUVGLEtBQUssb0JBQW9CLEdBQUcsSUFBSSxHQUFHO0lBQ2pDLGVBQWUsRUFBRSxZQUFZLENBQUM7SUFDOUIsS0FBSyxFQUFFLFlBQVksQ0FBQztJQUNwQiwyQ0FBMkM7SUFDM0MsY0FBYyxFQUFFLE1BQU0sQ0FBQztJQUN2Qix5RUFBeUU7SUFDekUsY0FBYyxFQUFFLE1BQU0sQ0FBQztDQUN4QixDQUFDO0FBRUY7O0dBRUc7QUFDSCxxQkFBYSxpQkFBa0IsWUFBVyxXQUFXOztJQUNuRCxRQUFRLENBQUMsU0FBUyxFQUFFLE1BQU0sQ0FBbUI7SUFnQjdDLE1BQU0seUNBQXVDO0lBRTdDLFlBQVksS0FBSyxFQUFFLGlCQUFpQixFQVFuQztJQUVEOzs7Ozs7Ozs7OztPQVdHO0lBQ0gsb0JBQW9CLENBQ2xCLGFBQWEsRUFBRSxhQUFhLEVBQzVCLFVBQVUsRUFBRSxFQUFFLEVBQ2QsVUFBVSxFQUFFLEVBQUUsRUFBRSxFQUNoQixxQkFBcUIsRUFBRSxFQUFFLEVBQ3pCLFFBQVEsRUFBRSxvQkFBb0IsRUFDOUIsS0FBSyxFQUFFLE1BQU0saUJBdUNkO0lBRUQ7Ozs7Ozs7Ozs7T0FVRztJQUNJLGdCQUFnQixDQUNyQixhQUFhLEVBQUUsYUFBYSxFQUM1QixNQUFNLEVBQUUsdUJBQXVCLEdBQzlCLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDLENBcUUvQjtJQUVEOzs7Ozs7Ozs7Ozs7Ozs7O09BZ0JHO0lBQ1UsUUFBUSxDQUFDLFdBQVcsRUFBRSxNQUFNLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0E0QnBGO0lBRUQ7Ozs7Ozs7OztPQVNHO0lBQ0gsTUFBTSxDQUFDLEtBQUssRUFBRSxNQUFNLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxDQWVuQztJQUVEOztPQUVHO0lBQ0gsYUFBYSxDQUFDLEtBQUssRUFBRSxNQUFNLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxDQUUxQztDQTRFRiJ9
91
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJpdmF0ZV9ldmVudF9zdG9yZS5kLnRzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3N0b3JhZ2UvcHJpdmF0ZV9ldmVudF9zdG9yZS9wcml2YXRlX2V2ZW50X3N0b3JlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxFQUFFLEVBQUUsTUFBTSxnQ0FBZ0MsQ0FBQztBQUdwRCxPQUFPLEtBQUssRUFBRSxpQkFBaUIsRUFBcUMsTUFBTSxpQkFBaUIsQ0FBQztBQUM1RixPQUFPLEtBQUssRUFBRSxhQUFhLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUN2RCxPQUFPLEtBQUssRUFBRSxZQUFZLEVBQUUsTUFBTSw2QkFBNkIsQ0FBQztBQUNoRSxPQUFPLEtBQUssRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFLE1BQU0sa0JBQWtCLENBQUM7QUFFckQsT0FBTyxLQUFLLEVBQUUsV0FBVyxFQUFFLE1BQU0sMENBQTBDLENBQUM7QUFDNUUsT0FBTyxLQUFLLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFHdkQsTUFBTSxNQUFNLHVCQUF1QixHQUFHO0lBQ3BDLGVBQWUsRUFBRSxZQUFZLENBQUM7SUFDOUIsU0FBUyxFQUFFLE1BQU0sQ0FBQztJQUNsQixPQUFPLEVBQUUsTUFBTSxDQUFDO0lBQ2hCLE1BQU0sRUFBRSxZQUFZLEVBQUUsQ0FBQztJQUN2QixNQUFNLENBQUMsRUFBRSxNQUFNLENBQUM7Q0FDakIsQ0FBQztBQUVGLEtBQUssb0JBQW9CLEdBQUcsSUFBSSxHQUFHO0lBQ2pDLGVBQWUsRUFBRSxZQUFZLENBQUM7SUFDOUIsS0FBSyxFQUFFLFlBQVksQ0FBQztJQUNwQiwyQ0FBMkM7SUFDM0MsY0FBYyxFQUFFLE1BQU0sQ0FBQztJQUN2Qix5RUFBeUU7SUFDekUsY0FBYyxFQUFFLE1BQU0sQ0FBQztDQUN4QixDQUFDO0FBRUY7O0dBRUc7QUFDSCxxQkFBYSxpQkFBa0IsWUFBVyxXQUFXOztJQUNuRCxRQUFRLENBQUMsU0FBUyxFQUFFLE1BQU0sQ0FBbUI7SUFnQjdDLE1BQU0seUNBQXVDO0lBRTdDLFlBQVksS0FBSyxFQUFFLGlCQUFpQixFQVFuQztJQUVEOzs7Ozs7Ozs7OztPQVdHO0lBQ0gsb0JBQW9CLENBQ2xCLGFBQWEsRUFBRSxhQUFhLEVBQzVCLFVBQVUsRUFBRSxFQUFFLEVBQ2QsVUFBVSxFQUFFLEVBQUUsRUFBRSxFQUNoQixxQkFBcUIsRUFBRSxFQUFFLEVBQ3pCLFFBQVEsRUFBRSxvQkFBb0IsRUFDOUIsS0FBSyxFQUFFLE1BQU0saUJBeUNkO0lBRUQ7Ozs7Ozs7Ozs7T0FVRztJQUNJLGdCQUFnQixDQUNyQixhQUFhLEVBQUUsYUFBYSxFQUM1QixNQUFNLEVBQUUsdUJBQXVCLEdBQzlCLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDLENBK0UvQjtJQUVEOzs7Ozs7Ozs7Ozs7Ozs7O09BZ0JHO0lBQ1UsUUFBUSxDQUFDLFdBQVcsRUFBRSxNQUFNLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FzQ3BGO0lBRUQ7Ozs7Ozs7OztPQVNHO0lBQ0csTUFBTSxDQUFDLEtBQUssRUFBRSxNQUFNLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQyxDQWdCekM7SUFFRDs7T0FFRztJQUNILGFBQWEsQ0FBQyxLQUFLLEVBQUUsTUFBTSxHQUFHLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FHMUM7Q0EwRUYifQ==
@@ -1 +1 @@
1
- {"version":3,"file":"private_event_store.d.ts","sourceRoot":"","sources":["../../../src/storage/private_event_store/private_event_store.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,EAAE,EAAE,MAAM,gCAAgC,CAAC;AAIpD,OAAO,KAAK,EAAE,iBAAiB,EAAqC,MAAM,iBAAiB,CAAC;AAC5F,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AACvD,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,6BAA6B,CAAC;AAChE,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAErD,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,0CAA0C,CAAC;AAC5E,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,cAAc,CAAC;AAGvD,MAAM,MAAM,uBAAuB,GAAG;IACpC,eAAe,EAAE,YAAY,CAAC;IAC9B,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,YAAY,EAAE,CAAC;IACvB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,KAAK,oBAAoB,GAAG,IAAI,GAAG;IACjC,eAAe,EAAE,YAAY,CAAC;IAC9B,KAAK,EAAE,YAAY,CAAC;IACpB,2CAA2C;IAC3C,cAAc,EAAE,MAAM,CAAC;IACvB,yEAAyE;IACzE,cAAc,EAAE,MAAM,CAAC;CACxB,CAAC;AAEF;;GAEG;AACH,qBAAa,iBAAkB,YAAW,WAAW;;IACnD,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAmB;IAgB7C,MAAM,yCAAuC;IAE7C,YAAY,KAAK,EAAE,iBAAiB,EAQnC;IAED;;;;;;;;;;;OAWG;IACH,oBAAoB,CAClB,aAAa,EAAE,aAAa,EAC5B,UAAU,EAAE,EAAE,EACd,UAAU,EAAE,EAAE,EAAE,EAChB,qBAAqB,EAAE,EAAE,EACzB,QAAQ,EAAE,oBAAoB,EAC9B,KAAK,EAAE,MAAM,iBAuCd;IAED;;;;;;;;;;OAUG;IACI,gBAAgB,CACrB,aAAa,EAAE,aAAa,EAC5B,MAAM,EAAE,uBAAuB,GAC9B,OAAO,CAAC,kBAAkB,EAAE,CAAC,CAqE/B;IAED;;;;;;;;;;;;;;;;OAgBG;IACU,QAAQ,CAAC,WAAW,EAAE,MAAM,EAAE,kBAAkB,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CA4BpF;IAED;;;;;;;;;OASG;IACH,MAAM,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAenC;IAED;;OAEG;IACH,aAAa,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAE1C;CA4EF"}
1
+ {"version":3,"file":"private_event_store.d.ts","sourceRoot":"","sources":["../../../src/storage/private_event_store/private_event_store.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,EAAE,EAAE,MAAM,gCAAgC,CAAC;AAGpD,OAAO,KAAK,EAAE,iBAAiB,EAAqC,MAAM,iBAAiB,CAAC;AAC5F,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AACvD,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,6BAA6B,CAAC;AAChE,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAErD,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,0CAA0C,CAAC;AAC5E,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,cAAc,CAAC;AAGvD,MAAM,MAAM,uBAAuB,GAAG;IACpC,eAAe,EAAE,YAAY,CAAC;IAC9B,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,YAAY,EAAE,CAAC;IACvB,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB,CAAC;AAEF,KAAK,oBAAoB,GAAG,IAAI,GAAG;IACjC,eAAe,EAAE,YAAY,CAAC;IAC9B,KAAK,EAAE,YAAY,CAAC;IACpB,2CAA2C;IAC3C,cAAc,EAAE,MAAM,CAAC;IACvB,yEAAyE;IACzE,cAAc,EAAE,MAAM,CAAC;CACxB,CAAC;AAEF;;GAEG;AACH,qBAAa,iBAAkB,YAAW,WAAW;;IACnD,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAmB;IAgB7C,MAAM,yCAAuC;IAE7C,YAAY,KAAK,EAAE,iBAAiB,EAQnC;IAED;;;;;;;;;;;OAWG;IACH,oBAAoB,CAClB,aAAa,EAAE,aAAa,EAC5B,UAAU,EAAE,EAAE,EACd,UAAU,EAAE,EAAE,EAAE,EAChB,qBAAqB,EAAE,EAAE,EACzB,QAAQ,EAAE,oBAAoB,EAC9B,KAAK,EAAE,MAAM,iBAyCd;IAED;;;;;;;;;;OAUG;IACI,gBAAgB,CACrB,aAAa,EAAE,aAAa,EAC5B,MAAM,EAAE,uBAAuB,GAC9B,OAAO,CAAC,kBAAkB,EAAE,CAAC,CA+E/B;IAED;;;;;;;;;;;;;;;;OAgBG;IACU,QAAQ,CAAC,WAAW,EAAE,MAAM,EAAE,kBAAkB,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAsCpF;IAED;;;;;;;;;OASG;IACG,MAAM,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAgBzC;IAED;;OAEG;IACH,aAAa,CAAC,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC,CAG1C;CA0EF"}
@@ -1,5 +1,4 @@
1
1
  import { BlockNumber } from '@aztec/foundation/branded-types';
2
- import { toArray } from '@aztec/foundation/iterable';
3
2
  import { createLogger } from '@aztec/foundation/log';
4
3
  import { Semaphore } from '@aztec/foundation/queue';
5
4
  import { StoredPrivateEvent } from './stored_private_event.js';
@@ -34,27 +33,27 @@ import { StoredPrivateEvent } from './stored_private_event.js';
34
33
  * txHash - The transaction hash of the event log.
35
34
  * blockNumber - The block number in which the event was emitted.
36
35
  */ storePrivateEventLog(eventSelector, randomness, msgContent, siloedEventCommitment, metadata, jobId) {
37
- return this.#withJobLock(jobId, async ()=>{
38
- const { contractAddress, scope, txHash, l2BlockNumber, l2BlockHash, txIndexInBlock, eventIndexInTx } = metadata;
39
- const eventId = siloedEventCommitment.toString();
40
- this.logger.verbose('storing private event log (job stage)', {
41
- eventId,
42
- contractAddress,
43
- scope,
44
- msgContent,
45
- l2BlockNumber
46
- });
47
- const existing = await this.#readEvent(eventId, jobId);
48
- if (existing) {
49
- // If we already stored this event, we still want to make sure to track it for the given scope
50
- existing.addScope(scope.toString());
51
- this.#writeEvent(eventId, existing, jobId);
52
- } else {
53
- this.#writeEvent(eventId, new StoredPrivateEvent(randomness, msgContent, l2BlockNumber, l2BlockHash, txHash, txIndexInBlock, eventIndexInTx, contractAddress, eventSelector, new Set([
54
- scope.toString()
55
- ])), jobId);
56
- }
57
- });
36
+ return this.#withJobLock(jobId, ()=>this.#store.transactionAsync(async ()=>{
37
+ const { contractAddress, scope, txHash, l2BlockNumber, l2BlockHash, txIndexInBlock, eventIndexInTx } = metadata;
38
+ const eventId = siloedEventCommitment.toString();
39
+ this.logger.verbose('storing private event log (job stage)', {
40
+ eventId,
41
+ contractAddress,
42
+ scope,
43
+ msgContent,
44
+ l2BlockNumber
45
+ });
46
+ const existing = await this.#readEvent(eventId, jobId);
47
+ if (existing) {
48
+ // If we already stored this event, we still want to make sure to track it for the given scope
49
+ existing.addScope(scope.toString());
50
+ this.#writeEvent(eventId, existing, jobId);
51
+ } else {
52
+ this.#writeEvent(eventId, new StoredPrivateEvent(randomness, msgContent, l2BlockNumber, l2BlockHash, txHash, txIndexInBlock, eventIndexInTx, contractAddress, eventSelector, new Set([
53
+ scope.toString()
54
+ ])), jobId);
55
+ }
56
+ }));
58
57
  }
59
58
  /**
60
59
  * Returns the private events given search parameters.
@@ -68,12 +67,22 @@ import { StoredPrivateEvent } from './stored_private_event.js';
68
67
  * included.
69
68
  */ getPrivateEvents(eventSelector, filter) {
70
69
  return this.#store.transactionAsync(async ()=>{
71
- const events = [];
72
70
  const key = this.#keyFor(filter.contractAddress, eventSelector);
73
71
  const targetScopes = new Set(filter.scopes.map((s)=>s.toString()));
74
- const eventIds = await toArray(this.#eventsByContractAndEventSelector.getValuesAsync(key));
75
- for (const eventId of eventIds){
76
- const eventBuffer = await this.#events.getAsync(eventId);
72
+ // Map from eventId to the promise that reads the event buffer.
73
+ // We start reads during iteration to keep DB requests pending and avoid IndexedDB auto-commit.
74
+ const eventReadPromises = new Map();
75
+ for await (const eventId of this.#eventsByContractAndEventSelector.getValuesAsync(key)){
76
+ eventReadPromises.set(eventId, this.#events.getAsync(eventId));
77
+ }
78
+ const eventIds = [
79
+ ...eventReadPromises.keys()
80
+ ];
81
+ const eventBuffers = await Promise.all(eventReadPromises.values());
82
+ const events = [];
83
+ for(let i = 0; i < eventIds.length; i++){
84
+ const eventId = eventIds[i];
85
+ const eventBuffer = eventBuffers[i];
77
86
  // Defensive, if it happens, there's a problem with how we're handling #eventsByContractAndEventSelector
78
87
  if (!eventBuffer) {
79
88
  this.logger.verbose(`EventId ${eventId} does not exist in main index but it is referenced from contract event selector index`);
@@ -135,21 +144,34 @@ import { StoredPrivateEvent } from './stored_private_event.js';
135
144
  *
136
145
  * IMPORTANT: This method must be called within a transaction to ensure atomicity.
137
146
  */ async rollback(blockNumber, synchedBlockNumber) {
138
- let removedCount = 0;
147
+ // First pass: collect all event IDs for all blocks, starting reads during iteration to keep tx alive.
148
+ const eventsByBlock = new Map();
139
149
  for(let block = blockNumber + 1; block <= synchedBlockNumber; block++){
140
- const eventIds = await toArray(this.#eventsByBlockNumber.getValuesAsync(block));
141
- if (eventIds.length > 0) {
142
- await this.#eventsByBlockNumber.delete(block);
143
- for (const eventId of eventIds){
144
- const buffer = await this.#events.getAsync(eventId);
145
- if (!buffer) {
146
- throw new Error(`Event not found for eventId ${eventId}`);
147
- }
148
- const entry = StoredPrivateEvent.fromBuffer(buffer);
149
- await this.#events.delete(eventId);
150
- await this.#eventsByContractAndEventSelector.deleteValue(this.#keyFor(entry.contractAddress, entry.eventSelector), eventId);
151
- removedCount++;
150
+ const blockEvents = [];
151
+ for await (const eventId of this.#eventsByBlockNumber.getValuesAsync(block)){
152
+ // Start read immediately during iteration to keep IndexedDB transaction alive
153
+ blockEvents.push({
154
+ eventId,
155
+ eventReadPromise: this.#events.getAsync(eventId)
156
+ });
157
+ }
158
+ if (blockEvents.length > 0) {
159
+ eventsByBlock.set(block, blockEvents);
160
+ }
161
+ }
162
+ // Second pass: await reads and perform deletes
163
+ let removedCount = 0;
164
+ for (const [block, events] of eventsByBlock){
165
+ await this.#eventsByBlockNumber.delete(block);
166
+ for (const { eventId, eventReadPromise } of events){
167
+ const buffer = await eventReadPromise;
168
+ if (!buffer) {
169
+ throw new Error(`Event not found for eventId ${eventId}`);
152
170
  }
171
+ const entry = StoredPrivateEvent.fromBuffer(buffer);
172
+ await this.#events.delete(eventId);
173
+ await this.#eventsByContractAndEventSelector.deleteValue(this.#keyFor(entry.contractAddress, entry.eventSelector), eventId);
174
+ removedCount++;
153
175
  }
154
176
  }
155
177
  this.logger.verbose(`Rolled back ${removedCount} private events after block ${blockNumber}`);
@@ -163,39 +185,40 @@ import { StoredPrivateEvent } from './stored_private_event.js';
163
185
  * (and using one would throw on IndexedDB as it does not support nested txs).
164
186
  *
165
187
  * @param jobId - The jobId identifying which staged data to commit
166
- */ commit(jobId) {
167
- return this.#withJobLock(jobId, async ()=>{
168
- for (const [eventId, entry] of this.#getEventsForJob(jobId).entries()){
169
- const lookupKey = this.#keyFor(entry.contractAddress, entry.eventSelector);
170
- this.logger.verbose('storing private event log', {
171
- eventId,
172
- lookupKey
173
- });
174
- await Promise.all([
175
- this.#events.set(eventId, entry.toBuffer()),
176
- this.#eventsByContractAndEventSelector.set(lookupKey, eventId),
177
- this.#eventsByBlockNumber.set(entry.l2BlockNumber, eventId)
178
- ]);
179
- }
180
- this.#clearJobData(jobId);
181
- });
188
+ */ async commit(jobId) {
189
+ // Note: Don't use #withJobLock here - commit runs within JobCoordinator's transactionAsync,
190
+ // and awaiting the lock would create a microtask boundary with no pending DB request,
191
+ // causing IndexedDB to auto-commit the transaction.
192
+ for (const [eventId, entry] of this.#getEventsForJob(jobId).entries()){
193
+ const lookupKey = this.#keyFor(entry.contractAddress, entry.eventSelector);
194
+ this.logger.verbose('storing private event log', {
195
+ eventId,
196
+ lookupKey
197
+ });
198
+ await Promise.all([
199
+ this.#events.set(eventId, entry.toBuffer()),
200
+ this.#eventsByContractAndEventSelector.set(lookupKey, eventId),
201
+ this.#eventsByBlockNumber.set(entry.l2BlockNumber, eventId)
202
+ ]);
203
+ }
204
+ this.#clearJobData(jobId);
182
205
  }
183
206
  /**
184
207
  * Discards in memory job data without persisting it.
185
208
  */ discardStaged(jobId) {
186
- return this.#withJobLock(jobId, ()=>Promise.resolve(this.#clearJobData(jobId)));
209
+ this.#clearJobData(jobId);
210
+ return Promise.resolve();
187
211
  }
188
212
  /**
189
213
  * Reads an event from in-memory job data first, falling back to persistent storage if not found.
190
214
  *
191
215
  * Returns undefined if the event does not exist in the store overall.
192
216
  */ async #readEvent(eventId, jobId) {
193
- const eventForJob = this.#getEventsForJob(jobId).get(eventId);
194
- if (eventForJob) {
195
- return eventForJob;
196
- }
217
+ // Always issue DB read to keep IndexedDB transaction alive (they auto-commit when a new micro-task starts and there
218
+ // are no pending read requests). The staged value still takes precedence if it exists.
197
219
  const buffer = await this.#events.getAsync(eventId);
198
- return buffer ? StoredPrivateEvent.fromBuffer(buffer) : undefined;
220
+ const eventForJob = this.#getEventsForJob(jobId).get(eventId);
221
+ return eventForJob ?? (buffer ? StoredPrivateEvent.fromBuffer(buffer) : undefined);
199
222
  }
200
223
  /**
201
224
  * Writes an event to in-memory job data.
@@ -1,23 +1,23 @@
1
1
  import { Fr } from '@aztec/foundation/curves/bn254';
2
2
  import { EventSelector } from '@aztec/stdlib/abi';
3
3
  import { AztecAddress } from '@aztec/stdlib/aztec-address';
4
- import { L2BlockHash } from '@aztec/stdlib/block';
4
+ import { BlockHash } from '@aztec/stdlib/block';
5
5
  import { TxHash } from '@aztec/stdlib/tx';
6
6
  /** Serializable private event entry with scope tracking. */
7
7
  export declare class StoredPrivateEvent {
8
8
  readonly randomness: Fr;
9
9
  readonly msgContent: Fr[];
10
10
  readonly l2BlockNumber: number;
11
- readonly l2BlockHash: L2BlockHash;
11
+ readonly l2BlockHash: BlockHash;
12
12
  readonly txHash: TxHash;
13
13
  readonly txIndexInBlock: number;
14
14
  readonly eventIndexInTx: number;
15
15
  readonly contractAddress: AztecAddress;
16
16
  readonly eventSelector: EventSelector;
17
17
  readonly scopes: Set<string>;
18
- constructor(randomness: Fr, msgContent: Fr[], l2BlockNumber: number, l2BlockHash: L2BlockHash, txHash: TxHash, txIndexInBlock: number, eventIndexInTx: number, contractAddress: AztecAddress, eventSelector: EventSelector, scopes: Set<string>);
18
+ constructor(randomness: Fr, msgContent: Fr[], l2BlockNumber: number, l2BlockHash: BlockHash, txHash: TxHash, txIndexInBlock: number, eventIndexInTx: number, contractAddress: AztecAddress, eventSelector: EventSelector, scopes: Set<string>);
19
19
  addScope(scope: string): void;
20
20
  toBuffer(): Buffer;
21
21
  static fromBuffer(buffer: Buffer): StoredPrivateEvent;
22
22
  }
23
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmVkX3ByaXZhdGVfZXZlbnQuZC50cyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zdG9yYWdlL3ByaXZhdGVfZXZlbnRfc3RvcmUvc3RvcmVkX3ByaXZhdGVfZXZlbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLEVBQUUsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBRXBELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUNsRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDM0QsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ2xELE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUUxQyw0REFBNEQ7QUFDNUQscUJBQWEsa0JBQWtCO0lBRTNCLFFBQVEsQ0FBQyxVQUFVLEVBQUUsRUFBRTtJQUN2QixRQUFRLENBQUMsVUFBVSxFQUFFLEVBQUUsRUFBRTtJQUN6QixRQUFRLENBQUMsYUFBYSxFQUFFLE1BQU07SUFDOUIsUUFBUSxDQUFDLFdBQVcsRUFBRSxXQUFXO0lBQ2pDLFFBQVEsQ0FBQyxNQUFNLEVBQUUsTUFBTTtJQUN2QixRQUFRLENBQUMsY0FBYyxFQUFFLE1BQU07SUFDL0IsUUFBUSxDQUFDLGNBQWMsRUFBRSxNQUFNO0lBQy9CLFFBQVEsQ0FBQyxlQUFlLEVBQUUsWUFBWTtJQUN0QyxRQUFRLENBQUMsYUFBYSxFQUFFLGFBQWE7SUFDckMsUUFBUSxDQUFDLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxDQUFDO0lBVjlCLFlBQ1csVUFBVSxFQUFFLEVBQUUsRUFDZCxVQUFVLEVBQUUsRUFBRSxFQUFFLEVBQ2hCLGFBQWEsRUFBRSxNQUFNLEVBQ3JCLFdBQVcsRUFBRSxXQUFXLEVBQ3hCLE1BQU0sRUFBRSxNQUFNLEVBQ2QsY0FBYyxFQUFFLE1BQU0sRUFDdEIsY0FBYyxFQUFFLE1BQU0sRUFDdEIsZUFBZSxFQUFFLFlBQVksRUFDN0IsYUFBYSxFQUFFLGFBQWEsRUFDNUIsTUFBTSxFQUFFLEdBQUcsQ0FBQyxNQUFNLENBQUMsRUFDMUI7SUFFSixRQUFRLENBQUMsS0FBSyxFQUFFLE1BQU0sUUFFckI7SUFFRCxRQUFRLElBQUksTUFBTSxDQWdCakI7SUFFRCxNQUFNLENBQUMsVUFBVSxDQUFDLE1BQU0sRUFBRSxNQUFNLEdBQUcsa0JBQWtCLENBMkJwRDtDQUNGIn0=
23
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmVkX3ByaXZhdGVfZXZlbnQuZC50cyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zdG9yYWdlL3ByaXZhdGVfZXZlbnRfc3RvcmUvc3RvcmVkX3ByaXZhdGVfZXZlbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLEVBQUUsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBRXBELE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQztBQUNsRCxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDM0QsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ2hELE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQztBQUUxQyw0REFBNEQ7QUFDNUQscUJBQWEsa0JBQWtCO0lBRTNCLFFBQVEsQ0FBQyxVQUFVLEVBQUUsRUFBRTtJQUN2QixRQUFRLENBQUMsVUFBVSxFQUFFLEVBQUUsRUFBRTtJQUN6QixRQUFRLENBQUMsYUFBYSxFQUFFLE1BQU07SUFDOUIsUUFBUSxDQUFDLFdBQVcsRUFBRSxTQUFTO0lBQy9CLFFBQVEsQ0FBQyxNQUFNLEVBQUUsTUFBTTtJQUN2QixRQUFRLENBQUMsY0FBYyxFQUFFLE1BQU07SUFDL0IsUUFBUSxDQUFDLGNBQWMsRUFBRSxNQUFNO0lBQy9CLFFBQVEsQ0FBQyxlQUFlLEVBQUUsWUFBWTtJQUN0QyxRQUFRLENBQUMsYUFBYSxFQUFFLGFBQWE7SUFDckMsUUFBUSxDQUFDLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxDQUFDO0lBVjlCLFlBQ1csVUFBVSxFQUFFLEVBQUUsRUFDZCxVQUFVLEVBQUUsRUFBRSxFQUFFLEVBQ2hCLGFBQWEsRUFBRSxNQUFNLEVBQ3JCLFdBQVcsRUFBRSxTQUFTLEVBQ3RCLE1BQU0sRUFBRSxNQUFNLEVBQ2QsY0FBYyxFQUFFLE1BQU0sRUFDdEIsY0FBYyxFQUFFLE1BQU0sRUFDdEIsZUFBZSxFQUFFLFlBQVksRUFDN0IsYUFBYSxFQUFFLGFBQWEsRUFDNUIsTUFBTSxFQUFFLEdBQUcsQ0FBQyxNQUFNLENBQUMsRUFDMUI7SUFFSixRQUFRLENBQUMsS0FBSyxFQUFFLE1BQU0sUUFFckI7SUFFRCxRQUFRLElBQUksTUFBTSxDQWdCakI7SUFFRCxNQUFNLENBQUMsVUFBVSxDQUFDLE1BQU0sRUFBRSxNQUFNLEdBQUcsa0JBQWtCLENBMkJwRDtDQUNGIn0=
@@ -1 +1 @@
1
- {"version":3,"file":"stored_private_event.d.ts","sourceRoot":"","sources":["../../../src/storage/private_event_store/stored_private_event.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,EAAE,EAAE,MAAM,gCAAgC,CAAC;AAEpD,OAAO,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAClD,OAAO,EAAE,YAAY,EAAE,MAAM,6BAA6B,CAAC;AAC3D,OAAO,EAAE,WAAW,EAAE,MAAM,qBAAqB,CAAC;AAClD,OAAO,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAE1C,4DAA4D;AAC5D,qBAAa,kBAAkB;IAE3B,QAAQ,CAAC,UAAU,EAAE,EAAE;IACvB,QAAQ,CAAC,UAAU,EAAE,EAAE,EAAE;IACzB,QAAQ,CAAC,aAAa,EAAE,MAAM;IAC9B,QAAQ,CAAC,WAAW,EAAE,WAAW;IACjC,QAAQ,CAAC,MAAM,EAAE,MAAM;IACvB,QAAQ,CAAC,cAAc,EAAE,MAAM;IAC/B,QAAQ,CAAC,cAAc,EAAE,MAAM;IAC/B,QAAQ,CAAC,eAAe,EAAE,YAAY;IACtC,QAAQ,CAAC,aAAa,EAAE,aAAa;IACrC,QAAQ,CAAC,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC;IAV9B,YACW,UAAU,EAAE,EAAE,EACd,UAAU,EAAE,EAAE,EAAE,EAChB,aAAa,EAAE,MAAM,EACrB,WAAW,EAAE,WAAW,EACxB,MAAM,EAAE,MAAM,EACd,cAAc,EAAE,MAAM,EACtB,cAAc,EAAE,MAAM,EACtB,eAAe,EAAE,YAAY,EAC7B,aAAa,EAAE,aAAa,EAC5B,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC,EAC1B;IAEJ,QAAQ,CAAC,KAAK,EAAE,MAAM,QAErB;IAED,QAAQ,IAAI,MAAM,CAgBjB;IAED,MAAM,CAAC,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,kBAAkB,CA2BpD;CACF"}
1
+ {"version":3,"file":"stored_private_event.d.ts","sourceRoot":"","sources":["../../../src/storage/private_event_store/stored_private_event.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,EAAE,EAAE,MAAM,gCAAgC,CAAC;AAEpD,OAAO,EAAE,aAAa,EAAE,MAAM,mBAAmB,CAAC;AAClD,OAAO,EAAE,YAAY,EAAE,MAAM,6BAA6B,CAAC;AAC3D,OAAO,EAAE,SAAS,EAAE,MAAM,qBAAqB,CAAC;AAChD,OAAO,EAAE,MAAM,EAAE,MAAM,kBAAkB,CAAC;AAE1C,4DAA4D;AAC5D,qBAAa,kBAAkB;IAE3B,QAAQ,CAAC,UAAU,EAAE,EAAE;IACvB,QAAQ,CAAC,UAAU,EAAE,EAAE,EAAE;IACzB,QAAQ,CAAC,aAAa,EAAE,MAAM;IAC9B,QAAQ,CAAC,WAAW,EAAE,SAAS;IAC/B,QAAQ,CAAC,MAAM,EAAE,MAAM;IACvB,QAAQ,CAAC,cAAc,EAAE,MAAM;IAC/B,QAAQ,CAAC,cAAc,EAAE,MAAM;IAC/B,QAAQ,CAAC,eAAe,EAAE,YAAY;IACtC,QAAQ,CAAC,aAAa,EAAE,aAAa;IACrC,QAAQ,CAAC,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC;IAV9B,YACW,UAAU,EAAE,EAAE,EACd,UAAU,EAAE,EAAE,EAAE,EAChB,aAAa,EAAE,MAAM,EACrB,WAAW,EAAE,SAAS,EACtB,MAAM,EAAE,MAAM,EACd,cAAc,EAAE,MAAM,EACtB,cAAc,EAAE,MAAM,EACtB,eAAe,EAAE,YAAY,EAC7B,aAAa,EAAE,aAAa,EAC5B,MAAM,EAAE,GAAG,CAAC,MAAM,CAAC,EAC1B;IAEJ,QAAQ,CAAC,KAAK,EAAE,MAAM,QAErB;IAED,QAAQ,IAAI,MAAM,CAgBjB;IAED,MAAM,CAAC,UAAU,CAAC,MAAM,EAAE,MAAM,GAAG,kBAAkB,CA2BpD;CACF"}
@@ -2,7 +2,7 @@ import { Fr } from '@aztec/foundation/curves/bn254';
2
2
  import { BufferReader, serializeToBuffer } from '@aztec/foundation/serialize';
3
3
  import { EventSelector } from '@aztec/stdlib/abi';
4
4
  import { AztecAddress } from '@aztec/stdlib/aztec-address';
5
- import { L2BlockHash } from '@aztec/stdlib/block';
5
+ import { BlockHash } from '@aztec/stdlib/block';
6
6
  import { TxHash } from '@aztec/stdlib/tx';
7
7
  /** Serializable private event entry with scope tracking. */ export class StoredPrivateEvent {
8
8
  randomness;
@@ -42,7 +42,7 @@ import { TxHash } from '@aztec/stdlib/tx';
42
42
  const msgContentLength = reader.readNumber();
43
43
  const msgContent = reader.readArray(msgContentLength, Fr);
44
44
  const l2BlockNumber = reader.readNumber();
45
- const l2BlockHash = L2BlockHash.fromBuffer(reader);
45
+ const l2BlockHash = new BlockHash(Fr.fromBuffer(reader));
46
46
  const txHash = TxHash.fromBuffer(reader);
47
47
  const txIndexInBlock = reader.readNumber();
48
48
  const eventIndexInTx = reader.readNumber();