@aztec/pxe 0.0.1-commit.001888fc → 0.0.1-commit.017a351

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 (343) hide show
  1. package/dest/bin/check_oracle_version.js +43 -99
  2. package/dest/bin/index.d.ts +2 -0
  3. package/dest/bin/index.d.ts.map +1 -0
  4. package/dest/bin/index.js +1 -0
  5. package/dest/bin/oracle_version_helpers.d.ts +26 -0
  6. package/dest/bin/oracle_version_helpers.d.ts.map +1 -0
  7. package/dest/bin/oracle_version_helpers.js +93 -0
  8. package/dest/block_synchronizer/block_stream_source.d.ts +10 -0
  9. package/dest/block_synchronizer/block_stream_source.d.ts.map +1 -0
  10. package/dest/block_synchronizer/block_stream_source.js +62 -0
  11. package/dest/block_synchronizer/block_synchronizer.d.ts +6 -2
  12. package/dest/block_synchronizer/block_synchronizer.d.ts.map +1 -1
  13. package/dest/block_synchronizer/block_synchronizer.js +30 -10
  14. package/dest/config/index.d.ts +7 -1
  15. package/dest/config/index.d.ts.map +1 -1
  16. package/dest/config/index.js +12 -14
  17. package/dest/config/package_info.js +1 -1
  18. package/dest/contract_function_simulator/contract_function_simulator.d.ts +9 -4
  19. package/dest/contract_function_simulator/contract_function_simulator.d.ts.map +1 -1
  20. package/dest/contract_function_simulator/contract_function_simulator.js +35 -14
  21. package/dest/contract_function_simulator/ephemeral_array_service.d.ts +28 -0
  22. package/dest/contract_function_simulator/ephemeral_array_service.d.ts.map +1 -0
  23. package/dest/contract_function_simulator/ephemeral_array_service.js +78 -0
  24. package/dest/contract_function_simulator/execution_note_cache.d.ts +2 -2
  25. package/dest/contract_function_simulator/execution_note_cache.d.ts.map +1 -1
  26. package/dest/contract_function_simulator/execution_tagging_index_cache.d.ts +5 -9
  27. package/dest/contract_function_simulator/execution_tagging_index_cache.d.ts.map +1 -1
  28. package/dest/contract_function_simulator/execution_tagging_index_cache.js +3 -7
  29. package/dest/contract_function_simulator/index.d.ts +13 -2
  30. package/dest/contract_function_simulator/index.d.ts.map +1 -1
  31. package/dest/contract_function_simulator/index.js +10 -0
  32. package/dest/contract_function_simulator/noir-structs/bounded_vec.d.ts +48 -0
  33. package/dest/contract_function_simulator/noir-structs/bounded_vec.d.ts.map +1 -0
  34. package/dest/contract_function_simulator/noir-structs/bounded_vec.js +45 -0
  35. package/dest/contract_function_simulator/noir-structs/ephemeral_array.d.ts +37 -0
  36. package/dest/contract_function_simulator/noir-structs/ephemeral_array.d.ts.map +1 -0
  37. package/dest/contract_function_simulator/noir-structs/ephemeral_array.js +59 -0
  38. package/dest/contract_function_simulator/noir-structs/event_validation_request.d.ts +5 -5
  39. package/dest/contract_function_simulator/noir-structs/event_validation_request.d.ts.map +1 -1
  40. package/dest/contract_function_simulator/noir-structs/event_validation_request.js +6 -8
  41. package/dest/contract_function_simulator/noir-structs/log_retrieval_request.d.ts +13 -3
  42. package/dest/contract_function_simulator/noir-structs/log_retrieval_request.d.ts.map +1 -1
  43. package/dest/contract_function_simulator/noir-structs/log_retrieval_request.js +35 -4
  44. package/dest/contract_function_simulator/noir-structs/log_retrieval_response.d.ts +1 -1
  45. package/dest/contract_function_simulator/noir-structs/log_retrieval_response.js +1 -1
  46. package/dest/contract_function_simulator/noir-structs/note_data.d.ts +27 -0
  47. package/dest/contract_function_simulator/noir-structs/note_data.d.ts.map +1 -0
  48. package/dest/contract_function_simulator/noir-structs/note_data.js +3 -0
  49. package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts +5 -5
  50. package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts.map +1 -1
  51. package/dest/contract_function_simulator/noir-structs/note_validation_request.js +6 -8
  52. package/dest/contract_function_simulator/noir-structs/option.d.ts +61 -0
  53. package/dest/contract_function_simulator/noir-structs/option.d.ts.map +1 -0
  54. package/dest/contract_function_simulator/noir-structs/option.js +62 -0
  55. package/dest/contract_function_simulator/noir-structs/provided_secret.d.ts +11 -0
  56. package/dest/contract_function_simulator/noir-structs/provided_secret.d.ts.map +1 -0
  57. package/dest/contract_function_simulator/noir-structs/provided_secret.js +24 -0
  58. package/dest/contract_function_simulator/oracle/interfaces.d.ts +16 -102
  59. package/dest/contract_function_simulator/oracle/interfaces.d.ts.map +1 -1
  60. package/dest/contract_function_simulator/oracle/interfaces.js +2 -2
  61. package/dest/contract_function_simulator/oracle/note_packing_utils.d.ts +2 -2
  62. package/dest/contract_function_simulator/oracle/note_packing_utils.d.ts.map +1 -1
  63. package/dest/contract_function_simulator/oracle/note_packing_utils.js +2 -2
  64. package/dest/contract_function_simulator/oracle/oracle.d.ts +74 -45
  65. package/dest/contract_function_simulator/oracle/oracle.d.ts.map +1 -1
  66. package/dest/contract_function_simulator/oracle/oracle.js +405 -270
  67. package/dest/contract_function_simulator/oracle/oracle_registry.d.ts +127 -0
  68. package/dest/contract_function_simulator/oracle/oracle_registry.d.ts.map +1 -0
  69. package/dest/contract_function_simulator/oracle/oracle_registry.js +786 -0
  70. package/dest/contract_function_simulator/oracle/oracle_type_mappings.d.ts +139 -0
  71. package/dest/contract_function_simulator/oracle/oracle_type_mappings.d.ts.map +1 -0
  72. package/dest/contract_function_simulator/oracle/oracle_type_mappings.js +560 -0
  73. package/dest/contract_function_simulator/oracle/private_execution.js +1 -1
  74. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts +33 -39
  75. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts.map +1 -1
  76. package/dest/contract_function_simulator/oracle/private_execution_oracle.js +70 -54
  77. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts +104 -60
  78. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts.map +1 -1
  79. package/dest/contract_function_simulator/oracle/utility_execution_oracle.js +374 -160
  80. package/dest/contract_function_simulator/pick_notes.d.ts +1 -1
  81. package/dest/contract_function_simulator/pick_notes.d.ts.map +1 -1
  82. package/dest/contract_function_simulator/pick_notes.js +20 -3
  83. package/dest/contract_function_simulator/proxied_contract_data_source.d.ts +1 -1
  84. package/dest/contract_function_simulator/proxied_contract_data_source.d.ts.map +1 -1
  85. package/dest/contract_function_simulator/proxied_contract_data_source.js +35 -64
  86. package/dest/contract_logging.d.ts +9 -4
  87. package/dest/contract_logging.d.ts.map +1 -1
  88. package/dest/contract_logging.js +21 -6
  89. package/dest/contract_sync/contract_sync_service.d.ts +7 -8
  90. package/dest/contract_sync/contract_sync_service.d.ts.map +1 -1
  91. package/dest/contract_sync/contract_sync_service.js +87 -49
  92. package/dest/contract_sync/helpers.d.ts +2 -4
  93. package/dest/contract_sync/helpers.d.ts.map +1 -1
  94. package/dest/contract_sync/helpers.js +12 -14
  95. package/dest/debug/pxe_debug_utils.d.ts +3 -8
  96. package/dest/debug/pxe_debug_utils.d.ts.map +1 -1
  97. package/dest/debug/pxe_debug_utils.js +0 -6
  98. package/dest/entrypoints/client/bundle/index.d.ts +1 -2
  99. package/dest/entrypoints/client/bundle/index.d.ts.map +1 -1
  100. package/dest/entrypoints/client/bundle/index.js +0 -1
  101. package/dest/entrypoints/client/bundle/utils.d.ts +2 -2
  102. package/dest/entrypoints/client/bundle/utils.d.ts.map +1 -1
  103. package/dest/entrypoints/client/bundle/utils.js +13 -5
  104. package/dest/entrypoints/client/lazy/index.d.ts +1 -2
  105. package/dest/entrypoints/client/lazy/index.d.ts.map +1 -1
  106. package/dest/entrypoints/client/lazy/index.js +0 -1
  107. package/dest/entrypoints/client/lazy/utils.d.ts +2 -2
  108. package/dest/entrypoints/client/lazy/utils.d.ts.map +1 -1
  109. package/dest/entrypoints/client/lazy/utils.js +13 -5
  110. package/dest/entrypoints/pxe_creation_options.d.ts +9 -1
  111. package/dest/entrypoints/pxe_creation_options.d.ts.map +1 -1
  112. package/dest/entrypoints/pxe_creation_options.js +3 -1
  113. package/dest/entrypoints/server/index.d.ts +3 -3
  114. package/dest/entrypoints/server/index.d.ts.map +1 -1
  115. package/dest/entrypoints/server/index.js +2 -2
  116. package/dest/entrypoints/server/utils.d.ts +4 -3
  117. package/dest/entrypoints/server/utils.d.ts.map +1 -1
  118. package/dest/entrypoints/server/utils.js +13 -5
  119. package/dest/events/event_service.d.ts +15 -6
  120. package/dest/events/event_service.d.ts.map +1 -1
  121. package/dest/events/event_service.js +44 -11
  122. package/dest/events/private_event_filter_validator.d.ts +3 -2
  123. package/dest/events/private_event_filter_validator.d.ts.map +1 -1
  124. package/dest/events/private_event_filter_validator.js +15 -0
  125. package/dest/hooks/authorize_utility_call.d.ts +41 -0
  126. package/dest/hooks/authorize_utility_call.d.ts.map +1 -0
  127. package/dest/hooks/authorize_utility_call.js +4 -0
  128. package/dest/hooks/execution_hooks.d.ts +42 -0
  129. package/dest/hooks/execution_hooks.d.ts.map +1 -0
  130. package/dest/hooks/execution_hooks.js +9 -0
  131. package/dest/hooks/index.d.ts +4 -0
  132. package/dest/hooks/index.d.ts.map +1 -0
  133. package/dest/hooks/index.js +1 -0
  134. package/dest/logs/log_service.d.ts +9 -9
  135. package/dest/logs/log_service.d.ts.map +1 -1
  136. package/dest/logs/log_service.js +126 -72
  137. package/dest/messages/message_context_service.d.ts +3 -3
  138. package/dest/messages/message_context_service.d.ts.map +1 -1
  139. package/dest/messages/message_context_service.js +30 -11
  140. package/dest/notes/note_service.d.ts +27 -6
  141. package/dest/notes/note_service.d.ts.map +1 -1
  142. package/dest/notes/note_service.js +80 -56
  143. package/dest/notes_filter.d.ts +2 -3
  144. package/dest/notes_filter.d.ts.map +1 -1
  145. package/dest/oracle_version.d.ts +4 -3
  146. package/dest/oracle_version.d.ts.map +1 -1
  147. package/dest/oracle_version.js +20 -10
  148. package/dest/private_kernel/batch_planner.d.ts +47 -0
  149. package/dest/private_kernel/batch_planner.d.ts.map +1 -0
  150. package/dest/private_kernel/batch_planner.js +104 -0
  151. package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.js +1 -1
  152. package/dest/private_kernel/hints/test_utils.d.ts +1 -1
  153. package/dest/private_kernel/hints/test_utils.d.ts.map +1 -1
  154. package/dest/private_kernel/hints/test_utils.js +2 -3
  155. package/dest/private_kernel/private_kernel_execution_prover.d.ts +6 -2
  156. package/dest/private_kernel/private_kernel_execution_prover.d.ts.map +1 -1
  157. package/dest/private_kernel/private_kernel_execution_prover.js +152 -59
  158. package/dest/private_kernel/private_kernel_oracle.d.ts +10 -10
  159. package/dest/private_kernel/private_kernel_oracle.d.ts.map +1 -1
  160. package/dest/private_kernel/private_kernel_oracle.js +24 -22
  161. package/dest/pxe.d.ts +56 -10
  162. package/dest/pxe.d.ts.map +1 -1
  163. package/dest/pxe.js +139 -88
  164. package/dest/storage/anchor_block_store/anchor_block_store.js +1 -1
  165. package/dest/storage/backwards_compatibility_tests/kv_store_snapshot.d.ts +42 -0
  166. package/dest/storage/backwards_compatibility_tests/kv_store_snapshot.d.ts.map +1 -0
  167. package/dest/storage/backwards_compatibility_tests/kv_store_snapshot.js +93 -0
  168. package/dest/storage/backwards_compatibility_tests/schema_tests.d.ts +15 -0
  169. package/dest/storage/backwards_compatibility_tests/schema_tests.d.ts.map +1 -0
  170. package/dest/storage/backwards_compatibility_tests/schema_tests.js +591 -0
  171. package/dest/storage/backwards_compatibility_tests/store_spy.d.ts +19 -0
  172. package/dest/storage/backwards_compatibility_tests/store_spy.d.ts.map +1 -0
  173. package/dest/storage/backwards_compatibility_tests/store_spy.js +63 -0
  174. package/dest/storage/capsule_store/capsule_service.d.ts +21 -0
  175. package/dest/storage/capsule_store/capsule_service.d.ts.map +1 -0
  176. package/dest/storage/capsule_store/capsule_service.js +50 -0
  177. package/dest/storage/capsule_store/capsule_store.d.ts +9 -9
  178. package/dest/storage/capsule_store/capsule_store.d.ts.map +1 -1
  179. package/dest/storage/capsule_store/capsule_store.js +36 -28
  180. package/dest/storage/capsule_store/index.d.ts +2 -1
  181. package/dest/storage/capsule_store/index.d.ts.map +1 -1
  182. package/dest/storage/capsule_store/index.js +1 -0
  183. package/dest/storage/contract_store/contract_store.d.ts +1 -1
  184. package/dest/storage/contract_store/contract_store.d.ts.map +1 -1
  185. package/dest/storage/contract_store/contract_store.js +9 -26
  186. package/dest/storage/metadata.d.ts +1 -1
  187. package/dest/storage/metadata.js +1 -1
  188. package/dest/storage/note_store/note_store.d.ts +1 -1
  189. package/dest/storage/note_store/note_store.d.ts.map +1 -1
  190. package/dest/storage/note_store/note_store.js +2 -2
  191. package/dest/storage/open_pxe_stores.d.ts +33 -0
  192. package/dest/storage/open_pxe_stores.d.ts.map +1 -0
  193. package/dest/storage/open_pxe_stores.js +27 -0
  194. package/dest/storage/private_event_store/private_event_store.d.ts +1 -1
  195. package/dest/storage/private_event_store/private_event_store.d.ts.map +1 -1
  196. package/dest/storage/private_event_store/private_event_store.js +3 -0
  197. package/dest/storage/private_event_store/stored_private_event.js +1 -1
  198. package/dest/storage/tagging_store/recipient_tagging_store.d.ts +6 -6
  199. package/dest/storage/tagging_store/recipient_tagging_store.d.ts.map +1 -1
  200. package/dest/storage/tagging_store/sender_tagging_store.d.ts +5 -5
  201. package/dest/storage/tagging_store/sender_tagging_store.d.ts.map +1 -1
  202. package/dest/storage/tagging_store/sender_tagging_store.js +3 -3
  203. package/dest/tagging/get_all_logs_by_tags.d.ts +34 -10
  204. package/dest/tagging/get_all_logs_by_tags.d.ts.map +1 -1
  205. package/dest/tagging/get_all_logs_by_tags.js +36 -37
  206. package/dest/tagging/index.d.ts +5 -4
  207. package/dest/tagging/index.d.ts.map +1 -1
  208. package/dest/tagging/index.js +4 -3
  209. package/dest/tagging/persist_sender_tagging_index_ranges.d.ts +29 -0
  210. package/dest/tagging/persist_sender_tagging_index_ranges.d.ts.map +1 -0
  211. package/dest/tagging/persist_sender_tagging_index_ranges.js +42 -0
  212. package/dest/tagging/recipient_sync/sync_tagged_private_logs.d.ts +56 -0
  213. package/dest/tagging/recipient_sync/sync_tagged_private_logs.d.ts.map +1 -0
  214. package/dest/tagging/recipient_sync/sync_tagged_private_logs.js +163 -0
  215. package/dest/tagging/recipient_sync/utils/find_highest_indexes.d.ts +3 -3
  216. package/dest/tagging/recipient_sync/utils/find_highest_indexes.d.ts.map +1 -1
  217. package/dest/tagging/reconcile_tagging_index_ranges.d.ts +36 -0
  218. package/dest/tagging/reconcile_tagging_index_ranges.d.ts.map +1 -0
  219. package/dest/tagging/reconcile_tagging_index_ranges.js +74 -0
  220. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts +4 -5
  221. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts.map +1 -1
  222. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.js +26 -14
  223. package/dest/tagging/sender_sync/utils/get_status_change_of_pending.d.ts +11 -6
  224. package/dest/tagging/sender_sync/utils/get_status_change_of_pending.d.ts.map +1 -1
  225. package/dest/tagging/sender_sync/utils/get_status_change_of_pending.js +21 -0
  226. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts +4 -4
  227. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts.map +1 -1
  228. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.js +2 -2
  229. package/package.json +20 -17
  230. package/src/bin/check_oracle_version.ts +49 -122
  231. package/src/bin/index.ts +1 -0
  232. package/src/bin/oracle_version_helpers.ts +121 -0
  233. package/src/block_synchronizer/block_stream_source.ts +81 -0
  234. package/src/block_synchronizer/block_synchronizer.ts +33 -11
  235. package/src/config/index.ts +14 -8
  236. package/src/config/package_info.ts +1 -1
  237. package/src/contract_function_simulator/contract_function_simulator.ts +55 -17
  238. package/src/contract_function_simulator/ephemeral_array_service.ts +110 -0
  239. package/src/contract_function_simulator/execution_note_cache.ts +1 -1
  240. package/src/contract_function_simulator/execution_tagging_index_cache.ts +5 -9
  241. package/src/contract_function_simulator/index.ts +50 -1
  242. package/src/contract_function_simulator/noir-structs/bounded_vec.ts +55 -0
  243. package/src/contract_function_simulator/noir-structs/ephemeral_array.ts +66 -0
  244. package/src/contract_function_simulator/noir-structs/event_validation_request.ts +4 -6
  245. package/src/contract_function_simulator/noir-structs/log_retrieval_request.ts +36 -3
  246. package/src/contract_function_simulator/noir-structs/log_retrieval_response.ts +1 -1
  247. package/src/contract_function_simulator/noir-structs/note_data.ts +27 -0
  248. package/src/contract_function_simulator/noir-structs/note_validation_request.ts +4 -6
  249. package/src/contract_function_simulator/noir-structs/option.ts +69 -0
  250. package/src/contract_function_simulator/noir-structs/provided_secret.ts +27 -0
  251. package/src/contract_function_simulator/oracle/interfaces.ts +12 -175
  252. package/src/contract_function_simulator/oracle/note_packing_utils.ts +3 -3
  253. package/src/contract_function_simulator/oracle/oracle.ts +498 -442
  254. package/src/contract_function_simulator/oracle/oracle_registry.ts +585 -0
  255. package/src/contract_function_simulator/oracle/oracle_type_mappings.ts +553 -0
  256. package/src/contract_function_simulator/oracle/private_execution.ts +1 -1
  257. package/src/contract_function_simulator/oracle/private_execution_oracle.ts +93 -75
  258. package/src/contract_function_simulator/oracle/utility_execution_oracle.ts +547 -265
  259. package/src/contract_function_simulator/pick_notes.ts +22 -3
  260. package/src/contract_function_simulator/proxied_contract_data_source.ts +41 -64
  261. package/src/contract_logging.ts +18 -5
  262. package/src/contract_sync/contract_sync_service.ts +120 -83
  263. package/src/contract_sync/helpers.ts +13 -25
  264. package/src/debug/pxe_debug_utils.ts +3 -11
  265. package/src/entrypoints/client/bundle/index.ts +0 -1
  266. package/src/entrypoints/client/bundle/utils.ts +10 -5
  267. package/src/entrypoints/client/lazy/index.ts +0 -1
  268. package/src/entrypoints/client/lazy/utils.ts +10 -5
  269. package/src/entrypoints/pxe_creation_options.ts +14 -0
  270. package/src/entrypoints/server/index.ts +2 -2
  271. package/src/entrypoints/server/utils.ts +15 -6
  272. package/src/events/event_service.ts +69 -21
  273. package/src/events/private_event_filter_validator.ts +21 -1
  274. package/src/hooks/authorize_utility_call.ts +44 -0
  275. package/src/hooks/execution_hooks.ts +48 -0
  276. package/src/hooks/index.ts +7 -0
  277. package/src/logs/log_service.ts +158 -134
  278. package/src/messages/message_context_service.ts +43 -26
  279. package/src/notes/note_service.ts +119 -85
  280. package/src/notes_filter.ts +1 -3
  281. package/src/oracle_version.ts +20 -10
  282. package/src/private_kernel/batch_planner.ts +169 -0
  283. package/src/private_kernel/hints/private_kernel_reset_private_inputs_builder.ts +1 -1
  284. package/src/private_kernel/hints/test_utils.ts +2 -9
  285. package/src/private_kernel/private_kernel_execution_prover.ts +240 -82
  286. package/src/private_kernel/private_kernel_oracle.ts +35 -25
  287. package/src/pxe.ts +227 -92
  288. package/src/storage/anchor_block_store/anchor_block_store.ts +1 -1
  289. package/src/storage/backwards_compatibility_tests/__snapshots__/AddressStore.json +22 -0
  290. package/src/storage/backwards_compatibility_tests/__snapshots__/AnchorBlockStore.json +3 -0
  291. package/src/storage/backwards_compatibility_tests/__snapshots__/CapsuleStore.json +16 -0
  292. package/src/storage/backwards_compatibility_tests/__snapshots__/ContractStore.json +28 -0
  293. package/src/storage/backwards_compatibility_tests/__snapshots__/KeyStore.json +52 -0
  294. package/src/storage/backwards_compatibility_tests/__snapshots__/L2TipsKVStore.json +46 -0
  295. package/src/storage/backwards_compatibility_tests/__snapshots__/NoteStore.json +36 -0
  296. package/src/storage/backwards_compatibility_tests/__snapshots__/PrivateEventStore.json +44 -0
  297. package/src/storage/backwards_compatibility_tests/__snapshots__/RecipientTaggingStore.json +18 -0
  298. package/src/storage/backwards_compatibility_tests/__snapshots__/SenderAddressBookStore.json +16 -0
  299. package/src/storage/backwards_compatibility_tests/__snapshots__/SenderTaggingStore.json +22 -0
  300. package/src/storage/backwards_compatibility_tests/__snapshots__/opened_stores.json +97 -0
  301. package/src/storage/backwards_compatibility_tests/kv_store_snapshot.ts +122 -0
  302. package/src/storage/backwards_compatibility_tests/schema_tests.ts +712 -0
  303. package/src/storage/backwards_compatibility_tests/store_spy.ts +73 -0
  304. package/src/storage/capsule_store/capsule_service.ts +90 -0
  305. package/src/storage/capsule_store/capsule_store.ts +44 -26
  306. package/src/storage/capsule_store/index.ts +1 -0
  307. package/src/storage/contract_store/contract_store.ts +14 -35
  308. package/src/storage/metadata.ts +1 -1
  309. package/src/storage/note_store/note_store.ts +2 -5
  310. package/src/storage/open_pxe_stores.ts +49 -0
  311. package/src/storage/private_event_store/private_event_store.ts +4 -0
  312. package/src/storage/private_event_store/stored_private_event.ts +1 -1
  313. package/src/storage/tagging_store/recipient_tagging_store.ts +5 -9
  314. package/src/storage/tagging_store/sender_tagging_store.ts +6 -6
  315. package/src/tagging/get_all_logs_by_tags.ts +78 -50
  316. package/src/tagging/index.ts +4 -3
  317. package/src/tagging/persist_sender_tagging_index_ranges.ts +57 -0
  318. package/src/tagging/recipient_sync/sync_tagged_private_logs.ts +240 -0
  319. package/src/tagging/recipient_sync/utils/find_highest_indexes.ts +2 -2
  320. package/src/tagging/reconcile_tagging_index_ranges.ts +102 -0
  321. package/src/tagging/sender_sync/sync_sender_tagging_indexes.ts +41 -19
  322. package/src/tagging/sender_sync/utils/get_status_change_of_pending.ts +23 -8
  323. package/src/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.ts +4 -5
  324. package/dest/access_scopes.d.ts +0 -9
  325. package/dest/access_scopes.d.ts.map +0 -1
  326. package/dest/access_scopes.js +0 -6
  327. package/dest/contract_function_simulator/noir-structs/message_tx_context.d.ts +0 -16
  328. package/dest/contract_function_simulator/noir-structs/message_tx_context.d.ts.map +0 -1
  329. package/dest/contract_function_simulator/noir-structs/message_tx_context.js +0 -57
  330. package/dest/contract_function_simulator/oracle/legacy_oracle_mappings.d.ts +0 -9
  331. package/dest/contract_function_simulator/oracle/legacy_oracle_mappings.d.ts.map +0 -1
  332. package/dest/contract_function_simulator/oracle/legacy_oracle_mappings.js +0 -43
  333. package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.d.ts +0 -14
  334. package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.d.ts.map +0 -1
  335. package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.js +0 -99
  336. package/dest/tagging/recipient_sync/utils/load_logs_for_range.d.ts +0 -14
  337. package/dest/tagging/recipient_sync/utils/load_logs_for_range.d.ts.map +0 -1
  338. package/dest/tagging/recipient_sync/utils/load_logs_for_range.js +0 -33
  339. package/src/access_scopes.ts +0 -9
  340. package/src/contract_function_simulator/noir-structs/message_tx_context.ts +0 -55
  341. package/src/contract_function_simulator/oracle/legacy_oracle_mappings.ts +0 -142
  342. package/src/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.ts +0 -140
  343. package/src/tagging/recipient_sync/utils/load_logs_for_range.ts +0 -44
@@ -1,49 +1,78 @@
1
- import type { ARCHIVE_HEIGHT, NOTE_HASH_TREE_HEIGHT } from '@aztec/constants';
1
+ import { ARCHIVE_HEIGHT, type NOTE_HASH_TREE_HEIGHT } from '@aztec/constants';
2
2
  import type { BlockNumber } from '@aztec/foundation/branded-types';
3
+ import { uniqueBy } from '@aztec/foundation/collection';
3
4
  import { Aes128 } from '@aztec/foundation/crypto/aes128';
4
5
  import { Fr } from '@aztec/foundation/curves/bn254';
5
- import { Point } from '@aztec/foundation/curves/grumpkin';
6
+ import type { Point } from '@aztec/foundation/curves/grumpkin';
6
7
  import { LogLevels, type Logger, createLogger } from '@aztec/foundation/log';
7
- import type { MembershipWitness } from '@aztec/foundation/trees';
8
+ import { MembershipWitness } from '@aztec/foundation/trees';
8
9
  import type { KeyStore } from '@aztec/key-store';
9
- import { isProtocolContract } from '@aztec/protocol-contracts';
10
+ import {
11
+ type CircuitSimulator,
12
+ ExecutionError,
13
+ extractCallStack,
14
+ resolveAssertionMessageFromError,
15
+ toACVMWitness,
16
+ witnessMapToFields,
17
+ } from '@aztec/simulator/client';
18
+ import { type FunctionCall, FunctionSelector } from '@aztec/stdlib/abi';
10
19
  import type { AuthWitness } from '@aztec/stdlib/auth-witness';
11
20
  import { AztecAddress } from '@aztec/stdlib/aztec-address';
12
- import { BlockHash } from '@aztec/stdlib/block';
21
+ import { BlockHash, type L2TipsProvider } from '@aztec/stdlib/block';
13
22
  import type { CompleteAddress, ContractInstance, PartialAddress } from '@aztec/stdlib/contract';
14
23
  import { siloNullifier } from '@aztec/stdlib/hash';
15
24
  import type { AztecNode } from '@aztec/stdlib/interfaces/server';
16
25
  import type { KeyValidationRequest } from '@aztec/stdlib/kernel';
17
- import { type PublicKeys, computeAddressSecret } from '@aztec/stdlib/keys';
18
- import { deriveEcdhSharedSecret } from '@aztec/stdlib/logs';
26
+ import { PublicKeys, computeAddressSecret, hashPublicKey } from '@aztec/stdlib/keys';
27
+ import {
28
+ AppTaggingSecret,
29
+ MessageContext,
30
+ type PendingTaggedLog,
31
+ deriveAppSiloedSharedSecret,
32
+ } from '@aztec/stdlib/logs';
19
33
  import { getNonNullifiedL1ToL2MessageWitness } from '@aztec/stdlib/messaging';
20
34
  import type { NoteStatus } from '@aztec/stdlib/note';
21
35
  import { MerkleTreeId, type NullifierMembershipWitness, PublicDataWitness } from '@aztec/stdlib/trees';
22
- import type { BlockHeader, Capsule, OffchainEffect } from '@aztec/stdlib/tx';
23
-
24
- import type { AccessScopes } from '../../access_scopes.js';
25
- import { createContractLogger, logContractMessage } from '../../contract_logging.js';
36
+ import {
37
+ type BlockHeader,
38
+ type Capsule,
39
+ type IndexedTxEffect,
40
+ type OffchainEffect,
41
+ TxEffect,
42
+ type TxHash,
43
+ } from '@aztec/stdlib/tx';
44
+
45
+ import { createContractLogger, logContractMessage, stripAztecnrLogPrefix } from '../../contract_logging.js';
46
+ import type { ContractSyncService } from '../../contract_sync/contract_sync_service.js';
26
47
  import { EventService } from '../../events/event_service.js';
48
+ import type { UtilityCallAuthorizationRequest } from '../../hooks/authorize_utility_call.js';
49
+ import type { ExecutionHooks } from '../../hooks/index.js';
27
50
  import { LogService } from '../../logs/log_service.js';
28
51
  import { MessageContextService } from '../../messages/message_context_service.js';
29
52
  import { NoteService } from '../../notes/note_service.js';
30
- import { ORACLE_VERSION } from '../../oracle_version.js';
53
+ import { ORACLE_VERSION_MAJOR } from '../../oracle_version.js';
31
54
  import type { AddressStore } from '../../storage/address_store/address_store.js';
32
- import type { CapsuleStore } from '../../storage/capsule_store/capsule_store.js';
55
+ import type { CapsuleService } from '../../storage/capsule_store/capsule_service.js';
33
56
  import type { ContractStore } from '../../storage/contract_store/contract_store.js';
34
57
  import type { NoteStore } from '../../storage/note_store/note_store.js';
35
58
  import type { PrivateEventStore } from '../../storage/private_event_store/private_event_store.js';
36
59
  import type { RecipientTaggingStore } from '../../storage/tagging_store/recipient_tagging_store.js';
37
60
  import type { SenderAddressBookStore } from '../../storage/tagging_store/sender_address_book_store.js';
38
- import { EventValidationRequest } from '../noir-structs/event_validation_request.js';
39
- import { LogRetrievalRequest } from '../noir-structs/log_retrieval_request.js';
40
- import { LogRetrievalResponse } from '../noir-structs/log_retrieval_response.js';
41
- import { MessageTxContext } from '../noir-structs/message_tx_context.js';
42
- import { NoteValidationRequest } from '../noir-structs/note_validation_request.js';
61
+ import { EphemeralArrayService } from '../ephemeral_array_service.js';
62
+ import { BoundedVec } from '../noir-structs/bounded_vec.js';
63
+ import { EphemeralArray } from '../noir-structs/ephemeral_array.js';
64
+ import type { EventValidationRequest } from '../noir-structs/event_validation_request.js';
65
+ import type { LogRetrievalRequest } from '../noir-structs/log_retrieval_request.js';
66
+ import type { LogRetrievalResponse } from '../noir-structs/log_retrieval_response.js';
67
+ import type { NoteData } from '../noir-structs/note_data.js';
68
+ import type { NoteValidationRequest } from '../noir-structs/note_validation_request.js';
69
+ import { Option } from '../noir-structs/option.js';
70
+ import type { ProvidedSecret } from '../noir-structs/provided_secret.js';
43
71
  import { UtilityContext } from '../noir-structs/utility_context.js';
44
72
  import { pickNotes } from '../pick_notes.js';
45
- import type { IMiscOracle, IUtilityExecutionOracle, NoteData } from './interfaces.js';
73
+ import type { IMiscOracle, IUtilityExecutionOracle } from './interfaces.js';
46
74
  import { MessageLoadOracleInputs } from './message_load_oracle_inputs.js';
75
+ import { Oracle } from './oracle.js';
47
76
 
48
77
  /** Args for UtilityExecutionOracle constructor. */
49
78
  export type UtilityExecutionOracleArgs = {
@@ -59,12 +88,18 @@ export type UtilityExecutionOracleArgs = {
59
88
  aztecNode: AztecNode;
60
89
  recipientTaggingStore: RecipientTaggingStore;
61
90
  senderAddressBookStore: SenderAddressBookStore;
62
- capsuleStore: CapsuleStore;
91
+ capsuleService: CapsuleService;
63
92
  privateEventStore: PrivateEventStore;
64
93
  messageContextService: MessageContextService;
94
+ contractSyncService: ContractSyncService;
95
+ l2TipsStore: L2TipsProvider;
65
96
  jobId: string;
66
97
  log?: ReturnType<typeof createLogger>;
67
- scopes: AccessScopes;
98
+ scopes: AztecAddress[];
99
+ simulator: CircuitSimulator;
100
+ hooks?: ExecutionHooks;
101
+ /** Needed to trigger contract synchronization before nested cross-contract calls. */
102
+ utilityExecutor: (call: FunctionCall, scopes: AztecAddress[]) => Promise<void>;
68
103
  };
69
104
 
70
105
  /**
@@ -75,7 +110,12 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
75
110
  isUtility = true as const;
76
111
 
77
112
  private contractLogger: Logger | undefined;
113
+ private aztecnrLogger: Logger | undefined;
78
114
  private offchainEffects: OffchainEffect[] = [];
115
+ private readonly ephemeralArrayService = new EphemeralArrayService();
116
+
117
+ // We store oracle version to be able to show a nice error message when an oracle handler is missing.
118
+ private contractOracleVersion: { major: number; minor: number } | undefined;
79
119
 
80
120
  protected readonly contractAddress: AztecAddress;
81
121
  protected readonly authWitnesses: AuthWitness[];
@@ -88,12 +128,17 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
88
128
  protected readonly aztecNode: AztecNode;
89
129
  protected readonly recipientTaggingStore: RecipientTaggingStore;
90
130
  protected readonly senderAddressBookStore: SenderAddressBookStore;
91
- protected readonly capsuleStore: CapsuleStore;
131
+ protected readonly capsuleService: CapsuleService;
92
132
  protected readonly privateEventStore: PrivateEventStore;
93
133
  protected readonly messageContextService: MessageContextService;
134
+ protected readonly contractSyncService: ContractSyncService;
135
+ protected readonly l2TipsStore: L2TipsProvider;
94
136
  protected readonly jobId: string;
95
137
  protected logger: ReturnType<typeof createLogger>;
96
- protected readonly scopes: AccessScopes;
138
+ protected readonly scopes: AztecAddress[];
139
+ protected readonly simulator: CircuitSimulator;
140
+ protected readonly hooks: ExecutionHooks | undefined;
141
+ protected readonly utilityExecutor: (call: FunctionCall, scopes: AztecAddress[]) => Promise<void>;
97
142
 
98
143
  constructor(args: UtilityExecutionOracleArgs) {
99
144
  this.contractAddress = args.contractAddress;
@@ -107,33 +152,36 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
107
152
  this.aztecNode = args.aztecNode;
108
153
  this.recipientTaggingStore = args.recipientTaggingStore;
109
154
  this.senderAddressBookStore = args.senderAddressBookStore;
110
- this.capsuleStore = args.capsuleStore;
155
+ this.capsuleService = args.capsuleService;
111
156
  this.privateEventStore = args.privateEventStore;
112
157
  this.messageContextService = args.messageContextService;
158
+ this.contractSyncService = args.contractSyncService;
159
+ this.l2TipsStore = args.l2TipsStore;
113
160
  this.jobId = args.jobId;
114
161
  this.logger = args.log ?? createLogger('simulator:client_view_context');
115
162
  this.scopes = args.scopes;
163
+ this.simulator = args.simulator;
164
+ this.hooks = args.hooks;
165
+ this.utilityExecutor = args.utilityExecutor;
116
166
  }
117
167
 
118
- public assertCompatibleOracleVersion(version: number): void {
119
- // TODO(F-416): Remove this hack on v5 when protocol contracts are redeployed.
120
- // Protocol contracts/canonical contracts shipped with committed bytecode that cannot be changed. Assert they use
121
- // the expected pinned version or the current one. We want to allow for both the pinned and the current versions
122
- // because we want this code to work with both the pinned and unpinned version since some branches do not have the
123
- // pinned contracts (like e.g. next)
124
- const LEGACY_ORACLE_VERSION = 12;
125
- if (isProtocolContract(this.contractAddress)) {
126
- if (version !== LEGACY_ORACLE_VERSION && version !== ORACLE_VERSION) {
127
- throw new Error(
128
- `Expected legacy oracle version ${LEGACY_ORACLE_VERSION} or current oracle version ${ORACLE_VERSION} for alpha payload contract at ${this.contractAddress}, got ${version}.`,
129
- );
130
- }
131
- return;
168
+ public assertCompatibleOracleVersion(major: number, minor: number): void {
169
+ if (major !== ORACLE_VERSION_MAJOR) {
170
+ const hint =
171
+ major > ORACLE_VERSION_MAJOR
172
+ ? 'The contract was compiled with a newer version of Aztec.nr than your private environment supports. Upgrade your private environment to a compatible version.'
173
+ : 'The contract was compiled with an older version of Aztec.nr than your private environment supports. Recompile the contract with a compatible version of Aztec.nr.';
174
+ throw new Error(
175
+ `Incompatible private environment version: ${hint} See https://docs.aztec.network/errors/8 (expected oracle major version ${ORACLE_VERSION_MAJOR}, got ${major})`,
176
+ );
132
177
  }
133
178
 
134
- if (version !== ORACLE_VERSION) {
135
- throw new Error(`Incompatible oracle version. Expected version ${ORACLE_VERSION}, got ${version}.`);
136
- }
179
+ this.contractOracleVersion = { major, minor };
180
+ }
181
+
182
+ // Prefixed with "nonOracleFunction" as it is not used as an oracle handler.
183
+ public nonOracleFunctionGetContractOracleVersion(): { major: number; minor: number } | undefined {
184
+ return this.contractOracleVersion;
137
185
  }
138
186
 
139
187
  public getRandomField(): Fr {
@@ -152,33 +200,36 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
152
200
  * @throws If scopes are defined and the account is not in the scopes.
153
201
  */
154
202
  public async getKeyValidationRequest(pkMHash: Fr): Promise<KeyValidationRequest> {
155
- // If scopes are defined, check that the key belongs to an account in the scopes.
156
- if (this.scopes !== 'ALL_SCOPES' && this.scopes.length > 0) {
157
- let hasAccess = false;
158
- for (let i = 0; i < this.scopes.length && !hasAccess; i++) {
159
- if (await this.keyStore.accountHasKey(this.scopes[i], pkMHash)) {
160
- hasAccess = true;
161
- }
162
- }
163
- if (!hasAccess) {
164
- throw new Error(`Key validation request denied: no scoped account has a key with hash ${pkMHash.toString()}.`);
203
+ let hasAccess = false;
204
+ for (let i = 0; i < this.scopes.length && !hasAccess; i++) {
205
+ if (await this.keyStore.accountHasKey(this.scopes[i], pkMHash)) {
206
+ hasAccess = true;
165
207
  }
166
208
  }
209
+ if (!hasAccess) {
210
+ throw new Error(`Key validation request denied: no scoped account has a key with hash ${pkMHash.toString()}.`);
211
+ }
167
212
  return this.keyStore.getKeyValidationRequest(pkMHash, this.contractAddress);
168
213
  }
169
214
 
170
215
  /**
171
216
  * Fetches the index and sibling path of a leaf at a given block from the note hash tree.
172
- * @param anchorBlockHash - The hash of a block that contains the note hash tree root in which to find the membership
173
- * witness.
217
+ * @param blockHash - The hash of a block that contains the note hash tree root in which to find the
218
+ * membership witness.
174
219
  * @param noteHash - The note hash to find in the note hash tree.
175
220
  * @returns The membership witness containing the leaf index and sibling path
176
221
  */
177
- public getNoteHashMembershipWitness(
178
- anchorBlockHash: BlockHash,
222
+ public async getNoteHashMembershipWitness(
223
+ blockHash: BlockHash,
179
224
  noteHash: Fr,
180
- ): Promise<MembershipWitness<typeof NOTE_HASH_TREE_HEIGHT> | undefined> {
181
- return this.aztecNode.getNoteHashMembershipWitness(anchorBlockHash, noteHash);
225
+ ): Promise<MembershipWitness<typeof NOTE_HASH_TREE_HEIGHT>> {
226
+ const witness = await this.#queryWithBlockHashNotAfterAnchor(blockHash, () =>
227
+ this.aztecNode.getNoteHashMembershipWitness(blockHash, noteHash),
228
+ );
229
+ if (!witness) {
230
+ throw new Error(`Note hash ${noteHash} not found in the note hash tree at block ${blockHash.toString()}.`);
231
+ }
232
+ return witness;
182
233
  }
183
234
 
184
235
  /**
@@ -187,16 +238,22 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
187
238
  * Block hashes are the leaves of the archive tree. Each time a new block is added to the chain,
188
239
  * its block hash is appended as a new leaf to the archive tree.
189
240
  *
190
- * @param anchorBlockHash - The hash of a block that contains the archive tree root in which to find the membership
241
+ * @param referenceBlockHash - The hash of a block that contains the archive tree root in which to find the membership
191
242
  * witness.
192
243
  * @param blockHash - The block hash to find in the archive tree.
193
244
  * @returns The membership witness containing the leaf index and sibling path
194
245
  */
195
- public getBlockHashMembershipWitness(
196
- anchorBlockHash: BlockHash,
246
+ public async getBlockHashMembershipWitness(
247
+ referenceBlockHash: BlockHash,
197
248
  blockHash: BlockHash,
198
- ): Promise<MembershipWitness<typeof ARCHIVE_HEIGHT> | undefined> {
199
- return this.aztecNode.getBlockHashMembershipWitness(anchorBlockHash, blockHash);
249
+ ): Promise<Option<MembershipWitness<typeof ARCHIVE_HEIGHT>>> {
250
+ // Note that we validate that the reference block hash is at or before the anchor block - we don't test the block
251
+ // hash at all. If the block hash did not exist by the reference block hash, then the node will not return the
252
+ // membership witness as there is none.
253
+ const witness = await this.#queryWithBlockHashNotAfterAnchor(referenceBlockHash, () =>
254
+ this.aztecNode.getBlockHashMembershipWitness(referenceBlockHash, blockHash),
255
+ );
256
+ return witness ? Option.some(witness) : Option.none(MembershipWitness.empty(ARCHIVE_HEIGHT));
200
257
  }
201
258
 
202
259
  /**
@@ -205,11 +262,14 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
205
262
  * @param nullifier - Nullifier we try to find witness for.
206
263
  * @returns The nullifier membership witness (if found).
207
264
  */
208
- public getNullifierMembershipWitness(
209
- blockHash: BlockHash,
210
- nullifier: Fr,
211
- ): Promise<NullifierMembershipWitness | undefined> {
212
- return this.aztecNode.getNullifierMembershipWitness(blockHash, nullifier);
265
+ public async getNullifierMembershipWitness(blockHash: BlockHash, nullifier: Fr): Promise<NullifierMembershipWitness> {
266
+ const witness = await this.#queryWithBlockHashNotAfterAnchor(blockHash, () =>
267
+ this.aztecNode.getNullifierMembershipWitness(blockHash, nullifier),
268
+ );
269
+ if (!witness) {
270
+ throw new Error(`Nullifier membership witness not found at block ${blockHash.toString()}.`);
271
+ }
272
+ return witness;
213
273
  }
214
274
 
215
275
  /**
@@ -221,11 +281,19 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
221
281
  * list structure" of leaves and proving that a lower nullifier is pointing to a bigger next value than the nullifier
222
282
  * we are trying to prove non-inclusion for.
223
283
  */
224
- public getLowNullifierMembershipWitness(
284
+ public async getLowNullifierMembershipWitness(
225
285
  blockHash: BlockHash,
226
286
  nullifier: Fr,
227
- ): Promise<NullifierMembershipWitness | undefined> {
228
- return this.aztecNode.getLowNullifierMembershipWitness(blockHash, nullifier);
287
+ ): Promise<NullifierMembershipWitness> {
288
+ const witness = await this.#queryWithBlockHashNotAfterAnchor(blockHash, () =>
289
+ this.aztecNode.getLowNullifierMembershipWitness(blockHash, nullifier),
290
+ );
291
+ if (!witness) {
292
+ throw new Error(
293
+ `Low nullifier witness not found for nullifier ${nullifier} at block hash ${blockHash.toString()}.`,
294
+ );
295
+ }
296
+ return witness;
229
297
  }
230
298
 
231
299
  /**
@@ -234,8 +302,14 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
234
302
  * @param leafSlot - The slot of the public data tree to get the witness for.
235
303
  * @returns - The witness
236
304
  */
237
- public getPublicDataWitness(blockHash: BlockHash, leafSlot: Fr): Promise<PublicDataWitness | undefined> {
238
- return this.aztecNode.getPublicDataWitness(blockHash, leafSlot);
305
+ public async getPublicDataWitness(blockHash: BlockHash, leafSlot: Fr): Promise<PublicDataWitness> {
306
+ const witness = await this.#queryWithBlockHashNotAfterAnchor(blockHash, () =>
307
+ this.aztecNode.getPublicDataWitness(blockHash, leafSlot),
308
+ );
309
+ if (!witness) {
310
+ throw new Error(`Public data witness not found for slot ${leafSlot} at block hash ${blockHash.toString()}.`);
311
+ }
312
+ return witness;
239
313
  }
240
314
 
241
315
  /**
@@ -243,14 +317,22 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
243
317
  * @param blockNumber - The number of a block of which to get the block header.
244
318
  * @returns Block extracted from a block with block number `blockNumber`.
245
319
  */
246
- public async getBlockHeader(blockNumber: BlockNumber): Promise<BlockHeader | undefined> {
320
+ public async getBlockHeader(blockNumber: BlockNumber): Promise<BlockHeader> {
247
321
  const anchorBlockNumber = this.anchorBlockHeader.getBlockNumber();
248
322
  if (blockNumber > anchorBlockNumber) {
249
323
  throw new Error(`Block number ${blockNumber} is higher than current block ${anchorBlockNumber}`);
250
324
  }
251
325
 
326
+ // Most contracts query state at the "current" block, which is the anchor. Skip the RPC when we can.
327
+ if (blockNumber === anchorBlockNumber) {
328
+ return this.anchorBlockHeader;
329
+ }
330
+
252
331
  const block = await this.aztecNode.getBlock(blockNumber);
253
- return block?.header;
332
+ if (!block?.header) {
333
+ throw new Error(`Block header not found for block ${blockNumber}.`);
334
+ }
335
+ return block.header;
254
336
  }
255
337
 
256
338
  /**
@@ -258,14 +340,14 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
258
340
  * @param account - The account address.
259
341
  * @returns The public keys and partial address, or `undefined` if the account is not registered.
260
342
  */
261
- public async tryGetPublicKeysAndPartialAddress(
343
+ public async getPublicKeysAndPartialAddress(
262
344
  account: AztecAddress,
263
- ): Promise<{ publicKeys: PublicKeys; partialAddress: PartialAddress } | undefined> {
345
+ ): Promise<Option<{ publicKeys: PublicKeys; partialAddress: PartialAddress }>> {
264
346
  const completeAddress = await this.addressStore.getCompleteAddress(account);
265
347
  if (!completeAddress) {
266
- return undefined;
348
+ return Option.none({ publicKeys: PublicKeys.default(), partialAddress: Fr.ZERO });
267
349
  }
268
- return { publicKeys: completeAddress.publicKeys, partialAddress: completeAddress.partialAddress };
350
+ return Option.some({ publicKeys: completeAddress.publicKeys, partialAddress: completeAddress.partialAddress });
269
351
  }
270
352
 
271
353
  protected async getCompleteAddressOrFail(account: AztecAddress): Promise<CompleteAddress> {
@@ -293,13 +375,16 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
293
375
  }
294
376
 
295
377
  /**
296
- * Returns an auth witness for the given message hash. Checks on the list of transient witnesses
297
- * for this transaction first, and falls back to the local database if not found.
378
+ * Returns an auth witness for the given message hash from the list of transient witnesses for this transaction.
298
379
  * @param messageHash - Hash of the message to authenticate.
299
- * @returns Authentication witness for the requested message hash.
380
+ * @returns Authentication witness for the requested message hash, or undefined if not found.
300
381
  */
301
- public getAuthWitness(messageHash: Fr): Promise<Fr[] | undefined> {
302
- return Promise.resolve(this.authWitnesses.find(w => w.requestHash.equals(messageHash))?.witness);
382
+ public getAuthWitness(messageHash: Fr): Promise<Fr[]> {
383
+ const witness = this.authWitnesses.find(w => w.requestHash.equals(messageHash))?.witness;
384
+ if (!witness) {
385
+ throw new Error(`Unknown auth witness for message hash ${messageHash}`);
386
+ }
387
+ return Promise.resolve(witness);
303
388
  }
304
389
 
305
390
  /**
@@ -325,7 +410,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
325
410
  * @returns Array of note data.
326
411
  */
327
412
  public async getNotes(
328
- owner: AztecAddress | undefined,
413
+ owner: Option<AztecAddress>,
329
414
  storageSlot: Fr,
330
415
  numSelects: number,
331
416
  selectByIndexes: number[],
@@ -340,11 +425,13 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
340
425
  limit: number,
341
426
  offset: number,
342
427
  status: NoteStatus,
343
- ): Promise<NoteData[]> {
428
+ maxNotes: number,
429
+ packedHintedNoteLength: number,
430
+ ): Promise<BoundedVec<NoteData>> {
344
431
  const noteService = new NoteService(this.noteStore, this.aztecNode, this.anchorBlockHeader, this.jobId);
345
432
 
346
- const dbNotes = await noteService.getNotes(this.contractAddress, owner, storageSlot, status, this.scopes);
347
- return pickNotes<NoteData>(dbNotes, {
433
+ const dbNotes = await noteService.getNotes(this.contractAddress, owner.value, storageSlot, status, this.scopes);
434
+ const picked = pickNotes<NoteData>(dbNotes, {
348
435
  selects: selectByIndexes.slice(0, numSelects).map((index, i) => ({
349
436
  selector: { index, offset: selectByOffsets[i], length: selectByLengths[i] },
350
437
  value: selectValues[i],
@@ -357,6 +444,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
357
444
  limit,
358
445
  offset,
359
446
  });
447
+ return BoundedVec.from({ data: picked, maxLength: maxNotes, elementSize: packedHintedNoteLength });
360
448
  }
361
449
 
362
450
  /**
@@ -364,7 +452,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
364
452
  * @param innerNullifier - The inner nullifier.
365
453
  * @returns A boolean indicating whether the nullifier exists in the tree or not.
366
454
  */
367
- public async checkNullifierExists(innerNullifier: Fr) {
455
+ public async doesNullifierExist(innerNullifier: Fr) {
368
456
  const [nullifier, anchorBlockHash] = await Promise.all([
369
457
  siloNullifier(this.contractAddress, innerNullifier!),
370
458
  this.anchorBlockHeader.hash(),
@@ -376,7 +464,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
376
464
  }
377
465
 
378
466
  /**
379
- * Fetches a message from the executionStore, given its key.
467
+ * Returns the membership witness of an un-nullified L1 to L2 message.
380
468
  * @param contractAddress - Address of a contract by which the message was emitted.
381
469
  * @param messageHash - Hash of the message.
382
470
  * @param secret - Secret used to compute a nullifier.
@@ -389,6 +477,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
389
477
  contractAddress,
390
478
  messageHash,
391
479
  secret,
480
+ await this.anchorBlockHeader.hash(),
392
481
  );
393
482
 
394
483
  return new MessageLoadOracleInputs(messageIndex, siblingPath);
@@ -401,29 +490,31 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
401
490
  * @param startStorageSlot - The starting storage slot.
402
491
  * @param numberOfElements - Number of elements to read from the starting storage slot.
403
492
  */
404
- public async storageRead(
493
+ public getFromPublicStorage(
405
494
  blockHash: BlockHash,
406
495
  contractAddress: AztecAddress,
407
496
  startStorageSlot: Fr,
408
497
  numberOfElements: number,
409
498
  ) {
410
- const slots = Array(numberOfElements)
411
- .fill(0)
412
- .map((_, i) => new Fr(startStorageSlot.value + BigInt(i)));
499
+ return this.#queryWithBlockHashNotAfterAnchor(blockHash, async () => {
500
+ const slots = Array(numberOfElements)
501
+ .fill(0)
502
+ .map((_, i) => new Fr(startStorageSlot.value + BigInt(i)));
413
503
 
414
- const values = await Promise.all(
415
- slots.map(storageSlot => this.aztecNode.getPublicStorageAt(blockHash, contractAddress, storageSlot)),
416
- );
504
+ const values = await Promise.all(
505
+ slots.map(storageSlot => this.aztecNode.getPublicStorageAt(blockHash, contractAddress, storageSlot)),
506
+ );
417
507
 
418
- this.logger.debug(
419
- `Oracle storage read: slots=[${slots.map(slot => slot.toString()).join(', ')}] address=${contractAddress.toString()} values=[${values.join(', ')}]`,
420
- );
508
+ this.logger.debug(
509
+ `Oracle storage read: slots=[${slots.map(slot => slot.toString()).join(', ')}] address=${contractAddress.toString()} values=[${values.join(', ')}]`,
510
+ );
421
511
 
422
- return values;
512
+ return values;
513
+ });
423
514
  }
424
515
 
425
516
  /**
426
- * Returns a per-contract logger whose output is prefixed with `contract_log::<name>(<addrAbbrev>)`.
517
+ * Returns a per-contract logger whose output is prefixed with `contract:<name>(<addrAbbrev>)`.
427
518
  */
428
519
  async #getContractLogger(): Promise<Logger> {
429
520
  if (!this.contractLogger) {
@@ -432,245 +523,277 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
432
523
  this.contractLogger = await createContractLogger(
433
524
  this.contractAddress,
434
525
  addr => this.contractStore.getDebugContractName(addr),
526
+ 'user',
435
527
  { instanceId: this.jobId },
436
528
  );
437
529
  }
438
530
  return this.contractLogger;
439
531
  }
440
532
 
441
- public async log(level: number, message: string, fields: Fr[]): Promise<void> {
533
+ /**
534
+ * Returns a per-contract logger whose output is prefixed with `aztecnr:<name>(<addrAbbrev>)`.
535
+ */
536
+ async #getAztecnrLogger(): Promise<Logger> {
537
+ if (!this.aztecnrLogger) {
538
+ // Purpose of instanceId is to distinguish logs from different instances of the same component. It makes sense
539
+ // to re-use jobId as instanceId here as executions of different PXE jobs are isolated.
540
+ this.aztecnrLogger = await createContractLogger(
541
+ this.contractAddress,
542
+ addr => this.contractStore.getDebugContractName(addr),
543
+ 'aztecnr',
544
+ { instanceId: this.jobId },
545
+ );
546
+ }
547
+ return this.aztecnrLogger;
548
+ }
549
+
550
+ public async log(level: number, message: string, _fieldsSize: number, fields: Fr[]): Promise<void> {
442
551
  if (!LogLevels[level]) {
443
552
  throw new Error(`Invalid log level: ${level}`);
444
553
  }
445
- const logger = await this.#getContractLogger();
446
- logContractMessage(logger, LogLevels[level], message, fields);
554
+
555
+ const { kind, message: strippedMessage } = stripAztecnrLogPrefix(message);
556
+
557
+ const logger = kind == 'aztecnr' ? await this.#getAztecnrLogger() : await this.#getContractLogger();
558
+ logContractMessage(logger, LogLevels[level], strippedMessage, fields);
447
559
  }
448
560
 
449
- public async fetchTaggedLogs(pendingTaggedLogArrayBaseSlot: Fr) {
450
- const logService = new LogService(
561
+ /** Fetches pending tagged logs into a freshly allocated ephemeral array and returns it. */
562
+ public async getPendingTaggedLogs(
563
+ scope: AztecAddress,
564
+ providedSecrets: EphemeralArray<ProvidedSecret>,
565
+ ): Promise<EphemeralArray<PendingTaggedLog>> {
566
+ const secrets = providedSecrets
567
+ .readAll(this.ephemeralArrayService)
568
+ .map(ps => new AppTaggingSecret(ps.secret, this.contractAddress, ps.mode));
569
+
570
+ const logService = this.#createLogService();
571
+ const logs = await logService.fetchTaggedLogs(this.contractAddress, scope, secrets);
572
+ return EphemeralArray.fromValues(this.ephemeralArrayService, logs);
573
+ }
574
+
575
+ #createLogService(): LogService {
576
+ return new LogService(
451
577
  this.aztecNode,
452
578
  this.anchorBlockHeader,
579
+ this.l2TipsStore,
453
580
  this.keyStore,
454
- this.capsuleStore,
455
581
  this.recipientTaggingStore,
456
582
  this.senderAddressBookStore,
457
583
  this.addressStore,
458
584
  this.jobId,
459
585
  this.logger.getBindings(),
460
586
  );
461
-
462
- await logService.fetchTaggedLogs(this.contractAddress, pendingTaggedLogArrayBaseSlot, this.scopes);
463
587
  }
464
588
 
465
- /**
466
- * Validates all note and event validation requests enqueued via `enqueue_note_for_validation` and
467
- * `enqueue_event_for_validation`, inserting them into the note database and event store respectively, making them
468
- * queryable via `get_notes` and `getPrivateEvents`.
469
- *
470
- * This automatically clears both validation request queues, so no further work needs to be done by the caller.
471
- * @param contractAddress - The address of the contract that the logs are tagged for.
472
- * @param noteValidationRequestsArrayBaseSlot - The base slot of capsule array containing note validation requests.
473
- * @param eventValidationRequestsArrayBaseSlot - The base slot of capsule array containing event validation requests.
474
- */
475
589
  public async validateAndStoreEnqueuedNotesAndEvents(
476
- contractAddress: AztecAddress,
477
- noteValidationRequestsArrayBaseSlot: Fr,
478
- eventValidationRequestsArrayBaseSlot: Fr,
479
- maxNotePackedLen: number,
480
- maxEventSerializedLen: number,
590
+ noteValidationRequests: EphemeralArray<NoteValidationRequest>,
591
+ eventValidationRequests: EphemeralArray<EventValidationRequest>,
592
+ scope: AztecAddress,
481
593
  ) {
482
- // TODO(#10727): allow other contracts to store notes
483
- if (!this.contractAddress.equals(contractAddress)) {
484
- throw new Error(`Got a note validation request from ${contractAddress}, expected ${this.contractAddress}`);
485
- }
486
-
487
- // We read all note and event validation requests and process them all concurrently. This makes the process much
488
- // faster as we don't need to wait for the network round-trip.
489
- const noteValidationRequests = (
490
- await this.capsuleStore.readCapsuleArray(contractAddress, noteValidationRequestsArrayBaseSlot, this.jobId)
491
- ).map(fields => NoteValidationRequest.fromFields(fields, maxNotePackedLen));
594
+ await this.#processValidationRequests(
595
+ noteValidationRequests.readAll(this.ephemeralArrayService),
596
+ eventValidationRequests.readAll(this.ephemeralArrayService),
597
+ scope,
598
+ );
599
+ }
492
600
 
493
- const eventValidationRequests = (
494
- await this.capsuleStore.readCapsuleArray(contractAddress, eventValidationRequestsArrayBaseSlot, this.jobId)
495
- ).map(fields => EventValidationRequest.fromFields(fields, maxEventSerializedLen));
601
+ async #processValidationRequests(
602
+ noteValidationRequests: NoteValidationRequest[],
603
+ eventValidationRequests: EventValidationRequest[],
604
+ scope: AztecAddress,
605
+ ) {
606
+ const txEffects = await this.#fetchTxEffects([
607
+ ...noteValidationRequests.map(r => r.txHash),
608
+ ...eventValidationRequests.map(r => r.txHash),
609
+ ]);
496
610
 
497
611
  const noteService = new NoteService(this.noteStore, this.aztecNode, this.anchorBlockHeader, this.jobId);
498
- const noteStorePromises = noteValidationRequests.map(request =>
499
- noteService.validateAndStoreNote(
500
- request.contractAddress,
501
- request.owner,
502
- request.storageSlot,
503
- request.randomness,
504
- request.noteNonce,
505
- request.content,
506
- request.noteHash,
507
- request.nullifier,
508
- request.txHash,
509
- request.recipient,
510
- ),
511
- );
512
-
513
612
  const eventService = new EventService(this.anchorBlockHeader, this.aztecNode, this.privateEventStore, this.jobId);
514
- const eventStorePromises = eventValidationRequests.map(request =>
515
- eventService.validateAndStoreEvent(
516
- request.contractAddress,
517
- request.eventTypeId,
518
- request.randomness,
519
- request.serializedEvent,
520
- request.eventCommitment,
521
- request.txHash,
522
- request.recipient,
523
- ),
524
- );
525
613
 
526
- await Promise.all([...noteStorePromises, ...eventStorePromises]);
527
-
528
- // Requests are cleared once we're done.
529
- await this.capsuleStore.setCapsuleArray(contractAddress, noteValidationRequestsArrayBaseSlot, [], this.jobId);
530
- await this.capsuleStore.setCapsuleArray(contractAddress, eventValidationRequestsArrayBaseSlot, [], this.jobId);
614
+ await Promise.all([
615
+ noteService.validateAndStoreNotes(noteValidationRequests, scope, txEffects),
616
+ eventService.validateAndStoreEvents(eventValidationRequests, scope, txEffects),
617
+ ]);
531
618
  }
532
619
 
533
- public async bulkRetrieveLogs(
534
- contractAddress: AztecAddress,
535
- logRetrievalRequestsArrayBaseSlot: Fr,
536
- logRetrievalResponsesArrayBaseSlot: Fr,
537
- ) {
538
- // TODO(#10727): allow other contracts to process partial notes
539
- if (!this.contractAddress.equals(contractAddress)) {
540
- throw new Error(`Got a note validation request from ${contractAddress}, expected ${this.contractAddress}`);
541
- }
620
+ public async getLogsByTag(
621
+ requests: EphemeralArray<LogRetrievalRequest>,
622
+ ): Promise<EphemeralArray<EphemeralArray<LogRetrievalResponse>>> {
623
+ const logRetrievalRequests = requests.readAll(this.ephemeralArrayService);
624
+ const logService = this.#createLogService();
542
625
 
543
- // We read all log retrieval requests and process them all concurrently. This makes the process much faster as we
544
- // don't need to wait for the network round-trip.
545
- const logRetrievalRequests = (
546
- await this.capsuleStore.readCapsuleArray(contractAddress, logRetrievalRequestsArrayBaseSlot, this.jobId)
547
- ).map(LogRetrievalRequest.fromFields);
626
+ const logRetrievalResponses = await logService.fetchLogsByTag(this.contractAddress, logRetrievalRequests);
548
627
 
549
- const logService = new LogService(
550
- this.aztecNode,
551
- this.anchorBlockHeader,
552
- this.keyStore,
553
- this.capsuleStore,
554
- this.recipientTaggingStore,
555
- this.senderAddressBookStore,
556
- this.addressStore,
557
- this.jobId,
558
- this.logger.getBindings(),
628
+ // Create an inner ephemeral array for each request's matching logs, then wrap all slots in an outer array.
629
+ const innerArrays = logRetrievalResponses.map(responses =>
630
+ EphemeralArray.fromValues(this.ephemeralArrayService, responses),
559
631
  );
560
632
 
561
- const maybeLogRetrievalResponses = await logService.bulkRetrieveLogs(logRetrievalRequests);
633
+ return EphemeralArray.fromValues(this.ephemeralArrayService, innerArrays);
634
+ }
562
635
 
563
- // Requests are cleared once we're done.
564
- await this.capsuleStore.setCapsuleArray(contractAddress, logRetrievalRequestsArrayBaseSlot, [], this.jobId);
636
+ /** Reads tx hash requests from an ephemeral array, resolves their contexts, and returns the response array. */
637
+ public async getMessageContextsByTxHash(
638
+ requests: EphemeralArray<Fr>,
639
+ ): Promise<EphemeralArray<Option<MessageContext>>> {
640
+ const txHashes = requests.readAll(this.ephemeralArrayService);
565
641
 
566
- // The responses are stored as Option<LogRetrievalResponse> in a second CapsuleArray.
567
- await this.capsuleStore.setCapsuleArray(
568
- contractAddress,
569
- logRetrievalResponsesArrayBaseSlot,
570
- maybeLogRetrievalResponses.map(LogRetrievalResponse.toSerializedOption),
571
- this.jobId,
642
+ const maybeMessageContexts = await this.messageContextService.getMessageContextsByTxHash(
643
+ txHashes,
644
+ this.anchorBlockHeader.getBlockNumber(),
572
645
  );
573
- }
574
646
 
575
- public async utilityResolveMessageContexts(
576
- contractAddress: AztecAddress,
577
- messageContextRequestsArrayBaseSlot: Fr,
578
- messageContextResponsesArrayBaseSlot: Fr,
579
- ) {
580
- try {
581
- if (!this.contractAddress.equals(contractAddress)) {
582
- throw new Error(`Got a message context request from ${contractAddress}, expected ${this.contractAddress}`);
583
- }
584
- const requestCapsules = await this.capsuleStore.readCapsuleArray(
585
- contractAddress,
586
- messageContextRequestsArrayBaseSlot,
587
- this.jobId,
588
- );
589
-
590
- const txHashes = requestCapsules.map((fields, i) => {
591
- if (fields.length !== 1) {
592
- throw new Error(
593
- `Malformed message context request at index ${i}: expected 1 field (tx hash), got ${fields.length}`,
594
- );
595
- }
596
- return fields[0];
597
- });
647
+ const options = maybeMessageContexts.map(mc =>
648
+ mc ? Option.some(mc) : Option.none<MessageContext>(MessageContext.empty()),
649
+ );
650
+ return EphemeralArray.fromValues(this.ephemeralArrayService, options);
651
+ }
598
652
 
599
- const maybeMessageContexts = await this.messageContextService.resolveMessageContexts(
600
- txHashes,
601
- this.anchorBlockHeader.getBlockNumber(),
602
- );
653
+ /**
654
+ * Fetches the effects of a transaction by its hash. Returns null if the tx is not found or is beyond the anchor
655
+ * block.
656
+ */
657
+ public async getTxEffect(txHash: TxHash): Promise<Option<TxEffect>> {
658
+ if (txHash.hash.isZero()) {
659
+ throw new Error('Invalid tx hash passed into aztec_utl_getTxEffect oracle handler');
660
+ }
603
661
 
604
- // Leave response in response capsule array.
605
- await this.capsuleStore.setCapsuleArray(
606
- contractAddress,
607
- messageContextResponsesArrayBaseSlot,
608
- maybeMessageContexts.map(MessageTxContext.toSerializedOption),
609
- this.jobId,
610
- );
611
- } finally {
612
- await this.capsuleStore.setCapsuleArray(contractAddress, messageContextRequestsArrayBaseSlot, [], this.jobId);
662
+ const receipt = await this.aztecNode.getTxReceipt(txHash, { includeTxEffect: true });
663
+ if (!receipt.isMined() || !receipt.txEffect || receipt.blockNumber > this.anchorBlockHeader.getBlockNumber()) {
664
+ return Option.none(TxEffect.empty());
613
665
  }
666
+
667
+ return Option.some(receipt.txEffect);
614
668
  }
615
669
 
616
- public storeCapsule(contractAddress: AztecAddress, slot: Fr, capsule: Fr[]): Promise<void> {
670
+ public setCapsule(contractAddress: AztecAddress, slot: Fr, capsule: Fr[], scope: AztecAddress): void {
617
671
  if (!contractAddress.equals(this.contractAddress)) {
618
672
  // TODO(#10727): instead of this check that this.contractAddress is allowed to access the external DB
619
673
  throw new Error(`Contract ${contractAddress} is not allowed to access ${this.contractAddress}'s PXE DB`);
620
674
  }
621
- this.capsuleStore.storeCapsule(this.contractAddress, slot, capsule, this.jobId);
622
- return Promise.resolve();
675
+ this.capsuleService.setCapsule(contractAddress, slot, capsule, this.jobId, scope);
623
676
  }
624
677
 
625
- public async loadCapsule(contractAddress: AztecAddress, slot: Fr): Promise<Fr[] | null> {
678
+ public async getCapsule(
679
+ contractAddress: AztecAddress,
680
+ slot: Fr,
681
+ tSize: number,
682
+ scope: AztecAddress,
683
+ ): Promise<Option<Fr[]>> {
626
684
  if (!contractAddress.equals(this.contractAddress)) {
627
685
  // TODO(#10727): instead of this check that this.contractAddress is allowed to access the external DB
628
686
  throw new Error(`Contract ${contractAddress} is not allowed to access ${this.contractAddress}'s PXE DB`);
629
687
  }
630
- return (
631
- // TODO(#12425): On the following line, the pertinent capsule gets overshadowed by the transient one. Tackle this.
632
- this.capsules.find(c => c.contractAddress.equals(contractAddress) && c.storageSlot.equals(slot))?.data ??
633
- (await this.capsuleStore.loadCapsule(this.contractAddress, slot, this.jobId))
634
- );
688
+ const values = await this.capsuleService.getCapsule(contractAddress, slot, this.jobId, scope, this.capsules);
689
+ return values ? Option.some(values) : Option.none(new Array(tSize).fill(Fr.ZERO));
635
690
  }
636
691
 
637
- public deleteCapsule(contractAddress: AztecAddress, slot: Fr): Promise<void> {
692
+ public deleteCapsule(contractAddress: AztecAddress, slot: Fr, scope: AztecAddress): void {
638
693
  if (!contractAddress.equals(this.contractAddress)) {
639
694
  // TODO(#10727): instead of this check that this.contractAddress is allowed to access the external DB
640
695
  throw new Error(`Contract ${contractAddress} is not allowed to access ${this.contractAddress}'s PXE DB`);
641
696
  }
642
- this.capsuleStore.deleteCapsule(this.contractAddress, slot, this.jobId);
643
- return Promise.resolve();
697
+ this.capsuleService.deleteCapsule(contractAddress, slot, this.jobId, scope);
644
698
  }
645
699
 
646
- public copyCapsule(contractAddress: AztecAddress, srcSlot: Fr, dstSlot: Fr, numEntries: number): Promise<void> {
700
+ public copyCapsule(
701
+ contractAddress: AztecAddress,
702
+ srcSlot: Fr,
703
+ dstSlot: Fr,
704
+ numEntries: number,
705
+ scope: AztecAddress,
706
+ ): Promise<void> {
647
707
  if (!contractAddress.equals(this.contractAddress)) {
648
708
  // TODO(#10727): instead of this check that this.contractAddress is allowed to access the external DB
649
709
  throw new Error(`Contract ${contractAddress} is not allowed to access ${this.contractAddress}'s PXE DB`);
650
710
  }
651
- return this.capsuleStore.copyCapsule(this.contractAddress, srcSlot, dstSlot, numEntries, this.jobId);
711
+ return this.capsuleService.copyCapsule(contractAddress, srcSlot, dstSlot, numEntries, this.jobId, scope);
712
+ }
713
+
714
+ /**
715
+ * Clears cached sync state for a contract for a set of scopes, forcing re-sync on the next query so that newly
716
+ * stored notes or events are discovered.
717
+ */
718
+ public setContractSyncCacheInvalid(contractAddress: AztecAddress, scopes: BoundedVec<AztecAddress>): void {
719
+ if (!contractAddress.equals(this.contractAddress)) {
720
+ throw new Error(`Contract ${this.contractAddress} cannot invalidate sync cache of ${contractAddress}`);
721
+ }
722
+ this.contractSyncService.invalidateContractForScopes(contractAddress, scopes.data);
652
723
  }
653
724
 
654
725
  // TODO(#11849): consider replacing this oracle with a pure Noir implementation of aes decryption.
655
- public aes128Decrypt(ciphertext: Buffer, iv: Buffer, symKey: Buffer): Promise<Buffer> {
656
- const aes128 = new Aes128();
657
- return aes128.decryptBufferCBC(ciphertext, iv, symKey);
726
+ public async decryptAes128(
727
+ ciphertext: BoundedVec<number>,
728
+ iv: Buffer,
729
+ symKey: Buffer,
730
+ ): Promise<Option<BoundedVec<number>>> {
731
+ const capacity = ciphertext.maxLength;
732
+ try {
733
+ const aes128 = new Aes128();
734
+ const plaintext = await aes128.decryptBufferCBC(Buffer.from(ciphertext.data), iv, symKey);
735
+ return Option.some(BoundedVec.from<number>({ data: [...plaintext], maxLength: capacity }));
736
+ } catch {
737
+ return Option.none(BoundedVec.empty<number>({ maxLength: capacity }));
738
+ }
658
739
  }
659
740
 
660
741
  /**
661
- * Retrieves the shared secret for a given address and ephemeral public key.
662
- * @param address - The address to get the secret for.
663
- * @param ephPk - The ephemeral public key to get the secret for.
664
- * @returns The secret for the given address.
742
+ * Retrieves app-siloed shared secrets for multiple ephemeral public keys stored in an ephemeral array.
743
+ * @param address - The recipient address.
744
+ * @param ephPks - Ephemeral array containing the serialized Points.
745
+ * @param contractAddress - The contract address for app-siloing (validated against execution context).
746
+ * @returns A new ephemeral array containing the computed shared secrets.
665
747
  */
666
- public async getSharedSecret(address: AztecAddress, ephPk: Point): Promise<Point> {
667
- // TODO(#12656): return an app-siloed secret
748
+ public async getSharedSecrets(
749
+ address: AztecAddress,
750
+ ephPks: EphemeralArray<Point>,
751
+ contractAddress: AztecAddress,
752
+ ): Promise<EphemeralArray<Fr>> {
753
+ if (!contractAddress.equals(this.contractAddress)) {
754
+ throw new Error(
755
+ `getSharedSecrets called with contract address ${contractAddress}, expected ${this.contractAddress}`,
756
+ );
757
+ }
668
758
  const recipientCompleteAddress = await this.getCompleteAddressOrFail(address);
669
- const ivskM = await this.keyStore.getMasterSecretKey(
670
- recipientCompleteAddress.publicKeys.masterIncomingViewingPublicKey,
671
- );
759
+ const ivpkMHash = await hashPublicKey(recipientCompleteAddress.publicKeys.ivpkM);
760
+ const ivskM = await this.keyStore.getMasterSecretKey(ivpkMHash);
672
761
  const addressSecret = await computeAddressSecret(await recipientCompleteAddress.getPreaddress(), ivskM);
673
- return deriveEcdhSharedSecret(addressSecret, ephPk);
762
+
763
+ const ephPkPoints = ephPks.readAll(this.ephemeralArrayService);
764
+ const secrets = await Promise.all(
765
+ ephPkPoints.map(ephPk => deriveAppSiloedSharedSecret(addressSecret, ephPk, this.contractAddress)),
766
+ );
767
+
768
+ return EphemeralArray.fromValues(this.ephemeralArrayService, secrets);
769
+ }
770
+
771
+ public pushEphemeral(slot: Fr, elements: Fr[]): number {
772
+ return this.ephemeralArrayService.push(slot, elements);
773
+ }
774
+
775
+ public popEphemeral(slot: Fr): Fr[] {
776
+ return this.ephemeralArrayService.pop(slot);
777
+ }
778
+
779
+ public getEphemeral(slot: Fr, index: number): Fr[] {
780
+ return this.ephemeralArrayService.get(slot, index);
781
+ }
782
+
783
+ public setEphemeral(slot: Fr, index: number, elements: Fr[]): void {
784
+ this.ephemeralArrayService.set(slot, index, elements);
785
+ }
786
+
787
+ public getEphemeralLen(slot: Fr): number {
788
+ return this.ephemeralArrayService.len(slot);
789
+ }
790
+
791
+ public removeEphemeral(slot: Fr, index: number): void {
792
+ this.ephemeralArrayService.remove(slot, index);
793
+ }
794
+
795
+ public clearEphemeral(slot: Fr): void {
796
+ this.ephemeralArrayService.clear(slot);
674
797
  }
675
798
 
676
799
  public emitOffchainEffect(data: Fr[]): Promise<void> {
@@ -678,8 +801,167 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
678
801
  return Promise.resolve();
679
802
  }
680
803
 
804
+ /** Executes another utility function from within this one and returns its serialized return values. */
805
+ public async callUtilityFunction(
806
+ targetContractAddress: AztecAddress,
807
+ functionSelector: FunctionSelector,
808
+ args: Fr[],
809
+ ): Promise<Fr[]> {
810
+ const targetArtifact = await this.contractStore.getFunctionArtifactWithDebugMetadata(
811
+ targetContractAddress,
812
+ functionSelector,
813
+ );
814
+
815
+ if (!targetContractAddress.equals(this.contractAddress)) {
816
+ const [callerInstance, targetInstance] = await Promise.all([
817
+ this.getContractInstance(this.contractAddress),
818
+ this.getContractInstance(targetContractAddress),
819
+ ]);
820
+ const request: UtilityCallAuthorizationRequest = {
821
+ caller: this.contractAddress,
822
+ callerClassId: callerInstance.currentContractClassId,
823
+ target: targetContractAddress,
824
+ targetClassId: targetInstance.currentContractClassId,
825
+ functionSelector,
826
+ functionName: targetArtifact.name,
827
+ args,
828
+ callerContext: this.callerContext,
829
+ };
830
+
831
+ const response = this.hooks
832
+ ? await this.hooks.authorizeUtilityCall(request)
833
+ : { authorized: false, reason: 'No execution hooks configured' };
834
+
835
+ if (!response.authorized) {
836
+ const reason = response.reason ? `: ${response.reason}` : '';
837
+ throw new Error(
838
+ `Cross-contract utility call denied${reason}. ${this.contractAddress} attempted to call ` +
839
+ `${targetContractAddress}:${functionSelector} (${targetArtifact.name}). ` +
840
+ `See https://docs.aztec.network/errors/11`,
841
+ );
842
+ }
843
+
844
+ await this.contractSyncService.ensureContractSynced(
845
+ targetContractAddress,
846
+ functionSelector,
847
+ this.utilityExecutor,
848
+ this.anchorBlockHeader,
849
+ this.jobId,
850
+ this.scopes,
851
+ );
852
+ }
853
+
854
+ this.logger.debug(
855
+ `Calling nested utility function ${targetContractAddress}:${functionSelector} from ${this.contractAddress}`,
856
+ );
857
+
858
+ const nestedOracle = new UtilityExecutionOracle({
859
+ contractAddress: targetContractAddress,
860
+ authWitnesses: this.authWitnesses,
861
+ capsules: this.capsules,
862
+ anchorBlockHeader: this.anchorBlockHeader,
863
+ contractStore: this.contractStore,
864
+ noteStore: this.noteStore,
865
+ keyStore: this.keyStore,
866
+ addressStore: this.addressStore,
867
+ aztecNode: this.aztecNode,
868
+ recipientTaggingStore: this.recipientTaggingStore,
869
+ senderAddressBookStore: this.senderAddressBookStore,
870
+ capsuleService: this.capsuleService,
871
+ privateEventStore: this.privateEventStore,
872
+ messageContextService: this.messageContextService,
873
+ contractSyncService: this.contractSyncService,
874
+ l2TipsStore: this.l2TipsStore,
875
+ jobId: this.jobId,
876
+ scopes: this.scopes,
877
+ simulator: this.simulator,
878
+ hooks: this.hooks,
879
+ utilityExecutor: this.utilityExecutor,
880
+ log: this.logger,
881
+ });
882
+
883
+ const initialWitness = toACVMWitness(0, args);
884
+ const acvmCallback = new Oracle(nestedOracle);
885
+ const acirExecutionResult = await this.simulator
886
+ .executeUserCircuit(initialWitness, targetArtifact, acvmCallback.toACIRCallback())
887
+ .catch((err: Error) => {
888
+ err.message = resolveAssertionMessageFromError(err, targetArtifact);
889
+ throw new ExecutionError(
890
+ err.message,
891
+ { contractAddress: targetContractAddress, functionSelector },
892
+ extractCallStack(err, targetArtifact.debug),
893
+ { cause: err },
894
+ );
895
+ });
896
+
897
+ return witnessMapToFields(acirExecutionResult.returnWitness);
898
+ }
899
+
681
900
  /** Returns offchain effects collected during execution. */
682
901
  public getOffchainEffects(): OffchainEffect[] {
683
902
  return this.offchainEffects;
684
903
  }
904
+
905
+ /**
906
+ * Fetches tx effects for the given hashes in parallel, deduplicating repeated hashes so each tx is only requested
907
+ * once. Returns a map keyed by `TxHash.toString()`; hashes for which the node has no tx effect are omitted.
908
+ */
909
+ async #fetchTxEffects(txHashes: TxHash[]): Promise<Map<string, IndexedTxEffect>> {
910
+ const uniqueTxHashes = uniqueBy(txHashes, h => h.toString());
911
+ const fetched = await Promise.all(
912
+ uniqueTxHashes.map(h => this.aztecNode.getTxReceipt(h, { includeTxEffect: true })),
913
+ );
914
+ return new Map(
915
+ uniqueTxHashes
916
+ .map((h, i): [string, IndexedTxEffect | undefined] => {
917
+ const receipt = fetched[i];
918
+ if (!receipt.isMined() || !receipt.txEffect) {
919
+ return [h.toString(), undefined];
920
+ }
921
+ return [
922
+ h.toString(),
923
+ {
924
+ data: receipt.txEffect,
925
+ l2BlockNumber: receipt.blockNumber,
926
+ l2BlockHash: receipt.blockHash,
927
+ txIndexInBlock: receipt.txIndexInBlock,
928
+ slotNumber: receipt.slotNumber,
929
+ },
930
+ ];
931
+ })
932
+ .filter((entry): entry is [string, IndexedTxEffect] => entry[1] !== undefined),
933
+ );
934
+ }
935
+
936
+ /** Runs a query concurrently with a validation that the block hash is not ahead of the anchor block. */
937
+ async #queryWithBlockHashNotAfterAnchor<T>(blockHash: BlockHash, query: () => Promise<T>): Promise<T> {
938
+ // Most contracts query state at the "current" block, which is the anchor. Skip the validation when we can.
939
+ const anchorHash = await this.anchorBlockHeader.hash();
940
+ if (blockHash.equals(anchorHash)) {
941
+ return query();
942
+ }
943
+
944
+ const [response] = await Promise.all([
945
+ query(),
946
+ (async () => {
947
+ const block = await this.aztecNode.getBlock(blockHash);
948
+ const header = block?.header;
949
+ if (!header) {
950
+ throw new Error(`Could not find block header for block hash ${blockHash}`);
951
+ }
952
+
953
+ if (header.getBlockNumber() > this.anchorBlockHeader.getBlockNumber()) {
954
+ throw new Error(
955
+ `Made a node query with a reference block hash ${blockHash} with block number ${header.getBlockNumber()}, which is ahead of the anchor block number ${this.anchorBlockHeader.getBlockNumber()} (from anchor block hash ${anchorHash}).`,
956
+ );
957
+ }
958
+ })(),
959
+ ]);
960
+ return response;
961
+ }
962
+
963
+ /** The execution context of the current call. */
964
+ protected get callerContext(): 'private' | 'private view' | 'utility' {
965
+ return 'utility';
966
+ }
685
967
  }