@aztec/pxe 0.0.1-commit.1a99e26c → 0.0.1-commit.1bb068fb5

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 (127) 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 +4 -2
  5. package/dest/block_synchronizer/block_synchronizer.d.ts.map +1 -1
  6. package/dest/block_synchronizer/block_synchronizer.js +7 -1
  7. package/dest/contract_function_simulator/contract_function_simulator.d.ts +54 -30
  8. package/dest/contract_function_simulator/contract_function_simulator.d.ts.map +1 -1
  9. package/dest/contract_function_simulator/contract_function_simulator.js +169 -65
  10. package/dest/contract_function_simulator/oracle/interfaces.d.ts +9 -9
  11. package/dest/contract_function_simulator/oracle/interfaces.d.ts.map +1 -1
  12. package/dest/contract_function_simulator/oracle/oracle.d.ts +5 -5
  13. package/dest/contract_function_simulator/oracle/oracle.d.ts.map +1 -1
  14. package/dest/contract_function_simulator/oracle/oracle.js +32 -20
  15. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts +36 -36
  16. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts.map +1 -1
  17. package/dest/contract_function_simulator/oracle/private_execution_oracle.js +74 -21
  18. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts +48 -24
  19. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts.map +1 -1
  20. package/dest/contract_function_simulator/oracle/utility_execution_oracle.js +85 -61
  21. package/dest/contract_sync/contract_sync_service.d.ts +43 -0
  22. package/dest/contract_sync/contract_sync_service.d.ts.map +1 -0
  23. package/dest/contract_sync/contract_sync_service.js +97 -0
  24. package/dest/contract_sync/helpers.d.ts +29 -0
  25. package/dest/contract_sync/helpers.d.ts.map +1 -0
  26. package/dest/contract_sync/{index.js → helpers.js} +13 -12
  27. package/dest/debug/pxe_debug_utils.d.ts +14 -10
  28. package/dest/debug/pxe_debug_utils.d.ts.map +1 -1
  29. package/dest/debug/pxe_debug_utils.js +16 -15
  30. package/dest/entrypoints/client/bundle/index.d.ts +3 -1
  31. package/dest/entrypoints/client/bundle/index.d.ts.map +1 -1
  32. package/dest/entrypoints/client/bundle/index.js +2 -0
  33. package/dest/entrypoints/client/bundle/utils.d.ts +1 -1
  34. package/dest/entrypoints/client/bundle/utils.d.ts.map +1 -1
  35. package/dest/entrypoints/client/bundle/utils.js +11 -2
  36. package/dest/entrypoints/client/lazy/index.d.ts +3 -1
  37. package/dest/entrypoints/client/lazy/index.d.ts.map +1 -1
  38. package/dest/entrypoints/client/lazy/index.js +2 -0
  39. package/dest/entrypoints/client/lazy/utils.d.ts +1 -1
  40. package/dest/entrypoints/client/lazy/utils.d.ts.map +1 -1
  41. package/dest/entrypoints/client/lazy/utils.js +11 -2
  42. package/dest/entrypoints/server/index.d.ts +4 -2
  43. package/dest/entrypoints/server/index.d.ts.map +1 -1
  44. package/dest/entrypoints/server/index.js +3 -1
  45. package/dest/entrypoints/server/utils.js +9 -1
  46. package/dest/events/event_service.d.ts +4 -5
  47. package/dest/events/event_service.d.ts.map +1 -1
  48. package/dest/events/event_service.js +5 -6
  49. package/dest/logs/log_service.d.ts +6 -5
  50. package/dest/logs/log_service.d.ts.map +1 -1
  51. package/dest/logs/log_service.js +14 -24
  52. package/dest/notes/note_service.d.ts +7 -7
  53. package/dest/notes/note_service.d.ts.map +1 -1
  54. package/dest/notes/note_service.js +9 -9
  55. package/dest/notes_filter.d.ts +25 -0
  56. package/dest/notes_filter.d.ts.map +1 -0
  57. package/dest/notes_filter.js +4 -0
  58. package/dest/oracle_version.d.ts +3 -3
  59. package/dest/oracle_version.d.ts.map +1 -1
  60. package/dest/oracle_version.js +2 -2
  61. package/dest/pxe.d.ts +69 -23
  62. package/dest/pxe.d.ts.map +1 -1
  63. package/dest/pxe.js +72 -44
  64. package/dest/storage/address_store/address_store.d.ts +1 -1
  65. package/dest/storage/address_store/address_store.d.ts.map +1 -1
  66. package/dest/storage/address_store/address_store.js +12 -11
  67. package/dest/storage/anchor_block_store/anchor_block_store.d.ts +9 -1
  68. package/dest/storage/anchor_block_store/anchor_block_store.d.ts.map +1 -1
  69. package/dest/storage/anchor_block_store/anchor_block_store.js +8 -1
  70. package/dest/storage/capsule_store/capsule_store.js +6 -8
  71. package/dest/storage/contract_store/contract_store.d.ts +1 -1
  72. package/dest/storage/contract_store/contract_store.d.ts.map +1 -1
  73. package/dest/storage/contract_store/contract_store.js +27 -18
  74. package/dest/storage/metadata.d.ts +1 -1
  75. package/dest/storage/metadata.js +1 -1
  76. package/dest/storage/note_store/note_store.d.ts +13 -3
  77. package/dest/storage/note_store/note_store.d.ts.map +1 -1
  78. package/dest/storage/note_store/note_store.js +173 -131
  79. package/dest/storage/private_event_store/private_event_store.d.ts +1 -1
  80. package/dest/storage/private_event_store/private_event_store.d.ts.map +1 -1
  81. package/dest/storage/private_event_store/private_event_store.js +126 -101
  82. package/dest/storage/private_event_store/stored_private_event.js +1 -1
  83. package/dest/storage/tagging_store/recipient_tagging_store.d.ts +1 -1
  84. package/dest/storage/tagging_store/recipient_tagging_store.d.ts.map +1 -1
  85. package/dest/storage/tagging_store/recipient_tagging_store.js +31 -19
  86. package/dest/storage/tagging_store/sender_address_book_store.d.ts +1 -1
  87. package/dest/storage/tagging_store/sender_address_book_store.d.ts.map +1 -1
  88. package/dest/storage/tagging_store/sender_address_book_store.js +20 -14
  89. package/dest/storage/tagging_store/sender_tagging_store.d.ts +1 -1
  90. package/dest/storage/tagging_store/sender_tagging_store.d.ts.map +1 -1
  91. package/dest/storage/tagging_store/sender_tagging_store.js +183 -113
  92. package/package.json +25 -16
  93. package/src/access_scopes.ts +9 -0
  94. package/src/block_synchronizer/block_synchronizer.ts +6 -0
  95. package/src/contract_function_simulator/contract_function_simulator.ts +317 -124
  96. package/src/contract_function_simulator/oracle/interfaces.ts +10 -10
  97. package/src/contract_function_simulator/oracle/oracle.ts +35 -18
  98. package/src/contract_function_simulator/oracle/private_execution_oracle.ts +97 -101
  99. package/src/contract_function_simulator/oracle/utility_execution_oracle.ts +129 -63
  100. package/src/contract_sync/contract_sync_service.ts +152 -0
  101. package/src/contract_sync/{index.ts → helpers.ts} +21 -21
  102. package/src/debug/pxe_debug_utils.ts +48 -18
  103. package/src/entrypoints/client/bundle/index.ts +2 -0
  104. package/src/entrypoints/client/bundle/utils.ts +12 -2
  105. package/src/entrypoints/client/lazy/index.ts +2 -0
  106. package/src/entrypoints/client/lazy/utils.ts +12 -2
  107. package/src/entrypoints/server/index.ts +3 -1
  108. package/src/entrypoints/server/utils.ts +7 -7
  109. package/src/events/event_service.ts +4 -6
  110. package/src/logs/log_service.ts +14 -29
  111. package/src/notes/note_service.ts +9 -10
  112. package/src/notes_filter.ts +26 -0
  113. package/src/oracle_version.ts +2 -2
  114. package/src/pxe.ts +151 -88
  115. package/src/storage/address_store/address_store.ts +15 -15
  116. package/src/storage/anchor_block_store/anchor_block_store.ts +8 -0
  117. package/src/storage/capsule_store/capsule_store.ts +8 -8
  118. package/src/storage/contract_store/contract_store.ts +26 -15
  119. package/src/storage/metadata.ts +1 -1
  120. package/src/storage/note_store/note_store.ts +195 -153
  121. package/src/storage/private_event_store/private_event_store.ts +151 -128
  122. package/src/storage/private_event_store/stored_private_event.ts +1 -1
  123. package/src/storage/tagging_store/recipient_tagging_store.ts +31 -21
  124. package/src/storage/tagging_store/sender_address_book_store.ts +20 -14
  125. package/src/storage/tagging_store/sender_tagging_store.ts +210 -126
  126. package/dest/contract_sync/index.d.ts +0 -23
  127. package/dest/contract_sync/index.d.ts.map +0 -1
@@ -1,12 +1,12 @@
1
- import { toArray } from '@aztec/foundation/iterable';
2
1
  import { Semaphore } from '@aztec/foundation/queue';
3
2
  import type { Fr } from '@aztec/foundation/schemas';
4
3
  import type { AztecAsyncKVStore, AztecAsyncMap, AztecAsyncMultiMap } from '@aztec/kv-store';
5
4
  import type { AztecAddress } from '@aztec/stdlib/aztec-address';
6
5
  import type { DataInBlock } from '@aztec/stdlib/block';
7
- import { NoteDao, NoteStatus, type NotesFilter } from '@aztec/stdlib/note';
6
+ import { NoteDao, NoteStatus } from '@aztec/stdlib/note';
8
7
 
9
8
  import type { StagedStore } from '../../job_coordinator/job_coordinator.js';
9
+ import type { NotesFilter } from '../../notes_filter.js';
10
10
  import { StoredNote } from './stored_note.js';
11
11
 
12
12
  /**
@@ -18,6 +18,8 @@ import { StoredNote } from './stored_note.js';
18
18
  export class NoteStore implements StagedStore {
19
19
  readonly storeName: string = 'note';
20
20
 
21
+ #store: AztecAsyncKVStore;
22
+
21
23
  // Note that we use the siloedNullifier as the note id in the store as it's guaranteed to be unique.
22
24
 
23
25
  // Main storage for notes. Avoid performing full scans on it as it contains all notes PXE knows, use
@@ -46,6 +48,7 @@ export class NoteStore implements StagedStore {
46
48
  #jobLocks: Map<string, Semaphore>;
47
49
 
48
50
  constructor(store: AztecAsyncKVStore) {
51
+ this.#store = store;
49
52
  this.#notes = store.openMap('notes');
50
53
  this.#nullifiersByContractAddress = store.openMultiMap('note_nullifiers_by_contract');
51
54
  this.#nullifiersByNullificationBlockNumber = store.openMultiMap('note_block_number_to_nullifier');
@@ -65,32 +68,26 @@ export class NoteStore implements StagedStore {
65
68
  * @param jobId - The job context for staged writes
66
69
  */
67
70
  public addNotes(notes: NoteDao[], scope: AztecAddress, jobId: string): Promise<void[]> {
68
- return this.#withJobLock(jobId, () => Promise.all(notes.map(noteDao => this.#addNote(noteDao, scope, jobId))));
69
- }
70
-
71
- async #addNote(note: NoteDao, scope: AztecAddress, jobId: string) {
72
- const noteForJob =
73
- (await this.#readNote(note.siloedNullifier.toString(), jobId)) ?? new StoredNote(note, new Set());
74
-
75
- // Make sure the note is linked to the scope and staged for this job
76
- noteForJob.addScope(scope.toString());
77
- this.#writeNote(noteForJob, jobId);
71
+ return this.#withJobLock(jobId, () =>
72
+ this.#store.transactionAsync(() =>
73
+ Promise.all(
74
+ notes.map(async note => {
75
+ const noteForJob =
76
+ (await this.#readNote(note.siloedNullifier.toString(), jobId)) ?? new StoredNote(note, new Set());
77
+ noteForJob.addScope(scope.toString());
78
+ this.#writeNote(noteForJob, jobId);
79
+ }),
80
+ ),
81
+ ),
82
+ );
78
83
  }
79
84
 
80
85
  async #readNote(nullifier: string, jobId: string): Promise<StoredNote | undefined> {
81
- // First check staged notes for this job
82
- const noteForJob = this.#getNotesForJob(jobId).get(nullifier);
83
- if (noteForJob) {
84
- return noteForJob;
85
- }
86
-
87
- // Then check persistent storage
86
+ // Always issue DB read to keep IndexedDB transaction alive (they auto-commit when a new micro-task starts and there
87
+ // are no pending read requests). The staged value still takes precedence if it exists.
88
88
  const noteBuffer = await this.#notes.getAsync(nullifier);
89
- if (noteBuffer) {
90
- return StoredNote.fromBuffer(noteBuffer);
91
- }
92
-
93
- return undefined;
89
+ const noteForJob = this.#getNotesForJob(jobId).get(nullifier);
90
+ return noteForJob ?? (noteBuffer ? StoredNote.fromBuffer(noteBuffer) : undefined);
94
91
  }
95
92
 
96
93
  #writeNote(note: StoredNote, jobId: string) {
@@ -107,59 +104,102 @@ export class NoteStore implements StagedStore {
107
104
  * @params jobId - the job context to read from.
108
105
  * @returns Filtered and deduplicated notes (a note might be present in multiple scopes - we ensure it is only
109
106
  * returned once if this is the case)
110
- * @throws If filtering by an empty scopes array. Scopes have to be set to undefined or to a non-empty array.
111
107
  */
112
- async getNotes(filter: NotesFilter, jobId: string): Promise<NoteDao[]> {
113
- if (filter.scopes !== undefined && filter.scopes.length === 0) {
114
- throw new Error('Trying to get notes with an empty scopes array');
108
+ getNotes(filter: NotesFilter, jobId: string): Promise<NoteDao[]> {
109
+ if (filter.scopes !== 'ALL_SCOPES' && filter.scopes.length === 0) {
110
+ return Promise.resolve([]);
115
111
  }
116
112
 
117
- const targetStatus = filter.status ?? NoteStatus.ACTIVE;
113
+ return this.#store.transactionAsync(async () => {
114
+ const targetStatus = filter.status ?? NoteStatus.ACTIVE;
115
+
116
+ // The code below might read a bit unnatural, the reason is that we need to be careful in how we use `await` inside
117
+ // `transactionAsync`, otherwise browsers might choose to auto-commit the IndexedDB transaction forcing us to
118
+ // explicitly handle that condition. The rule we need to honor is: do not await unless you generate a database
119
+ // read or write or you're done using the DB for the remainder of the transaction. The following sequence is
120
+ // unsafe in IndexedDB:
121
+ //
122
+ // 1. start transactionAsync()
123
+ // 2. await readDb() <-- OK, transaction alive because we issued DB ops
124
+ // 3. run a bunch of computations (no await involved) <-- OK, tx alive because we are in the same microtask
125
+ // 4. await doSthNotInDb() <-- no DB ops issued in this task, browser's free to decide to commit the tx
126
+ // 5. await readDb() <-- BOOM, TransactionInactiveError
127
+ //
128
+ // Note that the real issue is in step number 5: we try to continue using a transaction that the browser might
129
+ // have already committed.
130
+ //
131
+ // We need to read candidate notes which are either indexed by contract address in the DB (in
132
+ // #nullifiersByContractAddress), or lie in memory for the not yet committed `jobId`.
133
+ // So we collect promises based on both sources without awaiting for them.
134
+ const noteReadPromises: Map<string, Promise<StoredNote | undefined>> = new Map();
135
+
136
+ // Awaiting the getValuesAsync iterator is fine because it's reading from the DB
137
+ for await (const nullifier of this.#nullifiersByContractAddress.getValuesAsync(
138
+ filter.contractAddress.toString(),
139
+ )) {
140
+ // Each #readNote will perform a DB read
141
+ noteReadPromises.set(nullifier, this.#readNote(nullifier, jobId));
142
+ }
118
143
 
119
- const foundNotes: Map<string, NoteDao> = new Map();
144
+ // Add staged nullifiers from job, no awaits involved, so we are fine
145
+ for (const storedNote of this.#getNotesForJob(jobId).values()) {
146
+ if (storedNote.noteDao.contractAddress.equals(filter.contractAddress)) {
147
+ const nullifier = storedNote.noteDao.siloedNullifier.toString();
148
+ if (!noteReadPromises.has(nullifier)) {
149
+ noteReadPromises.set(nullifier, Promise.resolve(storedNote));
150
+ }
151
+ }
152
+ }
120
153
 
121
- const nullifiersOfContract = await this.#nullifiersOfContract(filter.contractAddress, jobId);
122
- for (const nullifier of nullifiersOfContract) {
123
- const note = await this.#readNote(nullifier, jobId);
154
+ // By now we have pending DB requests from all the #readNote calls. Await them all together.
155
+ const notes = await Promise.all(noteReadPromises.values());
124
156
 
125
- // Defensive: hitting this case means we're mishandling contract indices or in-memory job data
126
- if (!note) {
127
- throw new Error('PXE note database is corrupted.');
128
- }
157
+ // The rest of the function is await-free, and just deals with filtering and sorting our findings.
158
+ const foundNotes: Map<string, NoteDao> = new Map();
129
159
 
130
- // Apply filters
131
- if (targetStatus === NoteStatus.ACTIVE && note.isNullified()) {
132
- continue;
133
- }
160
+ for (const note of notes) {
161
+ // Defensive: hitting this case means we're mishandling contract indices or in-memory job data
162
+ if (!note) {
163
+ throw new Error('PXE note database is corrupted.');
164
+ }
134
165
 
135
- if (filter.owner && !note.noteDao.owner.equals(filter.owner)) {
136
- continue;
137
- }
166
+ // Apply filters
167
+ if (targetStatus === NoteStatus.ACTIVE && note.isNullified()) {
168
+ continue;
169
+ }
138
170
 
139
- if (filter.storageSlot && !note.noteDao.storageSlot.equals(filter.storageSlot)) {
140
- continue;
141
- }
171
+ if (filter.owner && !note.noteDao.owner.equals(filter.owner)) {
172
+ continue;
173
+ }
142
174
 
143
- if (filter.siloedNullifier && !note.noteDao.siloedNullifier.equals(filter.siloedNullifier)) {
144
- continue;
145
- }
175
+ if (filter.storageSlot && !note.noteDao.storageSlot.equals(filter.storageSlot)) {
176
+ continue;
177
+ }
146
178
 
147
- if (filter.scopes && note.scopes.intersection(new Set(filter.scopes.map(s => s.toString()))).size === 0) {
148
- continue;
149
- }
179
+ if (filter.siloedNullifier && !note.noteDao.siloedNullifier.equals(filter.siloedNullifier)) {
180
+ continue;
181
+ }
150
182
 
151
- foundNotes.set(note.noteDao.siloedNullifier.toString(), note.noteDao);
152
- }
183
+ if (
184
+ filter.scopes !== 'ALL_SCOPES' &&
185
+ note.scopes.intersection(new Set(filter.scopes.map(s => s.toString()))).size === 0
186
+ ) {
187
+ continue;
188
+ }
153
189
 
154
- // Sort by block number, then by tx index within block, then by note index within tx
155
- return [...foundNotes.values()].sort((a, b) => {
156
- if (a.l2BlockNumber !== b.l2BlockNumber) {
157
- return a.l2BlockNumber - b.l2BlockNumber;
190
+ foundNotes.set(note.noteDao.siloedNullifier.toString(), note.noteDao);
158
191
  }
159
- if (a.txIndexInBlock !== b.txIndexInBlock) {
160
- return a.txIndexInBlock - b.txIndexInBlock;
161
- }
162
- return a.noteIndexInTx - b.noteIndexInTx;
192
+
193
+ // Sort by block number, then by tx index within block, then by note index within tx
194
+ return [...foundNotes.values()].sort((a, b) => {
195
+ if (a.l2BlockNumber !== b.l2BlockNumber) {
196
+ return a.l2BlockNumber - b.l2BlockNumber;
197
+ }
198
+ if (a.txIndexInBlock !== b.txIndexInBlock) {
199
+ return a.txIndexInBlock - b.txIndexInBlock;
200
+ }
201
+ return a.noteIndexInTx - b.noteIndexInTx;
202
+ });
163
203
  });
164
204
  }
165
205
 
@@ -179,41 +219,46 @@ export class NoteStore implements StagedStore {
179
219
  * @throws Error if any nullifier is not found in this notes store
180
220
  */
181
221
  applyNullifiers(nullifiers: DataInBlock<Fr>[], jobId: string): Promise<NoteDao[]> {
182
- return this.#withJobLock(jobId, async () => {
183
- if (nullifiers.length === 0) {
184
- return [];
185
- }
222
+ if (nullifiers.length === 0) {
223
+ return Promise.resolve([]);
224
+ }
186
225
 
187
- const notesToNullify = await Promise.all(
188
- nullifiers.map(async nullifierInBlock => {
189
- const nullifier = nullifierInBlock.data.toString();
226
+ if (nullifiers.some(n => n.l2BlockNumber === 0)) {
227
+ return Promise.reject(new Error('applyNullifiers: nullifiers cannot have been emitted at block 0'));
228
+ }
190
229
 
191
- const storedNote = await this.#readNote(nullifier, jobId);
192
- if (!storedNote) {
193
- throw new Error(`Attempted to mark a note as nullified which does not exist in PXE DB`);
194
- }
230
+ return this.#withJobLock(jobId, () =>
231
+ this.#store.transactionAsync(async () => {
232
+ const notesToNullify = await Promise.all(
233
+ nullifiers.map(async nullifierInBlock => {
234
+ const nullifier = nullifierInBlock.data.toString();
195
235
 
196
- return { storedNote: await this.#readNote(nullifier, jobId), blockNumber: nullifierInBlock.l2BlockNumber };
197
- }),
198
- );
236
+ const storedNote = await this.#readNote(nullifier, jobId);
237
+ if (!storedNote) {
238
+ throw new Error(`Attempted to mark a note as nullified which does not exist in PXE DB`);
239
+ }
199
240
 
200
- const notesNullifiedInThisCall: Map<string, NoteDao> = new Map();
201
- for (const noteToNullify of notesToNullify) {
202
- // Safe to coerce (!) because we throw if we find any undefined above
203
- const note = noteToNullify.storedNote!;
241
+ return { storedNote, blockNumber: nullifierInBlock.l2BlockNumber };
242
+ }),
243
+ );
204
244
 
205
- // Skip already nullified notes
206
- if (note.isNullified()) {
207
- continue;
208
- }
245
+ const notesNullifiedInThisCall: Map<string, NoteDao> = new Map();
246
+ for (const noteToNullify of notesToNullify) {
247
+ const note = noteToNullify.storedNote;
209
248
 
210
- note.markAsNullified(noteToNullify.blockNumber);
211
- this.#writeNote(note, jobId);
212
- notesNullifiedInThisCall.set(note.noteDao.siloedNullifier.toString(), note.noteDao);
213
- }
249
+ // Skip already nullified notes
250
+ if (note.isNullified()) {
251
+ continue;
252
+ }
214
253
 
215
- return [...notesNullifiedInThisCall.values()];
216
- });
254
+ note.markAsNullified(noteToNullify.blockNumber);
255
+ this.#writeNote(note, jobId);
256
+ notesNullifiedInThisCall.set(note.noteDao.siloedNullifier.toString(), note.noteDao);
257
+ }
258
+
259
+ return [...notesNullifiedInThisCall.values()];
260
+ }),
261
+ );
217
262
  }
218
263
 
219
264
  /**
@@ -244,18 +289,23 @@ export class NoteStore implements StagedStore {
244
289
  * @param blockNumber - Notes created after this block number will be deleted
245
290
  */
246
291
  async #deleteActiveNotesAfterBlock(blockNumber: number): Promise<void> {
247
- const notes = await toArray(this.#notes.valuesAsync());
248
- for (const noteBuffer of notes) {
292
+ // Collect notes to delete during iteration to keep IndexedDB transaction alive.
293
+ const notesToDelete: { nullifier: string; contractAddress: string }[] = [];
294
+ for await (const noteBuffer of this.#notes.valuesAsync()) {
249
295
  const storedNote = StoredNote.fromBuffer(noteBuffer);
250
296
  if (storedNote.noteDao.l2BlockNumber > blockNumber) {
251
- const noteNullifier = storedNote.noteDao.siloedNullifier.toString();
252
- await this.#notes.delete(noteNullifier);
253
- await this.#nullifiersByContractAddress.deleteValue(
254
- storedNote.noteDao.contractAddress.toString(),
255
- noteNullifier,
256
- );
297
+ notesToDelete.push({
298
+ nullifier: storedNote.noteDao.siloedNullifier.toString(),
299
+ contractAddress: storedNote.noteDao.contractAddress.toString(),
300
+ });
257
301
  }
258
302
  }
303
+
304
+ // Delete all collected notes. Each delete is a DB operation that keeps the transaction alive.
305
+ for (const { nullifier, contractAddress } of notesToDelete) {
306
+ await this.#notes.delete(nullifier);
307
+ await this.#nullifiersByContractAddress.deleteValue(contractAddress, nullifier);
308
+ }
259
309
  }
260
310
 
261
311
  /**
@@ -268,62 +318,69 @@ export class NoteStore implements StagedStore {
268
318
  * @param anchorBlockNumber - Upper bound for the block range to process
269
319
  */
270
320
  async #rewindNullifiedNotesAfterBlock(blockNumber: number, anchorBlockNumber: number): Promise<void> {
271
- const currentBlockNumber = blockNumber + 1;
272
- for (let i = currentBlockNumber; i <= anchorBlockNumber; i++) {
273
- const noteNullifiersToReinsert: string[] = await toArray(
274
- this.#nullifiersByNullificationBlockNumber.getValuesAsync(i),
275
- );
276
-
277
- const nullifiedNoteBuffers = await Promise.all(
278
- noteNullifiersToReinsert.map(async noteNullifier => {
279
- const note = await this.#notes.getAsync(noteNullifier);
280
-
281
- if (!note) {
282
- throw new Error(`PXE DB integrity error: no note found with nullifier ${noteNullifier}`);
283
- }
284
-
285
- return note;
286
- }),
287
- );
288
-
289
- const storedNotes = nullifiedNoteBuffers.map(buffer => StoredNote.fromBuffer(buffer));
321
+ // First pass: collect all nullifiers for all blocks, starting reads during iteration to keep tx alive.
322
+ const nullifiersByBlock: Map<number, { nullifier: string; noteReadPromise: Promise<Buffer | undefined> }[]> =
323
+ new Map();
324
+
325
+ for (let i = blockNumber + 1; i <= anchorBlockNumber; i++) {
326
+ const blockNullifiers: { nullifier: string; noteReadPromise: Promise<Buffer | undefined> }[] = [];
327
+ for await (const nullifier of this.#nullifiersByNullificationBlockNumber.getValuesAsync(i)) {
328
+ // Start read immediately during iteration to keep IndexedDB transaction alive
329
+ blockNullifiers.push({ nullifier, noteReadPromise: this.#notes.getAsync(nullifier) });
330
+ }
331
+ if (blockNullifiers.length > 0) {
332
+ nullifiersByBlock.set(i, blockNullifiers);
333
+ }
334
+ }
290
335
 
291
- for (const storedNote of storedNotes) {
292
- const noteNullifier = storedNote.noteDao.siloedNullifier.toString();
293
- const scopes = storedNote.scopes;
336
+ // Second pass: await reads and perform writes
337
+ for (const [block, nullifiers] of nullifiersByBlock) {
338
+ for (const { nullifier, noteReadPromise } of nullifiers) {
339
+ const noteBuffer = await noteReadPromise;
340
+ if (!noteBuffer) {
341
+ throw new Error(`PXE DB integrity error: no note found with nullifier ${nullifier}`);
342
+ }
294
343
 
295
- if (scopes.size === 0) {
296
- // We should never run into this error because notes always have a scope assigned to them - either on initial
297
- // insertion via `addNotes` or when removing their nullifiers.
298
- throw new Error(`No scopes found for nullified note with nullifier ${noteNullifier}`);
344
+ const storedNote = StoredNote.fromBuffer(noteBuffer);
345
+ if (storedNote.scopes.size === 0) {
346
+ throw new Error(`No scopes found for nullified note with nullifier ${nullifier}`);
299
347
  }
300
348
 
301
349
  storedNote.markAsActive();
302
350
 
303
351
  await Promise.all([
304
- this.#notes.set(noteNullifier, storedNote.toBuffer()),
305
- this.#nullifiersByNullificationBlockNumber.deleteValue(i, noteNullifier),
352
+ this.#notes.set(nullifier, storedNote.toBuffer()),
353
+ this.#nullifiersByNullificationBlockNumber.deleteValue(block, nullifier),
306
354
  ]);
307
355
  }
308
356
  }
309
357
  }
310
358
 
311
- commit(jobId: string): Promise<void> {
312
- return this.#withJobLock(jobId, async () => {
313
- for (const [nullifier, storedNote] of this.#getNotesForJob(jobId)) {
314
- await this.#notes.set(nullifier, storedNote.toBuffer());
315
- await this.#nullifiersByContractAddress.set(storedNote.noteDao.contractAddress.toString(), nullifier);
316
- if (storedNote.nullifiedAt !== undefined) {
317
- await this.#nullifiersByNullificationBlockNumber.set(storedNote.nullifiedAt, nullifier);
318
- }
359
+ /**
360
+ * Commits in memory job data to persistent storage.
361
+ *
362
+ * Called by JobCoordinator when a job completes successfully.
363
+ *
364
+ * Note: JobCoordinator wraps all commits in a single transaction, so we don't need our own transactionAsync here
365
+ * (and using one would throw on IndexedDB as it does not support nested txs).
366
+ *
367
+ * @param jobId - The jobId identifying which staged data to commit
368
+ */
369
+ async commit(jobId: string): Promise<void> {
370
+ for (const [nullifier, storedNote] of this.#getNotesForJob(jobId)) {
371
+ await this.#notes.set(nullifier, storedNote.toBuffer());
372
+ await this.#nullifiersByContractAddress.set(storedNote.noteDao.contractAddress.toString(), nullifier);
373
+ if (storedNote.nullifiedAt !== undefined) {
374
+ await this.#nullifiersByNullificationBlockNumber.set(storedNote.nullifiedAt, nullifier);
319
375
  }
376
+ }
320
377
 
321
- this.#clearJobData(jobId);
322
- });
378
+ this.#clearJobData(jobId);
323
379
  }
324
380
 
325
381
  discardStaged(jobId: string): Promise<void> {
326
- return this.#withJobLock(jobId, () => Promise.resolve(this.#clearJobData(jobId)));
382
+ this.#clearJobData(jobId);
383
+ return Promise.resolve();
327
384
  }
328
385
 
329
386
  #clearJobData(jobId: string) {
@@ -358,19 +415,4 @@ export class NoteStore implements StagedStore {
358
415
  }
359
416
  return notesForJob;
360
417
  }
361
-
362
- async #nullifiersOfContract(contractAddress: AztecAddress, jobId: string): Promise<Set<string>> {
363
- // Collect persisted nullifiers for this contract
364
- const persistedNullifiers: string[] = await toArray(
365
- this.#nullifiersByContractAddress.getValuesAsync(contractAddress.toString()),
366
- );
367
-
368
- // Collect staged nullifiers from the job where the note's contract matches
369
- const stagedNullifiers = this.#getNotesForJob(jobId)
370
- .values()
371
- .filter(storedNote => storedNote.noteDao.contractAddress.equals(contractAddress))
372
- .map(storedNote => storedNote.noteDao.siloedNullifier.toString());
373
-
374
- return new Set([...persistedNullifiers, ...stagedNullifiers]);
375
- }
376
418
  }