@aztec/pxe 0.0.1-commit.e3c1de76 → 0.0.1-commit.e57c76e

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 (347) 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 +9 -3
  12. package/dest/block_synchronizer/block_synchronizer.d.ts.map +1 -1
  13. package/dest/block_synchronizer/block_synchronizer.js +37 -11
  14. package/dest/config/index.d.ts +8 -2
  15. package/dest/config/index.d.ts.map +1 -1
  16. package/dest/config/index.js +13 -15
  17. package/dest/config/package_info.js +1 -1
  18. package/dest/contract_function_simulator/contract_function_simulator.d.ts +66 -29
  19. package/dest/contract_function_simulator/contract_function_simulator.d.ts.map +1 -1
  20. package/dest/contract_function_simulator/contract_function_simulator.js +216 -75
  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 +7 -11
  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 +19 -15
  29. package/dest/contract_function_simulator/index.d.ts +14 -2
  30. package/dest/contract_function_simulator/index.d.ts.map +1 -1
  31. package/dest/contract_function_simulator/index.js +11 -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 +3 -4
  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 +8 -9
  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 +2 -2
  45. package/dest/contract_function_simulator/noir-structs/log_retrieval_response.d.ts.map +1 -1
  46. package/dest/contract_function_simulator/noir-structs/log_retrieval_response.js +2 -4
  47. package/dest/contract_function_simulator/noir-structs/note_data.d.ts +27 -0
  48. package/dest/contract_function_simulator/noir-structs/note_data.d.ts.map +1 -0
  49. package/dest/contract_function_simulator/noir-structs/note_data.js +3 -0
  50. package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts +3 -5
  51. package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts.map +1 -1
  52. package/dest/contract_function_simulator/noir-structs/note_validation_request.js +6 -10
  53. package/dest/contract_function_simulator/noir-structs/option.d.ts +61 -0
  54. package/dest/contract_function_simulator/noir-structs/option.d.ts.map +1 -0
  55. package/dest/contract_function_simulator/noir-structs/option.js +62 -0
  56. package/dest/contract_function_simulator/noir-structs/provided_secret.d.ts +11 -0
  57. package/dest/contract_function_simulator/noir-structs/provided_secret.d.ts.map +1 -0
  58. package/dest/contract_function_simulator/noir-structs/provided_secret.js +24 -0
  59. package/dest/contract_function_simulator/oracle/interfaces.d.ts +16 -98
  60. package/dest/contract_function_simulator/oracle/interfaces.d.ts.map +1 -1
  61. package/dest/contract_function_simulator/oracle/interfaces.js +2 -2
  62. package/dest/contract_function_simulator/oracle/note_packing_utils.d.ts +2 -2
  63. package/dest/contract_function_simulator/oracle/note_packing_utils.d.ts.map +1 -1
  64. package/dest/contract_function_simulator/oracle/note_packing_utils.js +2 -2
  65. package/dest/contract_function_simulator/oracle/oracle.d.ts +75 -46
  66. package/dest/contract_function_simulator/oracle/oracle.d.ts.map +1 -1
  67. package/dest/contract_function_simulator/oracle/oracle.js +471 -275
  68. package/dest/contract_function_simulator/oracle/oracle_registry.d.ts +127 -0
  69. package/dest/contract_function_simulator/oracle/oracle_registry.d.ts.map +1 -0
  70. package/dest/contract_function_simulator/oracle/oracle_registry.js +786 -0
  71. package/dest/contract_function_simulator/oracle/oracle_type_mappings.d.ts +139 -0
  72. package/dest/contract_function_simulator/oracle/oracle_type_mappings.d.ts.map +1 -0
  73. package/dest/contract_function_simulator/oracle/oracle_type_mappings.js +560 -0
  74. package/dest/contract_function_simulator/oracle/private_execution.js +5 -3
  75. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts +68 -95
  76. package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts.map +1 -1
  77. package/dest/contract_function_simulator/oracle/private_execution_oracle.js +141 -110
  78. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts +156 -76
  79. package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts.map +1 -1
  80. package/dest/contract_function_simulator/oracle/utility_execution_oracle.js +450 -166
  81. package/dest/contract_function_simulator/pick_notes.d.ts +1 -1
  82. package/dest/contract_function_simulator/pick_notes.d.ts.map +1 -1
  83. package/dest/contract_function_simulator/pick_notes.js +20 -3
  84. package/dest/contract_function_simulator/proxied_contract_data_source.d.ts +1 -1
  85. package/dest/contract_function_simulator/proxied_contract_data_source.d.ts.map +1 -1
  86. package/dest/contract_function_simulator/proxied_contract_data_source.js +35 -64
  87. package/dest/contract_logging.d.ts +27 -0
  88. package/dest/contract_logging.d.ts.map +1 -0
  89. package/dest/contract_logging.js +38 -0
  90. package/dest/contract_sync/contract_sync_service.d.ts +42 -0
  91. package/dest/contract_sync/contract_sync_service.d.ts.map +1 -0
  92. package/dest/contract_sync/contract_sync_service.js +135 -0
  93. package/dest/contract_sync/helpers.d.ts +27 -0
  94. package/dest/contract_sync/helpers.d.ts.map +1 -0
  95. package/dest/contract_sync/{index.js → helpers.js} +16 -17
  96. package/dest/debug/pxe_debug_utils.d.ts +14 -15
  97. package/dest/debug/pxe_debug_utils.d.ts.map +1 -1
  98. package/dest/debug/pxe_debug_utils.js +16 -21
  99. package/dest/entrypoints/client/bundle/index.d.ts +3 -1
  100. package/dest/entrypoints/client/bundle/index.d.ts.map +1 -1
  101. package/dest/entrypoints/client/bundle/index.js +2 -0
  102. package/dest/entrypoints/client/bundle/utils.d.ts +2 -2
  103. package/dest/entrypoints/client/bundle/utils.d.ts.map +1 -1
  104. package/dest/entrypoints/client/bundle/utils.js +21 -5
  105. package/dest/entrypoints/client/lazy/index.d.ts +3 -1
  106. package/dest/entrypoints/client/lazy/index.d.ts.map +1 -1
  107. package/dest/entrypoints/client/lazy/index.js +2 -0
  108. package/dest/entrypoints/client/lazy/utils.d.ts +2 -2
  109. package/dest/entrypoints/client/lazy/utils.d.ts.map +1 -1
  110. package/dest/entrypoints/client/lazy/utils.js +21 -5
  111. package/dest/entrypoints/pxe_creation_options.d.ts +9 -1
  112. package/dest/entrypoints/pxe_creation_options.d.ts.map +1 -1
  113. package/dest/entrypoints/pxe_creation_options.js +3 -1
  114. package/dest/entrypoints/server/index.d.ts +5 -3
  115. package/dest/entrypoints/server/index.d.ts.map +1 -1
  116. package/dest/entrypoints/server/index.js +4 -2
  117. package/dest/entrypoints/server/utils.d.ts +4 -3
  118. package/dest/entrypoints/server/utils.d.ts.map +1 -1
  119. package/dest/entrypoints/server/utils.js +21 -5
  120. package/dest/events/event_service.d.ts +15 -6
  121. package/dest/events/event_service.d.ts.map +1 -1
  122. package/dest/events/event_service.js +44 -11
  123. package/dest/events/private_event_filter_validator.d.ts +3 -2
  124. package/dest/events/private_event_filter_validator.d.ts.map +1 -1
  125. package/dest/events/private_event_filter_validator.js +15 -0
  126. package/dest/hooks/authorize_utility_call.d.ts +41 -0
  127. package/dest/hooks/authorize_utility_call.d.ts.map +1 -0
  128. package/dest/hooks/authorize_utility_call.js +4 -0
  129. package/dest/hooks/execution_hooks.d.ts +42 -0
  130. package/dest/hooks/execution_hooks.d.ts.map +1 -0
  131. package/dest/hooks/execution_hooks.js +9 -0
  132. package/dest/hooks/index.d.ts +4 -0
  133. package/dest/hooks/index.d.ts.map +1 -0
  134. package/dest/hooks/index.js +1 -0
  135. package/dest/logs/log_service.d.ts +9 -8
  136. package/dest/logs/log_service.d.ts.map +1 -1
  137. package/dest/logs/log_service.js +131 -84
  138. package/dest/messages/message_context_service.d.ts +17 -0
  139. package/dest/messages/message_context_service.d.ts.map +1 -0
  140. package/dest/messages/message_context_service.js +55 -0
  141. package/dest/notes/note_service.d.ts +27 -5
  142. package/dest/notes/note_service.d.ts.map +1 -1
  143. package/dest/notes/note_service.js +83 -58
  144. package/dest/notes_filter.d.ts +24 -0
  145. package/dest/notes_filter.d.ts.map +1 -0
  146. package/dest/notes_filter.js +4 -0
  147. package/dest/oracle_version.d.ts +4 -3
  148. package/dest/oracle_version.d.ts.map +1 -1
  149. package/dest/oracle_version.js +20 -10
  150. package/dest/private_kernel/batch_planner.d.ts +47 -0
  151. package/dest/private_kernel/batch_planner.d.ts.map +1 -0
  152. package/dest/private_kernel/batch_planner.js +104 -0
  153. package/dest/private_kernel/hints/compute_tx_expiration_timestamp.d.ts +4 -0
  154. package/dest/private_kernel/hints/compute_tx_expiration_timestamp.d.ts.map +1 -0
  155. package/dest/private_kernel/hints/{compute_tx_include_by_timestamp.js → compute_tx_expiration_timestamp.js} +12 -12
  156. package/dest/private_kernel/hints/index.d.ts +1 -1
  157. package/dest/private_kernel/hints/index.js +1 -1
  158. package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.d.ts +4 -3
  159. package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.d.ts.map +1 -1
  160. package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.js +130 -69
  161. package/dest/private_kernel/hints/test_utils.d.ts +122 -0
  162. package/dest/private_kernel/hints/test_utils.d.ts.map +1 -0
  163. package/dest/private_kernel/hints/test_utils.js +202 -0
  164. package/dest/private_kernel/private_kernel_execution_prover.d.ts +6 -2
  165. package/dest/private_kernel/private_kernel_execution_prover.d.ts.map +1 -1
  166. package/dest/private_kernel/private_kernel_execution_prover.js +166 -65
  167. package/dest/private_kernel/private_kernel_oracle.d.ts +15 -11
  168. package/dest/private_kernel/private_kernel_oracle.d.ts.map +1 -1
  169. package/dest/private_kernel/private_kernel_oracle.js +31 -25
  170. package/dest/pxe.d.ts +120 -24
  171. package/dest/pxe.d.ts.map +1 -1
  172. package/dest/pxe.js +231 -134
  173. package/dest/storage/anchor_block_store/anchor_block_store.js +1 -1
  174. package/dest/storage/backwards_compatibility_tests/kv_store_snapshot.d.ts +42 -0
  175. package/dest/storage/backwards_compatibility_tests/kv_store_snapshot.d.ts.map +1 -0
  176. package/dest/storage/backwards_compatibility_tests/kv_store_snapshot.js +93 -0
  177. package/dest/storage/backwards_compatibility_tests/schema_tests.d.ts +15 -0
  178. package/dest/storage/backwards_compatibility_tests/schema_tests.d.ts.map +1 -0
  179. package/dest/storage/backwards_compatibility_tests/schema_tests.js +591 -0
  180. package/dest/storage/backwards_compatibility_tests/store_spy.d.ts +19 -0
  181. package/dest/storage/backwards_compatibility_tests/store_spy.d.ts.map +1 -0
  182. package/dest/storage/backwards_compatibility_tests/store_spy.js +63 -0
  183. package/dest/storage/capsule_store/capsule_service.d.ts +21 -0
  184. package/dest/storage/capsule_store/capsule_service.d.ts.map +1 -0
  185. package/dest/storage/capsule_store/capsule_service.js +50 -0
  186. package/dest/storage/capsule_store/capsule_store.d.ts +9 -9
  187. package/dest/storage/capsule_store/capsule_store.d.ts.map +1 -1
  188. package/dest/storage/capsule_store/capsule_store.js +36 -28
  189. package/dest/storage/capsule_store/index.d.ts +2 -1
  190. package/dest/storage/capsule_store/index.d.ts.map +1 -1
  191. package/dest/storage/capsule_store/index.js +1 -0
  192. package/dest/storage/contract_store/contract_store.d.ts +42 -15
  193. package/dest/storage/contract_store/contract_store.d.ts.map +1 -1
  194. package/dest/storage/contract_store/contract_store.js +146 -87
  195. package/dest/storage/metadata.d.ts +1 -1
  196. package/dest/storage/metadata.js +1 -1
  197. package/dest/storage/note_store/note_store.d.ts +3 -3
  198. package/dest/storage/note_store/note_store.d.ts.map +1 -1
  199. package/dest/storage/note_store/note_store.js +6 -4
  200. package/dest/storage/open_pxe_stores.d.ts +33 -0
  201. package/dest/storage/open_pxe_stores.d.ts.map +1 -0
  202. package/dest/storage/open_pxe_stores.js +27 -0
  203. package/dest/storage/private_event_store/private_event_store.d.ts +1 -1
  204. package/dest/storage/private_event_store/private_event_store.d.ts.map +1 -1
  205. package/dest/storage/private_event_store/private_event_store.js +3 -0
  206. package/dest/storage/private_event_store/stored_private_event.js +1 -1
  207. package/dest/storage/tagging_store/recipient_tagging_store.d.ts +6 -6
  208. package/dest/storage/tagging_store/recipient_tagging_store.d.ts.map +1 -1
  209. package/dest/storage/tagging_store/sender_tagging_store.d.ts +29 -28
  210. package/dest/storage/tagging_store/sender_tagging_store.d.ts.map +1 -1
  211. package/dest/storage/tagging_store/sender_tagging_store.js +141 -115
  212. package/dest/tagging/get_all_logs_by_tags.d.ts +34 -10
  213. package/dest/tagging/get_all_logs_by_tags.d.ts.map +1 -1
  214. package/dest/tagging/get_all_logs_by_tags.js +45 -32
  215. package/dest/tagging/index.d.ts +6 -5
  216. package/dest/tagging/index.d.ts.map +1 -1
  217. package/dest/tagging/index.js +4 -3
  218. package/dest/tagging/persist_sender_tagging_index_ranges.d.ts +29 -0
  219. package/dest/tagging/persist_sender_tagging_index_ranges.d.ts.map +1 -0
  220. package/dest/tagging/persist_sender_tagging_index_ranges.js +42 -0
  221. package/dest/tagging/recipient_sync/sync_tagged_private_logs.d.ts +56 -0
  222. package/dest/tagging/recipient_sync/sync_tagged_private_logs.d.ts.map +1 -0
  223. package/dest/tagging/recipient_sync/sync_tagged_private_logs.js +163 -0
  224. package/dest/tagging/recipient_sync/utils/find_highest_indexes.d.ts +3 -3
  225. package/dest/tagging/recipient_sync/utils/find_highest_indexes.d.ts.map +1 -1
  226. package/dest/tagging/recipient_sync/utils/find_highest_indexes.js +2 -2
  227. package/dest/tagging/reconcile_tagging_index_ranges.d.ts +36 -0
  228. package/dest/tagging/reconcile_tagging_index_ranges.d.ts.map +1 -0
  229. package/dest/tagging/reconcile_tagging_index_ranges.js +74 -0
  230. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts +4 -9
  231. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts.map +1 -1
  232. package/dest/tagging/sender_sync/sync_sender_tagging_indexes.js +32 -14
  233. package/dest/tagging/sender_sync/utils/get_status_change_of_pending.d.ts +13 -7
  234. package/dest/tagging/sender_sync/utils/get_status_change_of_pending.d.ts.map +1 -1
  235. package/dest/tagging/sender_sync/utils/get_status_change_of_pending.js +41 -10
  236. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts +5 -7
  237. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts.map +1 -1
  238. package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.js +37 -25
  239. package/package.json +29 -17
  240. package/src/bin/check_oracle_version.ts +49 -122
  241. package/src/bin/index.ts +1 -0
  242. package/src/bin/oracle_version_helpers.ts +121 -0
  243. package/src/block_synchronizer/block_stream_source.ts +81 -0
  244. package/src/block_synchronizer/block_synchronizer.ts +39 -11
  245. package/src/config/index.ts +15 -9
  246. package/src/config/package_info.ts +1 -1
  247. package/src/contract_function_simulator/contract_function_simulator.ts +391 -136
  248. package/src/contract_function_simulator/ephemeral_array_service.ts +110 -0
  249. package/src/contract_function_simulator/execution_note_cache.ts +1 -1
  250. package/src/contract_function_simulator/execution_tagging_index_cache.ts +19 -18
  251. package/src/contract_function_simulator/index.ts +51 -1
  252. package/src/contract_function_simulator/noir-structs/bounded_vec.ts +55 -0
  253. package/src/contract_function_simulator/noir-structs/ephemeral_array.ts +66 -0
  254. package/src/contract_function_simulator/noir-structs/event_validation_request.ts +9 -8
  255. package/src/contract_function_simulator/noir-structs/log_retrieval_request.ts +36 -3
  256. package/src/contract_function_simulator/noir-structs/log_retrieval_response.ts +2 -5
  257. package/src/contract_function_simulator/noir-structs/note_data.ts +27 -0
  258. package/src/contract_function_simulator/noir-structs/note_validation_request.ts +4 -9
  259. package/src/contract_function_simulator/noir-structs/option.ts +69 -0
  260. package/src/contract_function_simulator/noir-structs/provided_secret.ts +27 -0
  261. package/src/contract_function_simulator/oracle/interfaces.ts +12 -176
  262. package/src/contract_function_simulator/oracle/note_packing_utils.ts +3 -3
  263. package/src/contract_function_simulator/oracle/oracle.ts +564 -473
  264. package/src/contract_function_simulator/oracle/oracle_registry.ts +585 -0
  265. package/src/contract_function_simulator/oracle/oracle_type_mappings.ts +553 -0
  266. package/src/contract_function_simulator/oracle/private_execution.ts +4 -4
  267. package/src/contract_function_simulator/oracle/private_execution_oracle.ts +173 -205
  268. package/src/contract_function_simulator/oracle/utility_execution_oracle.ts +664 -245
  269. package/src/contract_function_simulator/pick_notes.ts +22 -3
  270. package/src/contract_function_simulator/proxied_contract_data_source.ts +41 -64
  271. package/src/contract_logging.ts +52 -0
  272. package/src/contract_sync/contract_sync_service.ts +189 -0
  273. package/src/contract_sync/{index.ts → helpers.ts} +17 -29
  274. package/src/debug/pxe_debug_utils.ts +47 -25
  275. package/src/entrypoints/client/bundle/index.ts +2 -0
  276. package/src/entrypoints/client/bundle/utils.ts +19 -6
  277. package/src/entrypoints/client/lazy/index.ts +2 -0
  278. package/src/entrypoints/client/lazy/utils.ts +19 -6
  279. package/src/entrypoints/pxe_creation_options.ts +14 -0
  280. package/src/entrypoints/server/index.ts +4 -2
  281. package/src/entrypoints/server/utils.ts +22 -13
  282. package/src/events/event_service.ts +69 -21
  283. package/src/events/private_event_filter_validator.ts +21 -1
  284. package/src/hooks/authorize_utility_call.ts +44 -0
  285. package/src/hooks/execution_hooks.ts +48 -0
  286. package/src/hooks/index.ts +7 -0
  287. package/src/logs/log_service.ts +162 -145
  288. package/src/messages/message_context_service.ts +62 -0
  289. package/src/notes/note_service.ts +120 -85
  290. package/src/notes_filter.ts +24 -0
  291. package/src/oracle_version.ts +20 -10
  292. package/src/private_kernel/batch_planner.ts +169 -0
  293. package/src/private_kernel/hints/{compute_tx_include_by_timestamp.ts → compute_tx_expiration_timestamp.ts} +13 -13
  294. package/src/private_kernel/hints/index.ts +1 -1
  295. package/src/private_kernel/hints/private_kernel_reset_private_inputs_builder.ts +165 -118
  296. package/src/private_kernel/hints/test_utils.ts +318 -0
  297. package/src/private_kernel/private_kernel_execution_prover.ts +254 -89
  298. package/src/private_kernel/private_kernel_oracle.ts +42 -32
  299. package/src/pxe.ts +416 -181
  300. package/src/storage/anchor_block_store/anchor_block_store.ts +1 -1
  301. package/src/storage/backwards_compatibility_tests/__snapshots__/AddressStore.json +22 -0
  302. package/src/storage/backwards_compatibility_tests/__snapshots__/AnchorBlockStore.json +3 -0
  303. package/src/storage/backwards_compatibility_tests/__snapshots__/CapsuleStore.json +16 -0
  304. package/src/storage/backwards_compatibility_tests/__snapshots__/ContractStore.json +28 -0
  305. package/src/storage/backwards_compatibility_tests/__snapshots__/KeyStore.json +52 -0
  306. package/src/storage/backwards_compatibility_tests/__snapshots__/L2TipsKVStore.json +46 -0
  307. package/src/storage/backwards_compatibility_tests/__snapshots__/NoteStore.json +36 -0
  308. package/src/storage/backwards_compatibility_tests/__snapshots__/PrivateEventStore.json +44 -0
  309. package/src/storage/backwards_compatibility_tests/__snapshots__/RecipientTaggingStore.json +18 -0
  310. package/src/storage/backwards_compatibility_tests/__snapshots__/SenderAddressBookStore.json +16 -0
  311. package/src/storage/backwards_compatibility_tests/__snapshots__/SenderTaggingStore.json +22 -0
  312. package/src/storage/backwards_compatibility_tests/__snapshots__/opened_stores.json +97 -0
  313. package/src/storage/backwards_compatibility_tests/kv_store_snapshot.ts +122 -0
  314. package/src/storage/backwards_compatibility_tests/schema_tests.ts +712 -0
  315. package/src/storage/backwards_compatibility_tests/store_spy.ts +73 -0
  316. package/src/storage/capsule_store/capsule_service.ts +90 -0
  317. package/src/storage/capsule_store/capsule_store.ts +44 -26
  318. package/src/storage/capsule_store/index.ts +1 -0
  319. package/src/storage/contract_store/contract_store.ts +182 -104
  320. package/src/storage/metadata.ts +1 -1
  321. package/src/storage/note_store/note_store.ts +9 -5
  322. package/src/storage/open_pxe_stores.ts +49 -0
  323. package/src/storage/private_event_store/private_event_store.ts +4 -0
  324. package/src/storage/private_event_store/stored_private_event.ts +1 -1
  325. package/src/storage/tagging_store/recipient_tagging_store.ts +5 -5
  326. package/src/storage/tagging_store/sender_tagging_store.ts +185 -138
  327. package/src/tagging/get_all_logs_by_tags.ts +87 -35
  328. package/src/tagging/index.ts +5 -4
  329. package/src/tagging/persist_sender_tagging_index_ranges.ts +57 -0
  330. package/src/tagging/recipient_sync/sync_tagged_private_logs.ts +240 -0
  331. package/src/tagging/recipient_sync/utils/find_highest_indexes.ts +4 -4
  332. package/src/tagging/reconcile_tagging_index_ranges.ts +102 -0
  333. package/src/tagging/sender_sync/sync_sender_tagging_indexes.ts +52 -17
  334. package/src/tagging/sender_sync/utils/get_status_change_of_pending.ts +44 -14
  335. package/src/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.ts +27 -27
  336. package/dest/contract_sync/index.d.ts +0 -23
  337. package/dest/contract_sync/index.d.ts.map +0 -1
  338. package/dest/private_kernel/hints/compute_tx_include_by_timestamp.d.ts +0 -4
  339. package/dest/private_kernel/hints/compute_tx_include_by_timestamp.d.ts.map +0 -1
  340. package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.d.ts +0 -15
  341. package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.d.ts.map +0 -1
  342. package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.js +0 -99
  343. package/dest/tagging/recipient_sync/utils/load_logs_for_range.d.ts +0 -15
  344. package/dest/tagging/recipient_sync/utils/load_logs_for_range.d.ts.map +0 -1
  345. package/dest/tagging/recipient_sync/utils/load_logs_for_range.js +0 -32
  346. package/src/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.ts +0 -143
  347. package/src/tagging/recipient_sync/utils/load_logs_for_range.ts +0 -49
@@ -1,32 +1,38 @@
1
- import type { Fr } from '@aztec/foundation/curves/bn254';
1
+ import type { BlockNumber } from '@aztec/foundation/branded-types';
2
2
  import { type Logger, type LoggerBindings, createLogger } from '@aztec/foundation/log';
3
3
  import type { KeyStore } from '@aztec/key-store';
4
4
  import { AztecAddress } from '@aztec/stdlib/aztec-address';
5
- import type { CompleteAddress } from '@aztec/stdlib/contract';
5
+ import type { BlockHash, L2TipsProvider } from '@aztec/stdlib/block';
6
6
  import type { AztecNode } from '@aztec/stdlib/interfaces/server';
7
- import { DirectionalAppTaggingSecret, PendingTaggedLog, SiloedTag, Tag, TxScopedL2Log } from '@aztec/stdlib/logs';
7
+ import { AppTaggingSecret, type LogResult, PendingTaggedLog, SiloedTag } from '@aztec/stdlib/logs';
8
8
  import type { BlockHeader } from '@aztec/stdlib/tx';
9
9
 
10
- import type { LogRetrievalRequest } from '../contract_function_simulator/noir-structs/log_retrieval_request.js';
10
+ import {
11
+ type LogRetrievalRequest,
12
+ LogSource,
13
+ } from '../contract_function_simulator/noir-structs/log_retrieval_request.js';
11
14
  import { LogRetrievalResponse } from '../contract_function_simulator/noir-structs/log_retrieval_response.js';
12
15
  import { AddressStore } from '../storage/address_store/address_store.js';
13
- import { CapsuleStore } from '../storage/capsule_store/capsule_store.js';
14
16
  import type { RecipientTaggingStore } from '../storage/tagging_store/recipient_tagging_store.js';
15
17
  import type { SenderAddressBookStore } from '../storage/tagging_store/sender_address_book_store.js';
16
18
  import {
17
19
  getAllPrivateLogsByTags,
18
20
  getAllPublicLogsByTagsFromContract,
19
- loadPrivateLogsForSenderRecipientPair,
21
+ syncTaggedPrivateLogs,
20
22
  } from '../tagging/index.js';
21
23
 
24
+ /** Key used to group requests by their (fromBlock, toBlock) range so each group becomes a single node call. */
25
+ type RangeKey = string;
26
+ const rangeKey = (fromBlock?: BlockNumber, toBlock?: BlockNumber): RangeKey => `${fromBlock ?? ''}-${toBlock ?? ''}`;
27
+
22
28
  export class LogService {
23
29
  private log: Logger;
24
30
 
25
31
  constructor(
26
32
  private readonly aztecNode: AztecNode,
27
33
  private readonly anchorBlockHeader: BlockHeader,
34
+ private readonly l2TipsStore: L2TipsProvider,
28
35
  private readonly keyStore: KeyStore,
29
- private readonly capsuleStore: CapsuleStore,
30
36
  private readonly recipientTaggingStore: RecipientTaggingStore,
31
37
  private readonly senderAddressBookStore: SenderAddressBookStore,
32
38
  private readonly addressStore: AddressStore,
@@ -36,134 +42,170 @@ export class LogService {
36
42
  this.log = createLogger('pxe:log_service', bindings);
37
43
  }
38
44
 
39
- public async bulkRetrieveLogs(logRetrievalRequests: LogRetrievalRequest[]): Promise<(LogRetrievalResponse | null)[]> {
40
- return await Promise.all(
41
- logRetrievalRequests.map(async request => {
42
- const [publicLog, privateLog] = await Promise.all([
43
- this.#getPublicLogByTag(request.tag, request.contractAddress),
44
- this.#getPrivateLogByTag(await SiloedTag.compute(request.tag, request.contractAddress)),
45
- ]);
46
-
47
- if (publicLog !== null && privateLog !== null) {
48
- throw new Error(
49
- `Found both a public and private log when searching for tag ${request.tag} from contract ${request.contractAddress}`,
50
- );
51
- }
45
+ /** Fetches all logs matching each request's tag, returning an array of log arrays (one per request). */
46
+ public async fetchLogsByTag(
47
+ contractAddress: AztecAddress,
48
+ logRetrievalRequests: LogRetrievalRequest[],
49
+ ): Promise<LogRetrievalResponse[][]> {
50
+ for (const request of logRetrievalRequests) {
51
+ if (!contractAddress.equals(request.contractAddress)) {
52
+ throw new Error(`Got a log retrieval request from ${request.contractAddress}, expected ${contractAddress}`);
53
+ }
54
+ }
52
55
 
53
- return publicLog ?? privateLog;
54
- }),
55
- );
56
- }
56
+ if (logRetrievalRequests.length === 0) {
57
+ return [];
58
+ }
57
59
 
58
- async #getPublicLogByTag(tag: Tag, contractAddress: AztecAddress): Promise<LogRetrievalResponse | null> {
59
60
  const anchorBlockHash = await this.anchorBlockHeader.hash();
60
- const allLogsPerTag = await getAllPublicLogsByTagsFromContract(
61
- this.aztecNode,
62
- contractAddress,
63
- [tag],
64
- anchorBlockHash,
65
- );
66
- const logsForTag = allLogsPerTag[0];
67
-
68
- if (logsForTag.length === 0) {
69
- return null;
70
- } else if (logsForTag.length > 1) {
71
- // TODO(#11627): handle this case
72
- throw new Error(
73
- `Got ${logsForTag.length} logs for tag ${tag} and contract ${contractAddress.toString()}. getPublicLogByTag currently only supports a single log per tag`,
74
- );
75
- }
76
61
 
77
- const scopedLog = logsForTag[0];
62
+ const [publicLogsPerRequest, privateLogsPerRequest] = await Promise.all([
63
+ this.#fetchPublicLogs(contractAddress, logRetrievalRequests, anchorBlockHash),
64
+ this.#fetchPrivateLogs(logRetrievalRequests, anchorBlockHash),
65
+ ]);
78
66
 
79
- return new LogRetrievalResponse(
80
- scopedLog.logData.slice(1), // Skip the tag
81
- scopedLog.txHash,
82
- scopedLog.noteHashes,
83
- scopedLog.firstNullifier,
84
- );
67
+ return logRetrievalRequests.map((_request, i) => [
68
+ ...publicLogsPerRequest[i].map(LogService.#toLogRetrievalResponse),
69
+ ...privateLogsPerRequest[i].map(LogService.#toLogRetrievalResponse),
70
+ ]);
85
71
  }
86
72
 
87
- async #getPrivateLogByTag(siloedTag: SiloedTag): Promise<LogRetrievalResponse | null> {
88
- const anchorBlockHash = await this.anchorBlockHeader.hash();
89
- const allLogsPerTag = await getAllPrivateLogsByTags(this.aztecNode, [siloedTag], anchorBlockHash);
90
- const logsForTag = allLogsPerTag[0];
91
-
92
- if (logsForTag.length === 0) {
93
- return null;
94
- } else if (logsForTag.length > 1) {
95
- // TODO(#11627): handle this case
96
- throw new Error(
97
- `Got ${logsForTag.length} logs for tag ${siloedTag}. getPrivateLogByTag currently only supports a single log per tag`,
98
- );
73
+ async #fetchPublicLogs(
74
+ contractAddress: AztecAddress,
75
+ requests: LogRetrievalRequest[],
76
+ anchorBlockHash: BlockHash,
77
+ ): Promise<LogResult[][]> {
78
+ const indices = requests.flatMap((r, i) => (r.source !== LogSource.PRIVATE ? [i] : []));
79
+ if (indices.length === 0) {
80
+ return requests.map(() => []);
99
81
  }
100
82
 
101
- const scopedLog = logsForTag[0];
83
+ const resultsPerRequest: LogResult[][] = requests.map(() => []);
84
+ const groups = LogService.#groupByRange(indices.map(i => ({ index: i, request: requests[i] })));
102
85
 
103
- return new LogRetrievalResponse(
104
- scopedLog.logData.slice(1), // Skip the tag
105
- scopedLog.txHash,
106
- scopedLog.noteHashes,
107
- scopedLog.firstNullifier,
86
+ await Promise.all(
87
+ Array.from(groups.values()).map(async group => {
88
+ const tags = group.entries.map(e => e.request.tag);
89
+ const results = await getAllPublicLogsByTagsFromContract(
90
+ this.aztecNode,
91
+ contractAddress,
92
+ tags,
93
+ anchorBlockHash,
94
+ { fromBlock: group.fromBlock, toBlock: group.toBlock, includeEffects: true },
95
+ );
96
+ group.entries.forEach((entry, i) => {
97
+ resultsPerRequest[entry.index] = results[i];
98
+ });
99
+ }),
108
100
  );
109
- }
110
101
 
111
- public async syncTaggedLogs(
112
- contractAddress: AztecAddress,
113
- pendingTaggedLogArrayBaseSlot: Fr,
114
- scopes?: AztecAddress[],
115
- ) {
116
- this.log.verbose('Searching for tagged logs', { contract: contractAddress });
102
+ return resultsPerRequest;
103
+ }
117
104
 
118
- // We only load logs from block up to and including the anchor block number
119
- const anchorBlockNumber = this.anchorBlockHeader.getBlockNumber();
120
- const anchorBlockHash = await this.anchorBlockHeader.hash();
105
+ async #fetchPrivateLogs(requests: LogRetrievalRequest[], anchorBlockHash: BlockHash): Promise<LogResult[][]> {
106
+ const indices = requests.flatMap((r, i) => (r.source !== LogSource.PUBLIC ? [i] : []));
107
+ if (indices.length === 0) {
108
+ return requests.map(() => []);
109
+ }
121
110
 
122
- // Determine recipients: use scopes if provided, otherwise get all accounts
123
- const recipients = scopes && scopes.length > 0 ? scopes : await this.keyStore.getAccounts();
111
+ const resultsPerRequest: LogResult[][] = requests.map(() => []);
112
+ const groups = LogService.#groupByRange(indices.map(i => ({ index: i, request: requests[i] })));
124
113
 
125
- // For each recipient, fetch secrets, load logs, and store them.
126
- // We run these per-recipient tasks in parallel so that logs are loaded for all recipients concurrently.
127
114
  await Promise.all(
128
- recipients.map(async recipient => {
129
- // Get all secrets for this recipient (one per sender)
130
- const secrets = await this.#getSecretsForSenders(contractAddress, recipient);
131
-
132
- // Load logs for all sender-recipient pairs in parallel
133
- const logArrays = await Promise.all(
134
- secrets.map(secret =>
135
- loadPrivateLogsForSenderRecipientPair(
136
- secret,
137
- contractAddress,
138
- this.aztecNode,
139
- this.recipientTaggingStore,
140
- anchorBlockNumber,
141
- anchorBlockHash,
142
- this.jobId,
143
- ),
144
- ),
115
+ Array.from(groups.values()).map(async group => {
116
+ const siloedTags = await Promise.all(
117
+ group.entries.map(e => SiloedTag.computeFromTagAndApp(e.request.tag, e.request.contractAddress)),
145
118
  );
119
+ const results = await getAllPrivateLogsByTags(this.aztecNode, siloedTags, anchorBlockHash, {
120
+ fromBlock: group.fromBlock,
121
+ toBlock: group.toBlock,
122
+ includeEffects: true,
123
+ });
124
+ group.entries.forEach((entry, i) => {
125
+ resultsPerRequest[entry.index] = results[i];
126
+ });
127
+ }),
128
+ );
146
129
 
147
- // Flatten all logs from all secrets
148
- const allLogs = logArrays.flat();
130
+ return resultsPerRequest;
131
+ }
149
132
 
150
- // Store the logs for this recipient
151
- if (allLogs.length > 0) {
152
- await this.#storePendingTaggedLogs(contractAddress, pendingTaggedLogArrayBaseSlot, recipient, allLogs);
153
- }
154
- }),
133
+ /**
134
+ * Groups requests by their (fromBlock, toBlock) range so each distinct range becomes a single node call with
135
+ * the range pushed down into the query (no in-memory filter).
136
+ */
137
+ static #groupByRange(
138
+ entries: Array<{ index: number; request: LogRetrievalRequest }>,
139
+ ): Map<RangeKey, { fromBlock?: BlockNumber; toBlock?: BlockNumber; entries: typeof entries }> {
140
+ const groups = new Map<RangeKey, { fromBlock?: BlockNumber; toBlock?: BlockNumber; entries: typeof entries }>();
141
+ for (const entry of entries) {
142
+ const key = rangeKey(entry.request.fromBlock, entry.request.toBlock);
143
+ const existing = groups.get(key);
144
+ if (existing) {
145
+ existing.entries.push(entry);
146
+ } else {
147
+ groups.set(key, { fromBlock: entry.request.fromBlock, toBlock: entry.request.toBlock, entries: [entry] });
148
+ }
149
+ }
150
+ return groups;
151
+ }
152
+
153
+ static #toLogRetrievalResponse(log: LogResult): LogRetrievalResponse {
154
+ // includeEffects: true was used, so noteHashes and nullifiers are populated. Every tx has at least one nullifier
155
+ // (the first nullifier derived from the tx hash); empty here would indicate a buggy node.
156
+ const noteHashes = log.noteHashes!;
157
+ const nullifiers = log.nullifiers!;
158
+ if (nullifiers.length === 0) {
159
+ throw new Error(`Log for tx ${log.txHash} returned no nullifiers from the node`);
160
+ }
161
+ return new LogRetrievalResponse(
162
+ log.logData.slice(1), // Skip the tag
163
+ log.txHash,
164
+ noteHashes,
165
+ nullifiers[0],
155
166
  );
156
167
  }
157
168
 
158
- async #getSecretsForSenders(
169
+ public async fetchTaggedLogs(
159
170
  contractAddress: AztecAddress,
160
171
  recipient: AztecAddress,
161
- ): Promise<DirectionalAppTaggingSecret[]> {
162
- const recipientCompleteAddress = await this.#getCompleteAddress(recipient);
172
+ providedSecrets: AppTaggingSecret[],
173
+ ): Promise<PendingTaggedLog[]> {
174
+ this.log.verbose(`Fetching tagged logs for ${contractAddress.toString()}`);
175
+
176
+ const l2Tips = await this.l2TipsStore.getL2Tips();
177
+ // The secrets PXE derives or stores internally, plus any the app supplies explicitly for secrets PXE cannot
178
+ // enumerate itself (e.g. handshake-derived ones).
179
+ const secrets = [...(await this.#getSecretsForSenders(contractAddress, recipient)), ...providedSecrets];
180
+
181
+ const logs = await syncTaggedPrivateLogs(
182
+ secrets,
183
+ this.aztecNode,
184
+ this.recipientTaggingStore,
185
+ this.anchorBlockHeader,
186
+ l2Tips.finalized.block.number,
187
+ this.jobId,
188
+ );
189
+
190
+ return logs.map(log => {
191
+ const noteHashes = log.noteHashes!;
192
+ const nullifiers = log.nullifiers!;
193
+ if (nullifiers.length === 0) {
194
+ throw new Error(`Log for tx ${log.txHash} returned no nullifiers from the node`);
195
+ }
196
+ return new PendingTaggedLog(log.logData, log.txHash, noteHashes, nullifiers[0]);
197
+ });
198
+ }
199
+
200
+ async #getSecretsForSenders(contractAddress: AztecAddress, recipient: AztecAddress): Promise<AppTaggingSecret[]> {
201
+ const recipientCompleteAddress = await this.addressStore.getCompleteAddress(recipient);
202
+ if (!recipientCompleteAddress) {
203
+ return [];
204
+ }
163
205
  const recipientIvsk = await this.keyStore.getMasterIncomingViewingSecretKey(recipient);
164
206
 
165
- // We implicitly add all PXE accounts as senders, this helps us decrypt tags on notes that we send to ourselves
166
- // (recipient = us, sender = us)
207
+ // We implicitly add all PXE accounts as senders, this helps us find tagged logs with messages that are sent to a
208
+ // local account (recipient = us, sender = us)
167
209
  const allSenders = [...(await this.senderAddressBookStore.getSenders()), ...(await this.keyStore.getAccounts())];
168
210
 
169
211
  // We deduplicate the senders by adding them to a set and then converting the set back to an array
@@ -172,49 +214,24 @@ export class LogService {
172
214
  );
173
215
 
174
216
  return Promise.all(
175
- deduplicatedSenders.map(sender => {
176
- return DirectionalAppTaggingSecret.compute(
217
+ deduplicatedSenders.map(async sender => {
218
+ const secret = await AppTaggingSecret.computeUnconstrained(
177
219
  recipientCompleteAddress,
178
220
  recipientIvsk,
179
221
  sender,
180
222
  contractAddress,
181
223
  recipient,
182
224
  );
183
- }),
184
- );
185
- }
186
-
187
- #storePendingTaggedLogs(
188
- contractAddress: AztecAddress,
189
- capsuleArrayBaseSlot: Fr,
190
- recipient: AztecAddress,
191
- privateLogs: TxScopedL2Log[],
192
- ) {
193
- // Build all pending tagged logs from the scoped logs
194
- const pendingTaggedLogs = privateLogs.map(scopedLog => {
195
- const pendingTaggedLog = new PendingTaggedLog(
196
- scopedLog.logData,
197
- scopedLog.txHash,
198
- scopedLog.noteHashes,
199
- scopedLog.firstNullifier,
200
- recipient,
201
- );
202
-
203
- return pendingTaggedLog.toFields();
204
- });
205
225
 
206
- // TODO: This looks like it could belong more at the oracle interface level
207
- return this.capsuleStore.appendToCapsuleArray(contractAddress, capsuleArrayBaseSlot, pendingTaggedLogs, this.jobId);
208
- }
226
+ if (!secret) {
227
+ // Note that all senders originate from either the SenderAddressBookStore or the KeyStore.
228
+ throw new Error(
229
+ `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.`,
230
+ );
231
+ }
209
232
 
210
- async #getCompleteAddress(account: AztecAddress): Promise<CompleteAddress> {
211
- const completeAddress = await this.addressStore.getCompleteAddress(account);
212
- if (!completeAddress) {
213
- throw new Error(
214
- `No public key registered for address ${account}.
215
- Register it by calling pxe.addAccount(...).\nSee docs for context: https://docs.aztec.network/developers/resources/debugging/aztecnr-errors#simulation-error-no-public-key-registered-for-address-0x0-register-it-by-calling-pxeregisterrecipient-or-pxeregisteraccount`,
216
- );
217
- }
218
- return completeAddress;
233
+ return secret;
234
+ }),
235
+ );
219
236
  }
220
237
  }
@@ -0,0 +1,62 @@
1
+ import { uniqueBy } from '@aztec/foundation/collection';
2
+ import { Fr } from '@aztec/foundation/curves/bn254';
3
+ import type { AztecNode } from '@aztec/stdlib/interfaces/server';
4
+ import { MessageContext } from '@aztec/stdlib/logs';
5
+ import { type IndexedTxEffect, TxHash } from '@aztec/stdlib/tx';
6
+
7
+ /** Resolves transaction hashes into the context needed to process messages. */
8
+ export class MessageContextService {
9
+ constructor(private readonly aztecNode: AztecNode) {}
10
+
11
+ /**
12
+ * Resolves a list of tx hashes into their message contexts.
13
+ *
14
+ * For each tx hash, looks up the corresponding tx effect and extracts the note hashes and first nullifier needed to
15
+ * process messages that originated from that transaction. Returns `null` for tx hashes that are zero, not yet
16
+ * available, or in blocks beyond the anchor block.
17
+ */
18
+ async getMessageContextsByTxHash(txHashes: Fr[], anchorBlockNumber: number): Promise<(MessageContext | null)[]> {
19
+ const nonZeroTxHashes = txHashes.filter(h => !h.isZero()).map(h => TxHash.fromField(h));
20
+ const uniqueTxHashes = uniqueBy(nonZeroTxHashes, h => h.toString());
21
+ const fetched = await Promise.all(
22
+ uniqueTxHashes.map(h => this.aztecNode.getTxReceipt(h, { includeTxEffect: true })),
23
+ );
24
+ const txEffects = new Map(
25
+ uniqueTxHashes
26
+ .map((h, i): [string, IndexedTxEffect | undefined] => {
27
+ const receipt = fetched[i];
28
+ if (!receipt.isMined() || !receipt.txEffect) {
29
+ return [h.toString(), undefined];
30
+ }
31
+ return [
32
+ h.toString(),
33
+ {
34
+ data: receipt.txEffect,
35
+ l2BlockNumber: receipt.blockNumber,
36
+ l2BlockHash: receipt.blockHash,
37
+ txIndexInBlock: receipt.txIndexInBlock,
38
+ slotNumber: receipt.slotNumber,
39
+ },
40
+ ];
41
+ })
42
+ .filter((entry): entry is [string, IndexedTxEffect] => entry[1] !== undefined),
43
+ );
44
+
45
+ return txHashes.map(txHashField => {
46
+ const txHash = TxHash.fromField(txHashField);
47
+ const txEffect = txEffects.get(txHash.toString());
48
+ if (!txEffect || txEffect.l2BlockNumber > anchorBlockNumber) {
49
+ return null;
50
+ }
51
+
52
+ // Every tx has at least one nullifier (the first nullifier derived from the tx hash). Hitting this condition
53
+ // would mean a buggy node, but since we need to access data.nullifiers[0], the defensive check does no harm.
54
+ const data = txEffect.data;
55
+ if (data.nullifiers.length === 0) {
56
+ throw new Error(`Tx effect for ${txHash} has no nullifiers`);
57
+ }
58
+
59
+ return new MessageContext(data.txHash, data.noteHashes, data.nullifiers[0]);
60
+ });
61
+ }
62
+ }