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

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 (183) hide show
  1. package/dest/bin/check_oracle_version.js +4 -4
  2. package/dest/block_synchronizer/block_synchronizer.d.ts +1 -1
  3. package/dest/block_synchronizer/block_synchronizer.d.ts.map +1 -1
  4. package/dest/block_synchronizer/block_synchronizer.js +6 -0
  5. package/dest/config/index.d.ts +2 -2
  6. package/dest/config/index.d.ts.map +1 -1
  7. package/dest/config/index.js +1 -1
  8. package/dest/contract_function_simulator/contract_function_simulator.d.ts +10 -5
  9. package/dest/contract_function_simulator/contract_function_simulator.d.ts.map +1 -1
  10. package/dest/contract_function_simulator/contract_function_simulator.js +31 -8
  11. package/dest/contract_function_simulator/ephemeral_array_service.d.ts +28 -0
  12. package/dest/contract_function_simulator/ephemeral_array_service.d.ts.map +1 -0
  13. package/dest/contract_function_simulator/ephemeral_array_service.js +78 -0
  14. package/dest/contract_function_simulator/execution_tagging_index_cache.d.ts +5 -5
  15. package/dest/contract_function_simulator/execution_tagging_index_cache.d.ts.map +1 -1
  16. package/dest/contract_function_simulator/execution_tagging_index_cache.js +17 -9
  17. package/dest/contract_function_simulator/index.d.ts +2 -1
  18. package/dest/contract_function_simulator/index.d.ts.map +1 -1
  19. package/dest/contract_function_simulator/index.js +1 -0
  20. package/dest/contract_function_simulator/noir-structs/event_validation_request.d.ts +4 -6
  21. package/dest/contract_function_simulator/noir-structs/event_validation_request.d.ts.map +1 -1
  22. package/dest/contract_function_simulator/noir-structs/event_validation_request.js +8 -10
  23. package/dest/contract_function_simulator/noir-structs/log_retrieval_request.d.ts +1 -1
  24. package/dest/contract_function_simulator/noir-structs/log_retrieval_request.js +1 -1
  25. package/dest/contract_function_simulator/noir-structs/log_retrieval_response.d.ts +2 -2
  26. package/dest/contract_function_simulator/noir-structs/log_retrieval_response.d.ts.map +1 -1
  27. package/dest/contract_function_simulator/noir-structs/log_retrieval_response.js +2 -4
  28. package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts +4 -7
  29. package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts.map +1 -1
  30. package/dest/contract_function_simulator/noir-structs/note_validation_request.js +6 -11
  31. package/dest/contract_function_simulator/oracle/interfaces.d.ts +61 -45
  32. package/dest/contract_function_simulator/oracle/interfaces.d.ts.map +1 -1
  33. package/dest/contract_function_simulator/oracle/legacy_oracle_mappings.d.ts +9 -0
  34. package/dest/contract_function_simulator/oracle/legacy_oracle_mappings.d.ts.map +1 -0
  35. package/dest/contract_function_simulator/oracle/legacy_oracle_mappings.js +47 -0
  36. package/dest/contract_function_simulator/oracle/oracle.d.ts +75 -44
  37. package/dest/contract_function_simulator/oracle/oracle.d.ts.map +1 -1
  38. package/dest/contract_function_simulator/oracle/oracle.js +281 -96
  39. package/dest/contract_function_simulator/oracle/private_execution.js +5 -3
  40. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts +23 -49
  41. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts.map +1 -1
  42. package/dest/contract_function_simulator/oracle/private_execution_oracle.js +40 -63
  43. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts +89 -56
  44. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts.map +1 -1
  45. package/dest/contract_function_simulator/oracle/utility_execution_oracle.js +272 -107
  46. package/dest/contract_function_simulator/pick_notes.d.ts +1 -1
  47. package/dest/contract_function_simulator/pick_notes.d.ts.map +1 -1
  48. package/dest/contract_function_simulator/pick_notes.js +9 -2
  49. package/dest/contract_function_simulator/proxied_contract_data_source.d.ts +1 -1
  50. package/dest/contract_function_simulator/proxied_contract_data_source.d.ts.map +1 -1
  51. package/dest/contract_function_simulator/proxied_contract_data_source.js +3 -0
  52. package/dest/contract_logging.d.ts +9 -4
  53. package/dest/contract_logging.d.ts.map +1 -1
  54. package/dest/contract_logging.js +21 -6
  55. package/dest/contract_sync/contract_sync_service.d.ts +6 -5
  56. package/dest/contract_sync/contract_sync_service.d.ts.map +1 -1
  57. package/dest/contract_sync/contract_sync_service.js +44 -37
  58. package/dest/contract_sync/helpers.d.ts +2 -3
  59. package/dest/contract_sync/helpers.d.ts.map +1 -1
  60. package/dest/contract_sync/helpers.js +7 -2
  61. package/dest/debug/pxe_debug_utils.d.ts +3 -3
  62. package/dest/debug/pxe_debug_utils.d.ts.map +1 -1
  63. package/dest/entrypoints/client/bundle/index.d.ts +1 -2
  64. package/dest/entrypoints/client/bundle/index.d.ts.map +1 -1
  65. package/dest/entrypoints/client/bundle/index.js +0 -1
  66. package/dest/entrypoints/client/bundle/utils.d.ts +2 -2
  67. package/dest/entrypoints/client/bundle/utils.d.ts.map +1 -1
  68. package/dest/entrypoints/client/bundle/utils.js +2 -2
  69. package/dest/entrypoints/client/lazy/index.d.ts +1 -2
  70. package/dest/entrypoints/client/lazy/index.d.ts.map +1 -1
  71. package/dest/entrypoints/client/lazy/index.js +0 -1
  72. package/dest/entrypoints/client/lazy/utils.d.ts +2 -2
  73. package/dest/entrypoints/client/lazy/utils.d.ts.map +1 -1
  74. package/dest/entrypoints/client/lazy/utils.js +2 -2
  75. package/dest/entrypoints/pxe_creation_options.d.ts +3 -1
  76. package/dest/entrypoints/pxe_creation_options.d.ts.map +1 -1
  77. package/dest/entrypoints/pxe_creation_options.js +3 -1
  78. package/dest/entrypoints/server/index.d.ts +2 -3
  79. package/dest/entrypoints/server/index.d.ts.map +1 -1
  80. package/dest/entrypoints/server/index.js +1 -2
  81. package/dest/entrypoints/server/utils.d.ts +2 -2
  82. package/dest/entrypoints/server/utils.d.ts.map +1 -1
  83. package/dest/entrypoints/server/utils.js +2 -2
  84. package/dest/events/event_service.d.ts +3 -2
  85. package/dest/events/event_service.d.ts.map +1 -1
  86. package/dest/events/event_service.js +16 -4
  87. package/dest/logs/log_service.d.ts +5 -8
  88. package/dest/logs/log_service.d.ts.map +1 -1
  89. package/dest/logs/log_service.js +24 -37
  90. package/dest/messages/message_context_service.d.ts +17 -0
  91. package/dest/messages/message_context_service.d.ts.map +1 -0
  92. package/dest/messages/message_context_service.js +36 -0
  93. package/dest/notes/note_service.d.ts +4 -5
  94. package/dest/notes/note_service.d.ts.map +1 -1
  95. package/dest/notes/note_service.js +14 -5
  96. package/dest/notes_filter.d.ts +2 -3
  97. package/dest/notes_filter.d.ts.map +1 -1
  98. package/dest/oracle_version.d.ts +4 -3
  99. package/dest/oracle_version.d.ts.map +1 -1
  100. package/dest/oracle_version.js +20 -10
  101. package/dest/pxe.d.ts +11 -8
  102. package/dest/pxe.d.ts.map +1 -1
  103. package/dest/pxe.js +45 -26
  104. package/dest/storage/capsule_store/capsule_service.d.ts +21 -0
  105. package/dest/storage/capsule_store/capsule_service.d.ts.map +1 -0
  106. package/dest/storage/capsule_store/capsule_service.js +50 -0
  107. package/dest/storage/capsule_store/capsule_store.d.ts +9 -9
  108. package/dest/storage/capsule_store/capsule_store.d.ts.map +1 -1
  109. package/dest/storage/capsule_store/capsule_store.js +33 -28
  110. package/dest/storage/capsule_store/index.d.ts +2 -1
  111. package/dest/storage/capsule_store/index.d.ts.map +1 -1
  112. package/dest/storage/capsule_store/index.js +1 -0
  113. package/dest/storage/metadata.d.ts +1 -1
  114. package/dest/storage/metadata.js +1 -1
  115. package/dest/storage/note_store/note_store.d.ts +1 -1
  116. package/dest/storage/note_store/note_store.d.ts.map +1 -1
  117. package/dest/storage/note_store/note_store.js +2 -2
  118. package/dest/storage/tagging_store/sender_tagging_store.d.ts +26 -25
  119. package/dest/storage/tagging_store/sender_tagging_store.d.ts.map +1 -1
  120. package/dest/storage/tagging_store/sender_tagging_store.js +141 -115
  121. package/dest/tagging/index.d.ts +2 -2
  122. package/dest/tagging/index.d.ts.map +1 -1
  123. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts +1 -1
  124. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts.map +1 -1
  125. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.js +10 -1
  126. package/dest/tagging/sender_sync/utils/get_status_change_of_pending.d.ts +4 -3
  127. package/dest/tagging/sender_sync/utils/get_status_change_of_pending.d.ts.map +1 -1
  128. package/dest/tagging/sender_sync/utils/get_status_change_of_pending.js +20 -10
  129. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts +2 -1
  130. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts.map +1 -1
  131. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.js +24 -11
  132. package/package.json +16 -16
  133. package/src/bin/check_oracle_version.ts +4 -4
  134. package/src/block_synchronizer/block_synchronizer.ts +6 -0
  135. package/src/config/index.ts +1 -1
  136. package/src/contract_function_simulator/contract_function_simulator.ts +44 -12
  137. package/src/contract_function_simulator/ephemeral_array_service.ts +110 -0
  138. package/src/contract_function_simulator/execution_tagging_index_cache.ts +16 -11
  139. package/src/contract_function_simulator/index.ts +1 -0
  140. package/src/contract_function_simulator/noir-structs/event_validation_request.ts +9 -9
  141. package/src/contract_function_simulator/noir-structs/log_retrieval_request.ts +1 -1
  142. package/src/contract_function_simulator/noir-structs/log_retrieval_response.ts +2 -5
  143. package/src/contract_function_simulator/noir-structs/note_validation_request.ts +4 -10
  144. package/src/contract_function_simulator/oracle/interfaces.ts +82 -54
  145. package/src/contract_function_simulator/oracle/legacy_oracle_mappings.ts +104 -0
  146. package/src/contract_function_simulator/oracle/oracle.ts +363 -139
  147. package/src/contract_function_simulator/oracle/private_execution.ts +4 -4
  148. package/src/contract_function_simulator/oracle/private_execution_oracle.ts +45 -84
  149. package/src/contract_function_simulator/oracle/utility_execution_oracle.ts +444 -143
  150. package/src/contract_function_simulator/pick_notes.ts +9 -2
  151. package/src/contract_function_simulator/proxied_contract_data_source.ts +8 -1
  152. package/src/contract_logging.ts +18 -5
  153. package/src/contract_sync/contract_sync_service.ts +77 -59
  154. package/src/contract_sync/helpers.ts +4 -4
  155. package/src/debug/pxe_debug_utils.ts +3 -3
  156. package/src/entrypoints/client/bundle/index.ts +0 -1
  157. package/src/entrypoints/client/bundle/utils.ts +2 -3
  158. package/src/entrypoints/client/lazy/index.ts +0 -1
  159. package/src/entrypoints/client/lazy/utils.ts +2 -3
  160. package/src/entrypoints/pxe_creation_options.ts +7 -0
  161. package/src/entrypoints/server/index.ts +1 -2
  162. package/src/entrypoints/server/utils.ts +2 -3
  163. package/src/events/event_service.ts +17 -4
  164. package/src/logs/log_service.ts +52 -78
  165. package/src/messages/message_context_service.ts +44 -0
  166. package/src/notes/note_service.ts +18 -8
  167. package/src/notes_filter.ts +1 -3
  168. package/src/oracle_version.ts +20 -10
  169. package/src/pxe.ts +68 -31
  170. package/src/storage/capsule_store/capsule_service.ts +90 -0
  171. package/src/storage/capsule_store/capsule_store.ts +34 -26
  172. package/src/storage/capsule_store/index.ts +1 -0
  173. package/src/storage/metadata.ts +1 -1
  174. package/src/storage/note_store/note_store.ts +2 -5
  175. package/src/storage/tagging_store/sender_tagging_store.ts +182 -135
  176. package/src/tagging/index.ts +1 -1
  177. package/src/tagging/sender_sync/sync_sender_tagging_indexes.ts +19 -1
  178. package/src/tagging/sender_sync/utils/get_status_change_of_pending.ts +26 -11
  179. package/src/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.ts +19 -9
  180. package/dest/access_scopes.d.ts +0 -9
  181. package/dest/access_scopes.d.ts.map +0 -1
  182. package/dest/access_scopes.js +0 -6
  183. package/src/access_scopes.ts +0 -9
@@ -1,22 +1,13 @@
1
- import type { Fr } from '@aztec/foundation/curves/bn254';
2
1
  import { type Logger, type LoggerBindings, createLogger } from '@aztec/foundation/log';
3
2
  import type { KeyStore } from '@aztec/key-store';
4
3
  import { AztecAddress } from '@aztec/stdlib/aztec-address';
5
4
  import type { AztecNode } from '@aztec/stdlib/interfaces/server';
6
- import {
7
- ExtendedDirectionalAppTaggingSecret,
8
- PendingTaggedLog,
9
- SiloedTag,
10
- Tag,
11
- TxScopedL2Log,
12
- } from '@aztec/stdlib/logs';
5
+ import { ExtendedDirectionalAppTaggingSecret, PendingTaggedLog, SiloedTag, Tag } from '@aztec/stdlib/logs';
13
6
  import type { BlockHeader } from '@aztec/stdlib/tx';
14
7
 
15
- import type { AccessScopes } from '../access_scopes.js';
16
8
  import type { LogRetrievalRequest } from '../contract_function_simulator/noir-structs/log_retrieval_request.js';
17
9
  import { LogRetrievalResponse } from '../contract_function_simulator/noir-structs/log_retrieval_response.js';
18
10
  import { AddressStore } from '../storage/address_store/address_store.js';
19
- import { CapsuleStore } from '../storage/capsule_store/capsule_store.js';
20
11
  import type { RecipientTaggingStore } from '../storage/tagging_store/recipient_tagging_store.js';
21
12
  import type { SenderAddressBookStore } from '../storage/tagging_store/sender_address_book_store.js';
22
13
  import {
@@ -32,7 +23,6 @@ export class LogService {
32
23
  private readonly aztecNode: AztecNode,
33
24
  private readonly anchorBlockHeader: BlockHeader,
34
25
  private readonly keyStore: KeyStore,
35
- private readonly capsuleStore: CapsuleStore,
36
26
  private readonly recipientTaggingStore: RecipientTaggingStore,
37
27
  private readonly senderAddressBookStore: SenderAddressBookStore,
38
28
  private readonly addressStore: AddressStore,
@@ -42,7 +32,16 @@ export class LogService {
42
32
  this.log = createLogger('pxe:log_service', bindings);
43
33
  }
44
34
 
45
- public async bulkRetrieveLogs(logRetrievalRequests: LogRetrievalRequest[]): Promise<(LogRetrievalResponse | null)[]> {
35
+ public async fetchLogsByTag(
36
+ contractAddress: AztecAddress,
37
+ logRetrievalRequests: LogRetrievalRequest[],
38
+ ): Promise<(LogRetrievalResponse | null)[]> {
39
+ for (const request of logRetrievalRequests) {
40
+ if (!contractAddress.equals(request.contractAddress)) {
41
+ throw new Error(`Got a log retrieval request from ${request.contractAddress}, expected ${contractAddress}`);
42
+ }
43
+ }
44
+
46
45
  return await Promise.all(
47
46
  logRetrievalRequests.map(async request => {
48
47
  const [publicLog, privateLog] = await Promise.all([
@@ -51,8 +50,8 @@ export class LogService {
51
50
  ]);
52
51
 
53
52
  if (publicLog !== null && privateLog !== null) {
54
- throw new Error(
55
- `Found both a public and private log when searching for tag ${request.tag} from contract ${request.contractAddress}`,
53
+ this.log.warn(
54
+ `Found both a public and private log for tag ${request.tag} from contract ${request.contractAddress}. This may indicate a contract bug. Returning the public log.`,
56
55
  );
57
56
  }
58
57
 
@@ -74,9 +73,8 @@ export class LogService {
74
73
  if (logsForTag.length === 0) {
75
74
  return null;
76
75
  } else if (logsForTag.length > 1) {
77
- // TODO(#11627): handle this case
78
- throw new Error(
79
- `Got ${logsForTag.length} logs for tag ${tag} and contract ${contractAddress.toString()}. getPublicLogByTag currently only supports a single log per tag`,
76
+ this.log.warn(
77
+ `Expected at most 1 public log for tag ${tag} and contract ${contractAddress.toString()}, got ${logsForTag.length}. This may indicate a contract bug. Returning the first log.`,
80
78
  );
81
79
  }
82
80
 
@@ -98,9 +96,8 @@ export class LogService {
98
96
  if (logsForTag.length === 0) {
99
97
  return null;
100
98
  } else if (logsForTag.length > 1) {
101
- // TODO(#11627): handle this case
102
- throw new Error(
103
- `Got ${logsForTag.length} logs for tag ${siloedTag}. getPrivateLogByTag currently only supports a single log per tag`,
99
+ this.log.warn(
100
+ `Expected at most 1 private log for tag ${siloedTag}, got ${logsForTag.length}. This may indicate a contract bug. Returning the first log.`,
104
101
  );
105
102
  }
106
103
 
@@ -114,46 +111,36 @@ export class LogService {
114
111
  );
115
112
  }
116
113
 
117
- public async fetchTaggedLogs(contractAddress: AztecAddress, pendingTaggedLogArrayBaseSlot: Fr, scopes: AccessScopes) {
114
+ public async fetchTaggedLogs(contractAddress: AztecAddress, recipient: AztecAddress): Promise<PendingTaggedLog[]> {
118
115
  this.log.verbose(`Fetching tagged logs for ${contractAddress.toString()}`);
119
116
 
120
117
  // We only load logs from block up to and including the anchor block number
121
118
  const anchorBlockNumber = this.anchorBlockHeader.getBlockNumber();
122
119
  const anchorBlockHash = await this.anchorBlockHeader.hash();
123
120
 
124
- // Determine recipients: use scopes if provided, otherwise get all accounts
125
- const recipients = scopes !== 'ALL_SCOPES' && scopes.length > 0 ? scopes : await this.keyStore.getAccounts();
126
-
127
- // For each recipient, fetch secrets, load logs, and store them.
128
- // We run these per-recipient tasks in parallel so that logs are loaded for all recipients concurrently.
129
- await Promise.all(
130
- recipients.map(async recipient => {
131
- // Get all secrets for this recipient (one per sender)
132
- const secrets = await this.#getSecretsForSenders(contractAddress, recipient);
133
-
134
- // Load logs for all sender-recipient pairs in parallel
135
- const logArrays = await Promise.all(
136
- secrets.map(secret =>
137
- loadPrivateLogsForSenderRecipientPair(
138
- secret,
139
- this.aztecNode,
140
- this.recipientTaggingStore,
141
- anchorBlockNumber,
142
- anchorBlockHash,
143
- this.jobId,
144
- ),
145
- ),
146
- );
147
-
148
- // Flatten all logs from all secrets
149
- const allLogs = logArrays.flat();
150
-
151
- // Store the logs for this recipient
152
- if (allLogs.length > 0) {
153
- await this.#storePendingTaggedLogs(contractAddress, pendingTaggedLogArrayBaseSlot, recipient, allLogs);
154
- }
155
- }),
121
+ // Get all secrets for this recipient (one per sender)
122
+ const secrets = await this.#getSecretsForSenders(contractAddress, recipient);
123
+
124
+ // Load logs for all sender-recipient pairs in parallel
125
+ const logArrays = await Promise.all(
126
+ secrets.map(secret =>
127
+ loadPrivateLogsForSenderRecipientPair(
128
+ secret,
129
+ this.aztecNode,
130
+ this.recipientTaggingStore,
131
+ anchorBlockNumber,
132
+ anchorBlockHash,
133
+ this.jobId,
134
+ ),
135
+ ),
156
136
  );
137
+
138
+ return logArrays
139
+ .flat()
140
+ .map(
141
+ scopedLog =>
142
+ new PendingTaggedLog(scopedLog.logData, scopedLog.txHash, scopedLog.noteHashes, scopedLog.firstNullifier),
143
+ );
157
144
  }
158
145
 
159
146
  async #getSecretsForSenders(
@@ -176,38 +163,25 @@ export class LogService {
176
163
  );
177
164
 
178
165
  return Promise.all(
179
- deduplicatedSenders.map(sender => {
180
- return ExtendedDirectionalAppTaggingSecret.compute(
166
+ deduplicatedSenders.map(async sender => {
167
+ const secret = await ExtendedDirectionalAppTaggingSecret.compute(
181
168
  recipientCompleteAddress,
182
169
  recipientIvsk,
183
170
  sender,
184
171
  contractAddress,
185
172
  recipient,
186
173
  );
187
- }),
188
- );
189
- }
190
-
191
- #storePendingTaggedLogs(
192
- contractAddress: AztecAddress,
193
- capsuleArrayBaseSlot: Fr,
194
- recipient: AztecAddress,
195
- privateLogs: TxScopedL2Log[],
196
- ) {
197
- // Build all pending tagged logs from the scoped logs
198
- const pendingTaggedLogs = privateLogs.map(scopedLog => {
199
- const pendingTaggedLog = new PendingTaggedLog(
200
- scopedLog.logData,
201
- scopedLog.txHash,
202
- scopedLog.noteHashes,
203
- scopedLog.firstNullifier,
204
- recipient,
205
- );
206
174
 
207
- return pendingTaggedLog.toFields();
208
- });
175
+ if (!secret) {
176
+ // Note that all senders originate from either the SenderAddressBookStore or the KeyStore.
177
+ // TODO(F-512): make sure we actually prevent registering invalid senders.
178
+ throw new Error(
179
+ `Failed to compute a tagging secret for sender ${sender} - this implies this is an invalid address, which should not happen as they have been previously registered in PXE.`,
180
+ );
181
+ }
209
182
 
210
- // TODO: This looks like it could belong more at the oracle interface level
211
- return this.capsuleStore.appendToCapsuleArray(contractAddress, capsuleArrayBaseSlot, pendingTaggedLogs, this.jobId);
183
+ return secret;
184
+ }),
185
+ );
212
186
  }
213
187
  }
@@ -0,0 +1,44 @@
1
+ import { Fr } from '@aztec/foundation/curves/bn254';
2
+ import type { AztecNode } from '@aztec/stdlib/interfaces/server';
3
+ import { MessageContext } from '@aztec/stdlib/logs';
4
+ import { TxHash } from '@aztec/stdlib/tx';
5
+
6
+ /** Resolves transaction hashes into the context needed to process messages. */
7
+ export class MessageContextService {
8
+ constructor(private readonly aztecNode: AztecNode) {}
9
+
10
+ /**
11
+ * Resolves a list of tx hashes into their message contexts.
12
+ *
13
+ * For each tx hash, looks up the corresponding tx effect and extracts the note hashes and first nullifier needed to
14
+ * process messages that originated from that transaction. Returns `null` for tx hashes that are zero, not yet
15
+ * available, or in blocks beyond the anchor block.
16
+ */
17
+ getMessageContextsByTxHash(txHashes: Fr[], anchorBlockNumber: number): Promise<(MessageContext | null)[]> {
18
+ // TODO: optimize, we might be hitting the node to get the same txHash repeatedly
19
+ return Promise.all(
20
+ txHashes.map(async txHashField => {
21
+ // A zero tx hash indicates a tx-less offchain message (e.g. one not tied to any onchain transaction).
22
+ // These messages don't have a transaction context to resolve, so we return null.
23
+ if (txHashField.isZero()) {
24
+ return null;
25
+ }
26
+
27
+ const txHash = TxHash.fromField(txHashField);
28
+ const txEffect = await this.aztecNode.getTxEffect(txHash);
29
+ if (!txEffect || txEffect.l2BlockNumber > anchorBlockNumber) {
30
+ return null;
31
+ }
32
+
33
+ // Every tx has at least one nullifier (the first nullifier derived from the tx hash). Hitting this condition
34
+ // would mean a buggy node, but since we need to access data.nullifiers[0], the defensive check does no harm.
35
+ const data = txEffect.data;
36
+ if (data.nullifiers.length === 0) {
37
+ throw new Error(`Tx effect for ${txHash} has no nullifiers`);
38
+ }
39
+
40
+ return new MessageContext(data.txHash, data.noteHashes, data.nullifiers[0]);
41
+ }),
42
+ );
43
+ }
44
+ }
@@ -7,7 +7,6 @@ import { Note, NoteDao, NoteStatus } from '@aztec/stdlib/note';
7
7
  import { MerkleTreeId } from '@aztec/stdlib/trees';
8
8
  import type { BlockHeader, TxHash } from '@aztec/stdlib/tx';
9
9
 
10
- import type { AccessScopes } from '../access_scopes.js';
11
10
  import type { NoteStore } from '../storage/note_store/note_store.js';
12
11
 
13
12
  export class NoteService {
@@ -32,7 +31,7 @@ export class NoteService {
32
31
  owner: AztecAddress | undefined,
33
32
  storageSlot: Fr,
34
33
  status: NoteStatus,
35
- scopes: AccessScopes,
34
+ scopes: AztecAddress[],
36
35
  ) {
37
36
  const noteDaos = await this.noteStore.getNotes(
38
37
  {
@@ -71,7 +70,7 @@ export class NoteService {
71
70
  *
72
71
  * @param contractAddress - The contract whose notes should be checked and nullified.
73
72
  */
74
- public async syncNoteNullifiers(contractAddress: AztecAddress, scopes: AccessScopes): Promise<void> {
73
+ public async syncNoteNullifiers(contractAddress: AztecAddress, scopes: AztecAddress[]): Promise<void> {
75
74
  const anchorBlockHash = await this.anchorBlockHeader.hash();
76
75
 
77
76
  const contractNotes = await this.noteStore.getNotes({ contractAddress, scopes }, this.jobId);
@@ -121,7 +120,7 @@ export class NoteService {
121
120
  noteHash: Fr,
122
121
  nullifier: Fr,
123
122
  txHash: TxHash,
124
- recipient: AztecAddress,
123
+ scope: AztecAddress,
125
124
  ): Promise<void> {
126
125
  // We are going to store the new note in the NoteStore, which will let us later return it via `getNotes`.
127
126
  // There's two things we need to check before we do this however:
@@ -155,16 +154,28 @@ export class NoteService {
155
154
  this.aztecNode.findLeavesIndexes(anchorBlockHash, MerkleTreeId.NULLIFIER_TREE, [siloedNullifier]),
156
155
  ]);
157
156
  if (!txEffect) {
158
- throw new Error(`Could not find tx effect for tx hash ${txHash}`);
157
+ // We error out instead of just logging a warning and skipping the note because this would indicate a bug. This
158
+ // is because the node has already served info about this tx either when obtaining the log (TxScopedL2Log contain
159
+ // tx info) or when getting metadata for the offchain message (before the message got passed to `process_log`).
160
+ throw new Error(`Could not find tx effect for tx hash ${txHash} when processing a note.`);
159
161
  }
160
162
 
161
163
  if (txEffect.l2BlockNumber > anchorBlockNumber) {
162
- throw new Error(`Could not find tx effect for tx hash ${txHash} as of block number ${anchorBlockNumber}`);
164
+ // If the message was delivered onchain, this would indicate a bug: log sync should never load logs from blocks
165
+ // newer than the anchor block. If the note came via an offchain message, it would likely also be a bug, since we
166
+ // sync a new anchor block before calling `process_message`. For this not to be a bug, the message would need to
167
+ // come from a newer block than the anchor served by the node, implying the node isn't properly synced.
168
+ // We therefore error out here rather than assuming the offchain message was constructed by a malicious
169
+ // sender with the intention of bricking recipient's PXE (if we assumed that we would just ignore the message).
170
+ throw new Error(
171
+ `Obtained a newer tx effect for ${txHash} for a note validation request than the anchor block ${anchorBlockNumber}. This is a bug as we should not ever be processing a note from a newer block than the anchor block.`,
172
+ );
163
173
  }
164
174
 
165
175
  // Find the index of the note hash in the noteHashes array to determine note ordering within the tx
166
176
  const noteIndexInTx = txEffect.data.noteHashes.findIndex(nh => nh.equals(uniqueNoteHash));
167
177
  if (noteIndexInTx === -1) {
178
+ // Similar to the comment above - we error out as this would indicate a bug in nonce discovery.
168
179
  throw new Error(`Note hash ${noteHash} (uniqued as ${uniqueNoteHash}) is not present in tx ${txHash}`);
169
180
  }
170
181
 
@@ -184,8 +195,7 @@ export class NoteService {
184
195
  noteIndexInTx,
185
196
  );
186
197
 
187
- // The note was found by `recipient`, so we use that as the scope when storing the note.
188
- await this.noteStore.addNotes([noteDao], recipient, this.jobId);
198
+ await this.noteStore.addNotes([noteDao], scope, this.jobId);
189
199
 
190
200
  if (nullifierIndex !== undefined) {
191
201
  // We found nullifier index which implies that the note has already been nullified.
@@ -2,8 +2,6 @@ import type { Fr } from '@aztec/foundation/curves/bn254';
2
2
  import type { AztecAddress } from '@aztec/stdlib/aztec-address';
3
3
  import type { NoteStatus } from '@aztec/stdlib/note';
4
4
 
5
- import type { AccessScopes } from './access_scopes.js';
6
-
7
5
  /**
8
6
  * A filter used to fetch notes.
9
7
  * @remarks This filter is applied as an intersection of all its params.
@@ -22,5 +20,5 @@ export type NotesFilter = {
22
20
  status?: NoteStatus;
23
21
  /** The siloed nullifier for the note. */
24
22
  siloedNullifier?: Fr;
25
- scopes: AccessScopes;
23
+ scopes: AztecAddress[];
26
24
  };
@@ -1,12 +1,22 @@
1
- /// The ORACLE_VERSION constant is used to check that the oracle interface is in sync between PXE and Aztec.nr. We need
2
- /// to version the oracle interface to ensure that developers get a reasonable error message if they use incompatible
3
- /// versions of Aztec.nr and PXE. The Noir counterpart is in `noir-projects/aztec-nr/aztec/src/oracle/version.nr`.
1
+ /// The oracle version constants are used to check that the oracle interface is in sync between PXE and Aztec.nr.
2
+ /// We version the oracle interface as `major.minor` where:
3
+ /// - `major` = backward-breaking changes (must match exactly between PXE and Aztec.nr)
4
+ /// - `minor` = oracle additions (non-breaking; PXE minor >= contract minor)
4
5
  ///
5
- /// @dev Whenever a contract function or Noir test is run, the `utilityAssertCompatibleOracleVersion` oracle is called
6
- /// and if the oracle version is incompatible an error is thrown.
7
- export const ORACLE_VERSION = 12;
6
+ /// The Noir counterparts are in `noir-projects/aztec-nr/aztec/src/oracle/version.nr`.
7
+ ///
8
+ /// @dev Whenever a contract function or Noir test is run, the `aztec_utl_assertCompatibleOracleVersion` oracle is called.
9
+ /// If the major version is incompatible, an error is thrown immediately. The minor version is recorded by the PXE and
10
+ /// used to provide helpful error messages if a contract calls an oracle that doesn't exist. We don't throw immediately
11
+ /// if AZTEC_NR_MINOR > PXE_MINOR because if a contract is updated to use a newer Aztec.nr dependency without actually
12
+ /// using any of the new oracles then there is no reason to throw.
13
+ export const ORACLE_VERSION_MAJOR = 22;
14
+ export const ORACLE_VERSION_MINOR = 1;
8
15
 
9
- /// This hash is computed as by hashing the Oracle interface and it is used to detect when the Oracle interface changes,
10
- /// which in turn implies that you need to update the ORACLE_VERSION constant in this file and in
11
- /// `noir-projects/aztec-nr/aztec/src/oracle/version.nr`.
12
- export const ORACLE_INTERFACE_HASH = '666a8a7fc697f72b29dbf0ae7464db269cf5afa019acac8861f814543147dbb4';
16
+ /// This hash is computed from the Oracle interface and is used to detect when that interface changes. When it does,
17
+ /// you need to either:
18
+ /// - increment `ORACLE_VERSION_MAJOR` and reset `ORACLE_VERSION_MINOR` to zero if the change is breaking, or
19
+ /// - increment only `ORACLE_VERSION_MINOR` if the change is additive (a new oracle was added).
20
+ ///
21
+ /// These constants must be kept in sync between this file and `noir-projects/aztec-nr/aztec/src/oracle/version.nr`.
22
+ export const ORACLE_INTERFACE_HASH = 'efafa0db2cc1f94e26d794d0079c8f71115261df0c3d0fa8cb5b64f17a12db92';
package/src/pxe.ts CHANGED
@@ -52,7 +52,6 @@ import {
52
52
 
53
53
  import { inspect } from 'util';
54
54
 
55
- import type { AccessScopes } from './access_scopes.js';
56
55
  import { BlockSynchronizer } from './block_synchronizer/index.js';
57
56
  import type { PXEConfig } from './config/index.js';
58
57
  import { BenchmarkedNodeFactory } from './contract_function_simulator/benchmarked_node.js';
@@ -68,6 +67,7 @@ import { PXEDebugUtils } from './debug/pxe_debug_utils.js';
68
67
  import { enrichPublicSimulationError, enrichSimulationError } from './error_enriching.js';
69
68
  import { PrivateEventFilterValidator } from './events/private_event_filter_validator.js';
70
69
  import { JobCoordinator } from './job_coordinator/job_coordinator.js';
70
+ import { MessageContextService } from './messages/message_context_service.js';
71
71
  import {
72
72
  PrivateKernelExecutionProver,
73
73
  type PrivateKernelExecutionProverConfig,
@@ -95,7 +95,7 @@ export type ProfileTxOpts = {
95
95
  /** If true, proof generation is skipped during profiling. Defaults to true. */
96
96
  skipProofGeneration?: boolean;
97
97
  /** Addresses whose private state and keys are accessible during private execution. */
98
- scopes: AccessScopes;
98
+ scopes: AztecAddress[];
99
99
  };
100
100
 
101
101
  /** Options for PXE.simulateTx. */
@@ -106,10 +106,12 @@ export type SimulateTxOpts = {
106
106
  skipTxValidation?: boolean;
107
107
  /** If false, fees are enforced. */
108
108
  skipFeeEnforcement?: boolean;
109
- /** State overrides for the simulation, such as contract instances and artifacts. */
109
+ /** If true, kernel logic is emulated in TS for simulation */
110
+ skipKernels?: boolean;
111
+ /** State overrides for the simulation, such as contract instances and artifacts. Requires skipKernels: true */
110
112
  overrides?: SimulationOverrides;
111
113
  /** Addresses whose private state and keys are accessible during private execution */
112
- scopes: AccessScopes;
114
+ scopes: AztecAddress[];
113
115
  };
114
116
 
115
117
  /** Options for PXE.executeUtility. */
@@ -117,7 +119,7 @@ export type ExecuteUtilityOpts = {
117
119
  /** The authentication witnesses required for the function call. */
118
120
  authwits?: AuthWitness[];
119
121
  /** The accounts whose notes we can access in this call */
120
- scopes: AccessScopes;
122
+ scopes: AztecAddress[];
121
123
  };
122
124
 
123
125
  /** Args for PXE.create. */
@@ -145,6 +147,7 @@ export type PXECreateArgs = {
145
147
  export class PXE {
146
148
  private constructor(
147
149
  private node: AztecNode,
150
+ private db: AztecAsyncKVStore,
148
151
  private blockStateSynchronizer: BlockSynchronizer,
149
152
  private keyStore: KeyStore,
150
153
  private contractStore: ContractStore,
@@ -157,6 +160,7 @@ export class PXE {
157
160
  private addressStore: AddressStore,
158
161
  private privateEventStore: PrivateEventStore,
159
162
  private contractSyncService: ContractSyncService,
163
+ private messageContextService: MessageContextService,
160
164
  private simulator: CircuitSimulator,
161
165
  private proverEnabled: boolean,
162
166
  private proofCreator: PrivateKernelProver,
@@ -212,6 +216,8 @@ export class PXE {
212
216
  noteStore,
213
217
  createLogger('pxe:contract_sync', bindings),
214
218
  );
219
+ const messageContextService = new MessageContextService(node);
220
+
215
221
  const synchronizer = new BlockSynchronizer(
216
222
  node,
217
223
  store,
@@ -240,6 +246,7 @@ export class PXE {
240
246
 
241
247
  const pxe = new PXE(
242
248
  node,
249
+ store,
243
250
  synchronizer,
244
251
  keyStore,
245
252
  contractStore,
@@ -252,6 +259,7 @@ export class PXE {
252
259
  addressStore,
253
260
  privateEventStore,
254
261
  contractSyncService,
262
+ messageContextService,
255
263
  simulator,
256
264
  proverEnabled,
257
265
  proofCreator,
@@ -293,6 +301,7 @@ export class PXE {
293
301
  privateEventStore: this.privateEventStore,
294
302
  simulator: this.simulator,
295
303
  contractSyncService: this.contractSyncService,
304
+ messageContextService: this.messageContextService,
296
305
  });
297
306
  }
298
307
 
@@ -358,7 +367,7 @@ export class PXE {
358
367
  async #executePrivate(
359
368
  contractFunctionSimulator: ContractFunctionSimulator,
360
369
  txRequest: TxExecutionRequest,
361
- scopes: AccessScopes,
370
+ scopes: AztecAddress[],
362
371
  jobId: string,
363
372
  ): Promise<PrivateExecutionResult> {
364
373
  const { origin: contractAddress, functionSelector } = txRequest;
@@ -407,12 +416,19 @@ export class PXE {
407
416
  contractFunctionSimulator: ContractFunctionSimulator,
408
417
  call: FunctionCall,
409
418
  authWitnesses: AuthWitness[] | undefined,
410
- scopes: AccessScopes,
419
+ scopes: AztecAddress[],
411
420
  jobId: string,
412
421
  ) {
413
422
  try {
414
423
  const anchorBlockHeader = await this.anchorBlockStore.getBlockHeader();
415
- return contractFunctionSimulator.runUtility(call, authWitnesses ?? [], anchorBlockHeader, scopes, jobId);
424
+ const { result, offchainEffects } = await contractFunctionSimulator.runUtility(
425
+ call,
426
+ authWitnesses ?? [],
427
+ anchorBlockHeader,
428
+ scopes,
429
+ jobId,
430
+ );
431
+ return { result, offchainEffects };
416
432
  } catch (err) {
417
433
  if (err instanceof SimulationError) {
418
434
  await enrichSimulationError(err, this.contractStore, this.log);
@@ -485,7 +501,9 @@ export class PXE {
485
501
  * @returns The synced block header
486
502
  */
487
503
  public getSyncedBlockHeader(): Promise<BlockHeader> {
488
- return this.anchorBlockStore.getBlockHeader();
504
+ return this.#putInJobQueue(() => {
505
+ return this.anchorBlockStore.getBlockHeader();
506
+ });
489
507
  }
490
508
 
491
509
  /**
@@ -552,6 +570,9 @@ export class PXE {
552
570
 
553
571
  if (wasAdded) {
554
572
  this.log.info(`Added sender:\n ${sender.toString()}`);
573
+ // Wipe the entire sync cache: the new sender's tagged logs could contain notes/events for any contract, so
574
+ // all contracts must re-sync to discover them.
575
+ this.contractSyncService.wipe();
555
576
  } else {
556
577
  this.log.info(`Sender:\n "${sender.toString()}"\n already registered.`);
557
578
  }
@@ -764,17 +785,17 @@ export class PXE {
764
785
  // transaction before this one is included in a block from this PXE, and that transaction contains a log with
765
786
  // a tag derived from the same secret, we would reuse the tag and the transactions would be linked. Hence
766
787
  // storing the tags here prevents linkage of txs sent from the same PXE.
767
- const preTagsUsedInTheTx = privateExecutionResult.entrypoint.preTags;
768
- if (preTagsUsedInTheTx.length > 0) {
788
+ const taggingIndexRangesUsedInTheTx = privateExecutionResult.entrypoint.taggingIndexRanges;
789
+ if (taggingIndexRangesUsedInTheTx.length > 0) {
769
790
  // TODO(benesjan): The following is an expensive operation. Figure out a way to avoid it.
770
791
  const txHash = (await txProvingResult.toTx()).txHash;
771
792
 
772
- await this.senderTaggingStore.storePendingIndexes(preTagsUsedInTheTx, txHash, jobId);
773
- this.log.debug(`Stored used pre-tags as sender for the tx`, {
774
- preTagsUsedInTheTx,
793
+ await this.senderTaggingStore.storePendingIndexes(taggingIndexRangesUsedInTheTx, txHash, jobId);
794
+ this.log.debug(`Stored used tagging index ranges as sender for the tx`, {
795
+ taggingIndexRangesUsedInTheTx,
775
796
  });
776
797
  } else {
777
- this.log.debug(`No pre-tags used in the tx`);
798
+ this.log.debug(`No tagging index ranges used in the tx`);
778
799
  }
779
800
 
780
801
  return txProvingResult;
@@ -881,7 +902,14 @@ export class PXE {
881
902
  */
882
903
  public simulateTx(
883
904
  txRequest: TxExecutionRequest,
884
- { simulatePublic, skipTxValidation = false, skipFeeEnforcement = false, overrides, scopes }: SimulateTxOpts,
905
+ {
906
+ simulatePublic,
907
+ skipTxValidation = false,
908
+ skipFeeEnforcement = false,
909
+ skipKernels = true,
910
+ overrides,
911
+ scopes,
912
+ }: SimulateTxOpts,
885
913
  ): Promise<TxSimulationResult> {
886
914
  // We disable concurrent simulations since those might execute oracles which read and write to the PXE stores (e.g.
887
915
  // to the capsules), and we need to prevent concurrent runs from interfering with one another (e.g. attempting to
@@ -905,17 +933,20 @@ export class PXE {
905
933
  await this.blockStateSynchronizer.sync();
906
934
  const syncTime = syncTimer.ms();
907
935
 
908
- const contractFunctionSimulator = this.#getSimulatorForTx(overrides);
909
- // Temporary: in case there are overrides, we have to skip the kernels or validations
910
- // will fail. Consider handing control to the user/wallet on whether they want to run them
911
- // or not.
912
936
  const overriddenContracts = overrides?.contracts ? new Set(Object.keys(overrides.contracts)) : undefined;
913
937
  const hasOverriddenContracts = overriddenContracts !== undefined && overriddenContracts.size > 0;
914
- const skipKernels = hasOverriddenContracts;
915
938
 
916
- // Set overridden contracts on the sync service so it knows to skip syncing them
939
+ if (hasOverriddenContracts && !skipKernels) {
940
+ throw new Error(
941
+ 'Simulating with overridden contracts is not compatible with kernel execution. Please set skipKernels to true when simulating with overridden contracts.',
942
+ );
943
+ }
944
+ const contractFunctionSimulator = this.#getSimulatorForTx(overrides);
945
+
917
946
  if (hasOverriddenContracts) {
918
- this.contractSyncService.setOverriddenContracts(jobId, overriddenContracts);
947
+ // Overridden contracts don't have a sync function, so calling sync on them would fail.
948
+ // We exclude them so the sync service skips them entirely.
949
+ this.contractSyncService.setExcludedFromSync(jobId, overriddenContracts);
919
950
  }
920
951
 
921
952
  // Execution of private functions only; no proving, and no kernel logic.
@@ -1010,7 +1041,7 @@ export class PXE {
1010
1041
  inspect(txRequest),
1011
1042
  `simulatePublic=${simulatePublic}`,
1012
1043
  `skipTxValidation=${skipTxValidation}`,
1013
- `scopes=${scopes === 'ALL_SCOPES' ? scopes : scopes.map(s => s.toString()).join(', ')}`,
1044
+ `scopes=${scopes.map(s => s.toString()).join(', ')}`,
1014
1045
  );
1015
1046
  }
1016
1047
  });
@@ -1022,7 +1053,7 @@ export class PXE {
1022
1053
  */
1023
1054
  public executeUtility(
1024
1055
  call: FunctionCall,
1025
- { authwits, scopes }: ExecuteUtilityOpts = { scopes: 'ALL_SCOPES' },
1056
+ { authwits, scopes }: ExecuteUtilityOpts = { scopes: [] },
1026
1057
  ): Promise<UtilityExecutionResult> {
1027
1058
  // We disable concurrent executions since those might execute oracles which read and write to the PXE stores (e.g.
1028
1059
  // to the capsules), and we need to prevent concurrent runs from interfering with one another (e.g. attempting to
@@ -1047,7 +1078,7 @@ export class PXE {
1047
1078
  scopes,
1048
1079
  );
1049
1080
 
1050
- const executionResult = await this.#executeUtility(
1081
+ const { result: executionResult, offchainEffects } = await this.#executeUtility(
1051
1082
  contractFunctionSimulator,
1052
1083
  call,
1053
1084
  authwits ?? [],
@@ -1068,14 +1099,19 @@ export class PXE {
1068
1099
  };
1069
1100
 
1070
1101
  const simulationStats = contractFunctionSimulator.getStats();
1071
- return { result: executionResult, stats: { timings, nodeRPCCalls: simulationStats.nodeRPCCalls } };
1102
+ return {
1103
+ result: executionResult,
1104
+ offchainEffects,
1105
+ anchorBlockTimestamp: anchorBlockHeader.globalVariables.timestamp,
1106
+ stats: { timings, nodeRPCCalls: simulationStats.nodeRPCCalls },
1107
+ };
1072
1108
  } catch (err: any) {
1073
1109
  const { to, name, args } = call;
1074
1110
  const stringifiedArgs = args.map(arg => arg.toString()).join(', ');
1075
1111
  throw this.#contextualizeError(
1076
1112
  err,
1077
1113
  `executeUtility ${to}:${name}(${stringifiedArgs})`,
1078
- `scopes=${scopes === 'ALL_SCOPES' ? scopes : scopes.map(s => s.toString()).join(', ')}`,
1114
+ `scopes=${scopes.map(s => s.toString()).join(', ')}`,
1079
1115
  );
1080
1116
  }
1081
1117
  });
@@ -1130,9 +1166,10 @@ export class PXE {
1130
1166
  }
1131
1167
 
1132
1168
  /**
1133
- * Stops the PXE's job queue.
1169
+ * Stops the PXE's job queue and closes the backing store.
1134
1170
  */
1135
- public stop(): Promise<void> {
1136
- return this.jobQueue.end();
1171
+ public async stop(): Promise<void> {
1172
+ await this.jobQueue.end();
1173
+ await this.db.close();
1137
1174
  }
1138
1175
  }