@aztec/pxe 0.0.1-commit.f2ce05ee → 0.0.1-commit.f5d02921e
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.
- package/dest/access_scopes.d.ts +9 -0
- package/dest/access_scopes.d.ts.map +1 -0
- package/dest/access_scopes.js +6 -0
- package/dest/block_synchronizer/block_synchronizer.d.ts +1 -1
- package/dest/block_synchronizer/block_synchronizer.d.ts.map +1 -1
- package/dest/block_synchronizer/block_synchronizer.js +6 -0
- package/dest/config/index.d.ts +2 -2
- package/dest/config/index.d.ts.map +1 -1
- package/dest/config/index.js +1 -1
- package/dest/config/package_info.js +1 -1
- package/dest/contract_function_simulator/contract_function_simulator.d.ts +60 -30
- package/dest/contract_function_simulator/contract_function_simulator.d.ts.map +1 -1
- package/dest/contract_function_simulator/contract_function_simulator.js +199 -72
- package/dest/contract_function_simulator/execution_tagging_index_cache.d.ts +7 -7
- package/dest/contract_function_simulator/execution_tagging_index_cache.d.ts.map +1 -1
- package/dest/contract_function_simulator/execution_tagging_index_cache.js +19 -11
- package/dest/contract_function_simulator/index.d.ts +2 -1
- package/dest/contract_function_simulator/index.d.ts.map +1 -1
- package/dest/contract_function_simulator/index.js +1 -0
- package/dest/contract_function_simulator/noir-structs/event_validation_request.d.ts +3 -5
- package/dest/contract_function_simulator/noir-structs/event_validation_request.d.ts.map +1 -1
- package/dest/contract_function_simulator/noir-structs/event_validation_request.js +7 -9
- package/dest/contract_function_simulator/noir-structs/log_retrieval_response.d.ts +1 -1
- package/dest/contract_function_simulator/noir-structs/log_retrieval_response.d.ts.map +1 -1
- package/dest/contract_function_simulator/noir-structs/log_retrieval_response.js +1 -3
- package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts +3 -6
- package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts.map +1 -1
- package/dest/contract_function_simulator/noir-structs/note_validation_request.js +5 -10
- package/dest/contract_function_simulator/oracle/interfaces.d.ts +50 -45
- package/dest/contract_function_simulator/oracle/interfaces.d.ts.map +1 -1
- package/dest/contract_function_simulator/oracle/legacy_oracle_mappings.d.ts +9 -0
- package/dest/contract_function_simulator/oracle/legacy_oracle_mappings.d.ts.map +1 -0
- package/dest/contract_function_simulator/oracle/legacy_oracle_mappings.js +38 -0
- package/dest/contract_function_simulator/oracle/oracle.d.ts +64 -44
- package/dest/contract_function_simulator/oracle/oracle.d.ts.map +1 -1
- package/dest/contract_function_simulator/oracle/oracle.js +187 -97
- package/dest/contract_function_simulator/oracle/private_execution.js +5 -3
- package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts +54 -80
- package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts.map +1 -1
- package/dest/contract_function_simulator/oracle/private_execution_oracle.js +98 -85
- package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts +90 -52
- package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts.map +1 -1
- package/dest/contract_function_simulator/oracle/utility_execution_oracle.js +213 -116
- package/dest/contract_logging.d.ts +27 -0
- package/dest/contract_logging.d.ts.map +1 -0
- package/dest/contract_logging.js +38 -0
- package/dest/contract_sync/contract_sync_service.d.ts +10 -5
- package/dest/contract_sync/contract_sync_service.d.ts.map +1 -1
- package/dest/contract_sync/contract_sync_service.js +72 -30
- package/dest/contract_sync/helpers.d.ts +3 -2
- package/dest/contract_sync/helpers.d.ts.map +1 -1
- package/dest/contract_sync/helpers.js +9 -4
- package/dest/debug/pxe_debug_utils.d.ts +5 -4
- package/dest/debug/pxe_debug_utils.d.ts.map +1 -1
- package/dest/debug/pxe_debug_utils.js +4 -4
- package/dest/entrypoints/client/bundle/index.d.ts +4 -1
- package/dest/entrypoints/client/bundle/index.d.ts.map +1 -1
- package/dest/entrypoints/client/bundle/index.js +3 -0
- package/dest/entrypoints/client/bundle/utils.d.ts +1 -1
- package/dest/entrypoints/client/bundle/utils.d.ts.map +1 -1
- package/dest/entrypoints/client/bundle/utils.js +9 -1
- package/dest/entrypoints/client/lazy/index.d.ts +4 -1
- package/dest/entrypoints/client/lazy/index.d.ts.map +1 -1
- package/dest/entrypoints/client/lazy/index.js +3 -0
- package/dest/entrypoints/client/lazy/utils.d.ts +1 -1
- package/dest/entrypoints/client/lazy/utils.d.ts.map +1 -1
- package/dest/entrypoints/client/lazy/utils.js +9 -1
- package/dest/entrypoints/server/index.d.ts +3 -1
- package/dest/entrypoints/server/index.d.ts.map +1 -1
- package/dest/entrypoints/server/index.js +2 -0
- package/dest/entrypoints/server/utils.js +9 -1
- package/dest/events/event_service.d.ts +3 -2
- package/dest/events/event_service.d.ts.map +1 -1
- package/dest/events/event_service.js +16 -4
- package/dest/logs/log_service.d.ts +6 -6
- package/dest/logs/log_service.d.ts.map +1 -1
- package/dest/logs/log_service.js +31 -40
- package/dest/messages/message_context_service.d.ts +17 -0
- package/dest/messages/message_context_service.d.ts.map +1 -0
- package/dest/messages/message_context_service.js +36 -0
- package/dest/notes/note_service.d.ts +5 -4
- package/dest/notes/note_service.d.ts.map +1 -1
- package/dest/notes/note_service.js +17 -7
- package/dest/notes_filter.d.ts +25 -0
- package/dest/notes_filter.d.ts.map +1 -0
- package/dest/notes_filter.js +4 -0
- package/dest/oracle_version.d.ts +2 -2
- package/dest/oracle_version.js +3 -3
- package/dest/private_kernel/hints/compute_tx_expiration_timestamp.d.ts +4 -0
- package/dest/private_kernel/hints/compute_tx_expiration_timestamp.d.ts.map +1 -0
- package/dest/private_kernel/hints/{compute_tx_include_by_timestamp.js → compute_tx_expiration_timestamp.js} +12 -12
- package/dest/private_kernel/hints/index.d.ts +1 -1
- package/dest/private_kernel/hints/index.js +1 -1
- package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.d.ts +4 -3
- package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.d.ts.map +1 -1
- package/dest/private_kernel/hints/private_kernel_reset_private_inputs_builder.js +129 -68
- package/dest/private_kernel/hints/test_utils.d.ts +122 -0
- package/dest/private_kernel/hints/test_utils.d.ts.map +1 -0
- package/dest/private_kernel/hints/test_utils.js +203 -0
- package/dest/private_kernel/private_kernel_execution_prover.d.ts +1 -1
- package/dest/private_kernel/private_kernel_execution_prover.d.ts.map +1 -1
- package/dest/private_kernel/private_kernel_execution_prover.js +19 -11
- package/dest/private_kernel/private_kernel_oracle.d.ts +6 -2
- package/dest/private_kernel/private_kernel_oracle.d.ts.map +1 -1
- package/dest/private_kernel/private_kernel_oracle.js +7 -3
- package/dest/pxe.d.ts +63 -24
- package/dest/pxe.d.ts.map +1 -1
- package/dest/pxe.js +105 -74
- package/dest/storage/capsule_store/capsule_service.d.ts +22 -0
- package/dest/storage/capsule_store/capsule_service.d.ts.map +1 -0
- package/dest/storage/capsule_store/capsule_service.js +50 -0
- package/dest/storage/capsule_store/capsule_store.d.ts +9 -9
- package/dest/storage/capsule_store/capsule_store.d.ts.map +1 -1
- package/dest/storage/capsule_store/capsule_store.js +33 -28
- package/dest/storage/capsule_store/index.d.ts +2 -1
- package/dest/storage/capsule_store/index.d.ts.map +1 -1
- package/dest/storage/capsule_store/index.js +1 -0
- package/dest/storage/contract_store/contract_store.d.ts +42 -15
- package/dest/storage/contract_store/contract_store.d.ts.map +1 -1
- package/dest/storage/contract_store/contract_store.js +140 -64
- package/dest/storage/metadata.d.ts +1 -1
- package/dest/storage/metadata.js +1 -1
- package/dest/storage/note_store/note_store.d.ts +3 -3
- package/dest/storage/note_store/note_store.d.ts.map +1 -1
- package/dest/storage/note_store/note_store.js +3 -4
- package/dest/storage/tagging_store/recipient_tagging_store.d.ts +6 -6
- package/dest/storage/tagging_store/recipient_tagging_store.d.ts.map +1 -1
- package/dest/storage/tagging_store/sender_tagging_store.d.ts +29 -28
- package/dest/storage/tagging_store/sender_tagging_store.d.ts.map +1 -1
- package/dest/storage/tagging_store/sender_tagging_store.js +141 -115
- package/dest/tagging/get_all_logs_by_tags.d.ts +1 -1
- package/dest/tagging/get_all_logs_by_tags.d.ts.map +1 -1
- package/dest/tagging/get_all_logs_by_tags.js +17 -3
- package/dest/tagging/index.d.ts +3 -3
- package/dest/tagging/index.d.ts.map +1 -1
- package/dest/tagging/index.js +1 -1
- package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.d.ts +4 -5
- package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.d.ts.map +1 -1
- package/dest/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.js +7 -7
- package/dest/tagging/recipient_sync/utils/find_highest_indexes.js +2 -2
- package/dest/tagging/recipient_sync/utils/load_logs_for_range.d.ts +6 -7
- package/dest/tagging/recipient_sync/utils/load_logs_for_range.d.ts.map +1 -1
- package/dest/tagging/recipient_sync/utils/load_logs_for_range.js +12 -11
- package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts +4 -8
- package/dest/tagging/sender_sync/sync_sender_tagging_indexes.d.ts.map +1 -1
- package/dest/tagging/sender_sync/sync_sender_tagging_indexes.js +13 -7
- package/dest/tagging/sender_sync/utils/get_status_change_of_pending.d.ts +4 -3
- package/dest/tagging/sender_sync/utils/get_status_change_of_pending.d.ts.map +1 -1
- package/dest/tagging/sender_sync/utils/get_status_change_of_pending.js +20 -10
- package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts +5 -7
- package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.d.ts.map +1 -1
- package/dest/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.js +36 -24
- package/package.json +16 -16
- package/src/access_scopes.ts +9 -0
- package/src/block_synchronizer/block_synchronizer.ts +6 -0
- package/src/config/index.ts +1 -1
- package/src/config/package_info.ts +1 -1
- package/src/contract_function_simulator/contract_function_simulator.ts +358 -133
- package/src/contract_function_simulator/execution_tagging_index_cache.ts +19 -14
- package/src/contract_function_simulator/index.ts +1 -0
- package/src/contract_function_simulator/noir-structs/event_validation_request.ts +8 -8
- package/src/contract_function_simulator/noir-structs/log_retrieval_response.ts +1 -4
- package/src/contract_function_simulator/noir-structs/note_validation_request.ts +3 -9
- package/src/contract_function_simulator/oracle/interfaces.ts +63 -54
- package/src/contract_function_simulator/oracle/legacy_oracle_mappings.ts +98 -0
- package/src/contract_function_simulator/oracle/oracle.ts +223 -139
- package/src/contract_function_simulator/oracle/private_execution.ts +4 -4
- package/src/contract_function_simulator/oracle/private_execution_oracle.ts +122 -177
- package/src/contract_function_simulator/oracle/utility_execution_oracle.ts +375 -131
- package/src/contract_logging.ts +52 -0
- package/src/contract_sync/contract_sync_service.ts +108 -45
- package/src/contract_sync/helpers.ts +9 -3
- package/src/debug/pxe_debug_utils.ts +11 -9
- package/src/entrypoints/client/bundle/index.ts +3 -0
- package/src/entrypoints/client/bundle/utils.ts +9 -1
- package/src/entrypoints/client/lazy/index.ts +3 -0
- package/src/entrypoints/client/lazy/utils.ts +9 -1
- package/src/entrypoints/server/index.ts +2 -0
- package/src/entrypoints/server/utils.ts +7 -7
- package/src/events/event_service.ts +17 -4
- package/src/logs/log_service.ts +61 -63
- package/src/messages/message_context_service.ts +44 -0
- package/src/notes/note_service.ts +20 -8
- package/src/notes_filter.ts +26 -0
- package/src/oracle_version.ts +3 -3
- package/src/private_kernel/hints/{compute_tx_include_by_timestamp.ts → compute_tx_expiration_timestamp.ts} +13 -13
- package/src/private_kernel/hints/index.ts +1 -1
- package/src/private_kernel/hints/private_kernel_reset_private_inputs_builder.ts +164 -117
- package/src/private_kernel/hints/test_utils.ts +325 -0
- package/src/private_kernel/private_kernel_execution_prover.ts +19 -12
- package/src/private_kernel/private_kernel_oracle.ts +7 -7
- package/src/pxe.ts +190 -117
- package/src/storage/capsule_store/capsule_service.ts +91 -0
- package/src/storage/capsule_store/capsule_store.ts +34 -26
- package/src/storage/capsule_store/index.ts +1 -0
- package/src/storage/contract_store/contract_store.ts +170 -71
- package/src/storage/metadata.ts +1 -1
- package/src/storage/note_store/note_store.ts +8 -5
- package/src/storage/tagging_store/recipient_tagging_store.ts +9 -5
- package/src/storage/tagging_store/sender_tagging_store.ts +185 -138
- package/src/tagging/get_all_logs_by_tags.ts +28 -4
- package/src/tagging/index.ts +2 -2
- package/src/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.ts +7 -10
- package/src/tagging/recipient_sync/utils/find_highest_indexes.ts +2 -2
- package/src/tagging/recipient_sync/utils/load_logs_for_range.ts +10 -15
- package/src/tagging/sender_sync/sync_sender_tagging_indexes.ts +23 -10
- package/src/tagging/sender_sync/utils/get_status_change_of_pending.ts +26 -11
- package/src/tagging/sender_sync/utils/load_and_store_new_tagging_indexes.ts +27 -26
- package/dest/private_kernel/hints/compute_tx_include_by_timestamp.d.ts +0 -4
- package/dest/private_kernel/hints/compute_tx_include_by_timestamp.d.ts.map +0 -1
|
@@ -3,29 +3,34 @@ import type { BlockNumber } from '@aztec/foundation/branded-types';
|
|
|
3
3
|
import { Aes128 } from '@aztec/foundation/crypto/aes128';
|
|
4
4
|
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
5
5
|
import { Point } from '@aztec/foundation/curves/grumpkin';
|
|
6
|
-
import { LogLevels, type Logger,
|
|
6
|
+
import { LogLevels, type Logger, createLogger } from '@aztec/foundation/log';
|
|
7
7
|
import type { MembershipWitness } from '@aztec/foundation/trees';
|
|
8
8
|
import type { KeyStore } from '@aztec/key-store';
|
|
9
|
+
import { isProtocolContract } from '@aztec/protocol-contracts';
|
|
9
10
|
import type { AuthWitness } from '@aztec/stdlib/auth-witness';
|
|
10
11
|
import { AztecAddress } from '@aztec/stdlib/aztec-address';
|
|
11
12
|
import { BlockHash } from '@aztec/stdlib/block';
|
|
12
|
-
import type { CompleteAddress, ContractInstance } from '@aztec/stdlib/contract';
|
|
13
|
+
import type { CompleteAddress, ContractInstance, PartialAddress } from '@aztec/stdlib/contract';
|
|
13
14
|
import { siloNullifier } from '@aztec/stdlib/hash';
|
|
14
15
|
import type { AztecNode } from '@aztec/stdlib/interfaces/server';
|
|
15
16
|
import type { KeyValidationRequest } from '@aztec/stdlib/kernel';
|
|
16
|
-
import { computeAddressSecret } from '@aztec/stdlib/keys';
|
|
17
|
-
import {
|
|
17
|
+
import { type PublicKeys, computeAddressSecret } from '@aztec/stdlib/keys';
|
|
18
|
+
import { MessageContext, deriveAppSiloedSharedSecret } from '@aztec/stdlib/logs';
|
|
18
19
|
import { getNonNullifiedL1ToL2MessageWitness } from '@aztec/stdlib/messaging';
|
|
19
20
|
import type { NoteStatus } from '@aztec/stdlib/note';
|
|
20
21
|
import { MerkleTreeId, type NullifierMembershipWitness, PublicDataWitness } from '@aztec/stdlib/trees';
|
|
21
|
-
import type { BlockHeader, Capsule } from '@aztec/stdlib/tx';
|
|
22
|
+
import type { BlockHeader, Capsule, OffchainEffect } from '@aztec/stdlib/tx';
|
|
22
23
|
|
|
24
|
+
import type { AccessScopes } from '../../access_scopes.js';
|
|
25
|
+
import { createContractLogger, logContractMessage, stripAztecnrLogPrefix } from '../../contract_logging.js';
|
|
26
|
+
import type { ContractSyncService } from '../../contract_sync/contract_sync_service.js';
|
|
23
27
|
import { EventService } from '../../events/event_service.js';
|
|
24
28
|
import { LogService } from '../../logs/log_service.js';
|
|
29
|
+
import { MessageContextService } from '../../messages/message_context_service.js';
|
|
25
30
|
import { NoteService } from '../../notes/note_service.js';
|
|
26
31
|
import { ORACLE_VERSION } from '../../oracle_version.js';
|
|
27
32
|
import type { AddressStore } from '../../storage/address_store/address_store.js';
|
|
28
|
-
import type {
|
|
33
|
+
import type { CapsuleService } from '../../storage/capsule_store/capsule_service.js';
|
|
29
34
|
import type { ContractStore } from '../../storage/contract_store/contract_store.js';
|
|
30
35
|
import type { NoteStore } from '../../storage/note_store/note_store.js';
|
|
31
36
|
import type { PrivateEventStore } from '../../storage/private_event_store/private_event_store.js';
|
|
@@ -40,6 +45,29 @@ import { pickNotes } from '../pick_notes.js';
|
|
|
40
45
|
import type { IMiscOracle, IUtilityExecutionOracle, NoteData } from './interfaces.js';
|
|
41
46
|
import { MessageLoadOracleInputs } from './message_load_oracle_inputs.js';
|
|
42
47
|
|
|
48
|
+
/** Args for UtilityExecutionOracle constructor. */
|
|
49
|
+
export type UtilityExecutionOracleArgs = {
|
|
50
|
+
contractAddress: AztecAddress;
|
|
51
|
+
/** List of transient auth witnesses to be used during this simulation */
|
|
52
|
+
authWitnesses: AuthWitness[];
|
|
53
|
+
capsules: Capsule[]; // TODO(#12425): Rename to transientCapsules
|
|
54
|
+
anchorBlockHeader: BlockHeader;
|
|
55
|
+
contractStore: ContractStore;
|
|
56
|
+
noteStore: NoteStore;
|
|
57
|
+
keyStore: KeyStore;
|
|
58
|
+
addressStore: AddressStore;
|
|
59
|
+
aztecNode: AztecNode;
|
|
60
|
+
recipientTaggingStore: RecipientTaggingStore;
|
|
61
|
+
senderAddressBookStore: SenderAddressBookStore;
|
|
62
|
+
capsuleService: CapsuleService;
|
|
63
|
+
privateEventStore: PrivateEventStore;
|
|
64
|
+
messageContextService: MessageContextService;
|
|
65
|
+
contractSyncService: ContractSyncService;
|
|
66
|
+
jobId: string;
|
|
67
|
+
log?: ReturnType<typeof createLogger>;
|
|
68
|
+
scopes: AccessScopes;
|
|
69
|
+
};
|
|
70
|
+
|
|
43
71
|
/**
|
|
44
72
|
* The oracle for an execution of utility contract functions.
|
|
45
73
|
*/
|
|
@@ -48,38 +76,85 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
48
76
|
isUtility = true as const;
|
|
49
77
|
|
|
50
78
|
private contractLogger: Logger | undefined;
|
|
79
|
+
private aztecnrLogger: Logger | undefined;
|
|
80
|
+
private offchainEffects: OffchainEffect[] = [];
|
|
81
|
+
|
|
82
|
+
protected readonly contractAddress: AztecAddress;
|
|
83
|
+
protected readonly authWitnesses: AuthWitness[];
|
|
84
|
+
protected readonly capsules: Capsule[];
|
|
85
|
+
protected readonly anchorBlockHeader: BlockHeader;
|
|
86
|
+
protected readonly contractStore: ContractStore;
|
|
87
|
+
protected readonly noteStore: NoteStore;
|
|
88
|
+
protected readonly keyStore: KeyStore;
|
|
89
|
+
protected readonly addressStore: AddressStore;
|
|
90
|
+
protected readonly aztecNode: AztecNode;
|
|
91
|
+
protected readonly recipientTaggingStore: RecipientTaggingStore;
|
|
92
|
+
protected readonly senderAddressBookStore: SenderAddressBookStore;
|
|
93
|
+
protected readonly capsuleService: CapsuleService;
|
|
94
|
+
protected readonly privateEventStore: PrivateEventStore;
|
|
95
|
+
protected readonly messageContextService: MessageContextService;
|
|
96
|
+
protected readonly contractSyncService: ContractSyncService;
|
|
97
|
+
protected readonly jobId: string;
|
|
98
|
+
protected logger: ReturnType<typeof createLogger>;
|
|
99
|
+
protected readonly scopes: AccessScopes;
|
|
100
|
+
|
|
101
|
+
constructor(args: UtilityExecutionOracleArgs) {
|
|
102
|
+
this.contractAddress = args.contractAddress;
|
|
103
|
+
this.authWitnesses = args.authWitnesses;
|
|
104
|
+
this.capsules = args.capsules;
|
|
105
|
+
this.anchorBlockHeader = args.anchorBlockHeader;
|
|
106
|
+
this.contractStore = args.contractStore;
|
|
107
|
+
this.noteStore = args.noteStore;
|
|
108
|
+
this.keyStore = args.keyStore;
|
|
109
|
+
this.addressStore = args.addressStore;
|
|
110
|
+
this.aztecNode = args.aztecNode;
|
|
111
|
+
this.recipientTaggingStore = args.recipientTaggingStore;
|
|
112
|
+
this.senderAddressBookStore = args.senderAddressBookStore;
|
|
113
|
+
this.capsuleService = args.capsuleService;
|
|
114
|
+
this.privateEventStore = args.privateEventStore;
|
|
115
|
+
this.messageContextService = args.messageContextService;
|
|
116
|
+
this.contractSyncService = args.contractSyncService;
|
|
117
|
+
this.jobId = args.jobId;
|
|
118
|
+
this.logger = args.log ?? createLogger('simulator:client_view_context');
|
|
119
|
+
this.scopes = args.scopes;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
public assertCompatibleOracleVersion(version: number): void {
|
|
123
|
+
// TODO(F-416): Remove this hack on v5 when protocol contracts are redeployed.
|
|
124
|
+
// Protocol contracts/canonical contracts shipped with committed bytecode that cannot be changed. Assert they use
|
|
125
|
+
// the expected pinned version or the current one. We want to allow for both the pinned and the current versions
|
|
126
|
+
// because we want this code to work with both the pinned and unpinned version since some branches do not have the
|
|
127
|
+
// pinned contracts (like e.g. next)
|
|
128
|
+
const LEGACY_ORACLE_VERSION = 12;
|
|
129
|
+
if (isProtocolContract(this.contractAddress)) {
|
|
130
|
+
if (version !== LEGACY_ORACLE_VERSION && version !== ORACLE_VERSION) {
|
|
131
|
+
const hint =
|
|
132
|
+
version > ORACLE_VERSION
|
|
133
|
+
? 'The contract was compiled with a newer version of Aztec.nr than your private environment supports. Upgrade your private environment to a compatible version.'
|
|
134
|
+
: '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.';
|
|
135
|
+
throw new Error(
|
|
136
|
+
`Incompatible private environment version: ${hint} See https://docs.aztec.network/errors/8 (expected oracle version ${LEGACY_ORACLE_VERSION} or ${ORACLE_VERSION}, got ${version})`,
|
|
137
|
+
);
|
|
138
|
+
}
|
|
139
|
+
return;
|
|
140
|
+
}
|
|
51
141
|
|
|
52
|
-
constructor(
|
|
53
|
-
protected readonly contractAddress: AztecAddress,
|
|
54
|
-
/** List of transient auth witnesses to be used during this simulation */
|
|
55
|
-
protected readonly authWitnesses: AuthWitness[],
|
|
56
|
-
protected readonly capsules: Capsule[], // TODO(#12425): Rename to transientCapsules
|
|
57
|
-
protected readonly anchorBlockHeader: BlockHeader,
|
|
58
|
-
protected readonly contractStore: ContractStore,
|
|
59
|
-
protected readonly noteStore: NoteStore,
|
|
60
|
-
protected readonly keyStore: KeyStore,
|
|
61
|
-
protected readonly addressStore: AddressStore,
|
|
62
|
-
protected readonly aztecNode: AztecNode,
|
|
63
|
-
protected readonly recipientTaggingStore: RecipientTaggingStore,
|
|
64
|
-
protected readonly senderAddressBookStore: SenderAddressBookStore,
|
|
65
|
-
protected readonly capsuleStore: CapsuleStore,
|
|
66
|
-
protected readonly privateEventStore: PrivateEventStore,
|
|
67
|
-
protected readonly jobId: string,
|
|
68
|
-
protected log = createLogger('simulator:client_view_context'),
|
|
69
|
-
protected readonly scopes?: AztecAddress[],
|
|
70
|
-
) {}
|
|
71
|
-
|
|
72
|
-
public utilityAssertCompatibleOracleVersion(version: number): void {
|
|
73
142
|
if (version !== ORACLE_VERSION) {
|
|
74
|
-
|
|
143
|
+
const hint =
|
|
144
|
+
version > ORACLE_VERSION
|
|
145
|
+
? 'The contract was compiled with a newer version of Aztec.nr than your private environment supports. Upgrade your private environment to a compatible version.'
|
|
146
|
+
: '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.';
|
|
147
|
+
throw new Error(
|
|
148
|
+
`Incompatible private environment version: ${hint} See https://docs.aztec.network/errors/8 (expected oracle version ${ORACLE_VERSION}, got ${version})`,
|
|
149
|
+
);
|
|
75
150
|
}
|
|
76
151
|
}
|
|
77
152
|
|
|
78
|
-
public
|
|
153
|
+
public getRandomField(): Fr {
|
|
79
154
|
return Fr.random();
|
|
80
155
|
}
|
|
81
156
|
|
|
82
|
-
public
|
|
157
|
+
public getUtilityContext(): UtilityContext {
|
|
83
158
|
return new UtilityContext(this.anchorBlockHeader, this.contractAddress);
|
|
84
159
|
}
|
|
85
160
|
|
|
@@ -90,12 +165,17 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
90
165
|
* @throws If the keys are not registered in the key store.
|
|
91
166
|
* @throws If scopes are defined and the account is not in the scopes.
|
|
92
167
|
*/
|
|
93
|
-
public async
|
|
94
|
-
// If scopes are defined, check that the key belongs to an account in the scopes
|
|
95
|
-
if (this.scopes && this.scopes.length > 0) {
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
168
|
+
public async getKeyValidationRequest(pkMHash: Fr): Promise<KeyValidationRequest> {
|
|
169
|
+
// If scopes are defined, check that the key belongs to an account in the scopes.
|
|
170
|
+
if (this.scopes !== 'ALL_SCOPES' && this.scopes.length > 0) {
|
|
171
|
+
let hasAccess = false;
|
|
172
|
+
for (let i = 0; i < this.scopes.length && !hasAccess; i++) {
|
|
173
|
+
if (await this.keyStore.accountHasKey(this.scopes[i], pkMHash)) {
|
|
174
|
+
hasAccess = true;
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
if (!hasAccess) {
|
|
178
|
+
throw new Error(`Key validation request denied: no scoped account has a key with hash ${pkMHash.toString()}.`);
|
|
99
179
|
}
|
|
100
180
|
}
|
|
101
181
|
return this.keyStore.getKeyValidationRequest(pkMHash, this.contractAddress);
|
|
@@ -103,16 +183,18 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
103
183
|
|
|
104
184
|
/**
|
|
105
185
|
* Fetches the index and sibling path of a leaf at a given block from the note hash tree.
|
|
106
|
-
* @param
|
|
107
|
-
* witness.
|
|
186
|
+
* @param blockHash - The hash of a block that contains the note hash tree root in which to find the
|
|
187
|
+
* membership witness.
|
|
108
188
|
* @param noteHash - The note hash to find in the note hash tree.
|
|
109
189
|
* @returns The membership witness containing the leaf index and sibling path
|
|
110
190
|
*/
|
|
111
|
-
public
|
|
112
|
-
|
|
191
|
+
public getNoteHashMembershipWitness(
|
|
192
|
+
blockHash: BlockHash,
|
|
113
193
|
noteHash: Fr,
|
|
114
194
|
): Promise<MembershipWitness<typeof NOTE_HASH_TREE_HEIGHT> | undefined> {
|
|
115
|
-
return this
|
|
195
|
+
return this.#queryWithBlockHashNotAfterAnchor(blockHash, () =>
|
|
196
|
+
this.aztecNode.getNoteHashMembershipWitness(blockHash, noteHash),
|
|
197
|
+
);
|
|
116
198
|
}
|
|
117
199
|
|
|
118
200
|
/**
|
|
@@ -121,16 +203,21 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
121
203
|
* Block hashes are the leaves of the archive tree. Each time a new block is added to the chain,
|
|
122
204
|
* its block hash is appended as a new leaf to the archive tree.
|
|
123
205
|
*
|
|
124
|
-
* @param
|
|
206
|
+
* @param referenceBlockHash - The hash of a block that contains the archive tree root in which to find the membership
|
|
125
207
|
* witness.
|
|
126
208
|
* @param blockHash - The block hash to find in the archive tree.
|
|
127
209
|
* @returns The membership witness containing the leaf index and sibling path
|
|
128
210
|
*/
|
|
129
|
-
public
|
|
130
|
-
|
|
211
|
+
public getBlockHashMembershipWitness(
|
|
212
|
+
referenceBlockHash: BlockHash,
|
|
131
213
|
blockHash: BlockHash,
|
|
132
214
|
): Promise<MembershipWitness<typeof ARCHIVE_HEIGHT> | undefined> {
|
|
133
|
-
|
|
215
|
+
// Note that we validate that the reference block hash is at or before the anchor block - we don't test the block
|
|
216
|
+
// hash at all. If the block hash did not exist by the reference block hash, then the node will not return the
|
|
217
|
+
// membership witness as there is none.
|
|
218
|
+
return this.#queryWithBlockHashNotAfterAnchor(referenceBlockHash, () =>
|
|
219
|
+
this.aztecNode.getBlockHashMembershipWitness(referenceBlockHash, blockHash),
|
|
220
|
+
);
|
|
134
221
|
}
|
|
135
222
|
|
|
136
223
|
/**
|
|
@@ -139,11 +226,13 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
139
226
|
* @param nullifier - Nullifier we try to find witness for.
|
|
140
227
|
* @returns The nullifier membership witness (if found).
|
|
141
228
|
*/
|
|
142
|
-
public
|
|
229
|
+
public getNullifierMembershipWitness(
|
|
143
230
|
blockHash: BlockHash,
|
|
144
231
|
nullifier: Fr,
|
|
145
232
|
): Promise<NullifierMembershipWitness | undefined> {
|
|
146
|
-
return this
|
|
233
|
+
return this.#queryWithBlockHashNotAfterAnchor(blockHash, () =>
|
|
234
|
+
this.aztecNode.getNullifierMembershipWitness(blockHash, nullifier),
|
|
235
|
+
);
|
|
147
236
|
}
|
|
148
237
|
|
|
149
238
|
/**
|
|
@@ -155,11 +244,13 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
155
244
|
* list structure" of leaves and proving that a lower nullifier is pointing to a bigger next value than the nullifier
|
|
156
245
|
* we are trying to prove non-inclusion for.
|
|
157
246
|
*/
|
|
158
|
-
public
|
|
247
|
+
public getLowNullifierMembershipWitness(
|
|
159
248
|
blockHash: BlockHash,
|
|
160
249
|
nullifier: Fr,
|
|
161
250
|
): Promise<NullifierMembershipWitness | undefined> {
|
|
162
|
-
return this
|
|
251
|
+
return this.#queryWithBlockHashNotAfterAnchor(blockHash, () =>
|
|
252
|
+
this.aztecNode.getLowNullifierMembershipWitness(blockHash, nullifier),
|
|
253
|
+
);
|
|
163
254
|
}
|
|
164
255
|
|
|
165
256
|
/**
|
|
@@ -168,8 +259,10 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
168
259
|
* @param leafSlot - The slot of the public data tree to get the witness for.
|
|
169
260
|
* @returns - The witness
|
|
170
261
|
*/
|
|
171
|
-
public
|
|
172
|
-
return this
|
|
262
|
+
public getPublicDataWitness(blockHash: BlockHash, leafSlot: Fr): Promise<PublicDataWitness | undefined> {
|
|
263
|
+
return this.#queryWithBlockHashNotAfterAnchor(blockHash, () =>
|
|
264
|
+
this.aztecNode.getPublicDataWitness(blockHash, leafSlot),
|
|
265
|
+
);
|
|
173
266
|
}
|
|
174
267
|
|
|
175
268
|
/**
|
|
@@ -177,7 +270,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
177
270
|
* @param blockNumber - The number of a block of which to get the block header.
|
|
178
271
|
* @returns Block extracted from a block with block number `blockNumber`.
|
|
179
272
|
*/
|
|
180
|
-
public async
|
|
273
|
+
public async getBlockHeader(blockNumber: BlockNumber): Promise<BlockHeader | undefined> {
|
|
181
274
|
const anchorBlockNumber = this.anchorBlockHeader.getBlockNumber();
|
|
182
275
|
if (blockNumber > anchorBlockNumber) {
|
|
183
276
|
throw new Error(`Block number ${blockNumber} is higher than current block ${anchorBlockNumber}`);
|
|
@@ -188,12 +281,18 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
188
281
|
}
|
|
189
282
|
|
|
190
283
|
/**
|
|
191
|
-
* Retrieve the
|
|
284
|
+
* Retrieve the public keys and partial address associated to a given address.
|
|
192
285
|
* @param account - The account address.
|
|
193
|
-
* @returns
|
|
286
|
+
* @returns The public keys and partial address, or `undefined` if the account is not registered.
|
|
194
287
|
*/
|
|
195
|
-
public
|
|
196
|
-
|
|
288
|
+
public async getPublicKeysAndPartialAddress(
|
|
289
|
+
account: AztecAddress,
|
|
290
|
+
): Promise<{ publicKeys: PublicKeys; partialAddress: PartialAddress } | undefined> {
|
|
291
|
+
const completeAddress = await this.addressStore.getCompleteAddress(account);
|
|
292
|
+
if (!completeAddress) {
|
|
293
|
+
return undefined;
|
|
294
|
+
}
|
|
295
|
+
return { publicKeys: completeAddress.publicKeys, partialAddress: completeAddress.partialAddress };
|
|
197
296
|
}
|
|
198
297
|
|
|
199
298
|
protected async getCompleteAddressOrFail(account: AztecAddress): Promise<CompleteAddress> {
|
|
@@ -212,11 +311,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
212
311
|
* @param address - Address.
|
|
213
312
|
* @returns A contract instance.
|
|
214
313
|
*/
|
|
215
|
-
public
|
|
216
|
-
return this.getContractInstance(address);
|
|
217
|
-
}
|
|
218
|
-
|
|
219
|
-
protected async getContractInstance(address: AztecAddress): Promise<ContractInstance> {
|
|
314
|
+
public async getContractInstance(address: AztecAddress): Promise<ContractInstance> {
|
|
220
315
|
const instance = await this.contractStore.getContractInstance(address);
|
|
221
316
|
if (!instance) {
|
|
222
317
|
throw new Error(`No contract instance found for address ${address.toString()}`);
|
|
@@ -230,7 +325,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
230
325
|
* @param messageHash - Hash of the message to authenticate.
|
|
231
326
|
* @returns Authentication witness for the requested message hash.
|
|
232
327
|
*/
|
|
233
|
-
public
|
|
328
|
+
public getAuthWitness(messageHash: Fr): Promise<Fr[] | undefined> {
|
|
234
329
|
return Promise.resolve(this.authWitnesses.find(w => w.requestHash.equals(messageHash))?.witness);
|
|
235
330
|
}
|
|
236
331
|
|
|
@@ -256,7 +351,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
256
351
|
* @param status - The status of notes to fetch.
|
|
257
352
|
* @returns Array of note data.
|
|
258
353
|
*/
|
|
259
|
-
public async
|
|
354
|
+
public async getNotes(
|
|
260
355
|
owner: AztecAddress | undefined,
|
|
261
356
|
storageSlot: Fr,
|
|
262
357
|
numSelects: number,
|
|
@@ -296,7 +391,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
296
391
|
* @param innerNullifier - The inner nullifier.
|
|
297
392
|
* @returns A boolean indicating whether the nullifier exists in the tree or not.
|
|
298
393
|
*/
|
|
299
|
-
public async
|
|
394
|
+
public async doesNullifierExist(innerNullifier: Fr) {
|
|
300
395
|
const [nullifier, anchorBlockHash] = await Promise.all([
|
|
301
396
|
siloNullifier(this.contractAddress, innerNullifier!),
|
|
302
397
|
this.anchorBlockHeader.hash(),
|
|
@@ -308,19 +403,20 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
308
403
|
}
|
|
309
404
|
|
|
310
405
|
/**
|
|
311
|
-
*
|
|
406
|
+
* Returns the membership witness of an un-nullified L1 to L2 message.
|
|
312
407
|
* @param contractAddress - Address of a contract by which the message was emitted.
|
|
313
408
|
* @param messageHash - Hash of the message.
|
|
314
409
|
* @param secret - Secret used to compute a nullifier.
|
|
315
410
|
* @dev Contract address and secret are only used to compute the nullifier to get non-nullified messages
|
|
316
411
|
* @returns The l1 to l2 membership witness (index of message in the tree and sibling path).
|
|
317
412
|
*/
|
|
318
|
-
public async
|
|
413
|
+
public async getL1ToL2MembershipWitness(contractAddress: AztecAddress, messageHash: Fr, secret: Fr) {
|
|
319
414
|
const [messageIndex, siblingPath] = await getNonNullifiedL1ToL2MessageWitness(
|
|
320
415
|
this.aztecNode,
|
|
321
416
|
contractAddress,
|
|
322
417
|
messageHash,
|
|
323
418
|
secret,
|
|
419
|
+
await this.anchorBlockHeader.hash(),
|
|
324
420
|
);
|
|
325
421
|
|
|
326
422
|
return new MessageLoadOracleInputs(messageIndex, siblingPath);
|
|
@@ -333,65 +429,88 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
333
429
|
* @param startStorageSlot - The starting storage slot.
|
|
334
430
|
* @param numberOfElements - Number of elements to read from the starting storage slot.
|
|
335
431
|
*/
|
|
336
|
-
public
|
|
432
|
+
public getFromPublicStorage(
|
|
337
433
|
blockHash: BlockHash,
|
|
338
434
|
contractAddress: AztecAddress,
|
|
339
435
|
startStorageSlot: Fr,
|
|
340
436
|
numberOfElements: number,
|
|
341
437
|
) {
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
438
|
+
return this.#queryWithBlockHashNotAfterAnchor(blockHash, async () => {
|
|
439
|
+
const slots = Array(numberOfElements)
|
|
440
|
+
.fill(0)
|
|
441
|
+
.map((_, i) => new Fr(startStorageSlot.value + BigInt(i)));
|
|
345
442
|
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
443
|
+
const values = await Promise.all(
|
|
444
|
+
slots.map(storageSlot => this.aztecNode.getPublicStorageAt(blockHash, contractAddress, storageSlot)),
|
|
445
|
+
);
|
|
349
446
|
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
|
|
447
|
+
this.logger.debug(
|
|
448
|
+
`Oracle storage read: slots=[${slots.map(slot => slot.toString()).join(', ')}] address=${contractAddress.toString()} values=[${values.join(', ')}]`,
|
|
449
|
+
);
|
|
353
450
|
|
|
354
|
-
|
|
451
|
+
return values;
|
|
452
|
+
});
|
|
355
453
|
}
|
|
356
454
|
|
|
357
455
|
/**
|
|
358
|
-
* Returns a per-contract logger whose output is prefixed with `
|
|
456
|
+
* Returns a per-contract logger whose output is prefixed with `contract:<name>(<addrAbbrev>)`.
|
|
359
457
|
*/
|
|
360
458
|
async #getContractLogger(): Promise<Logger> {
|
|
361
459
|
if (!this.contractLogger) {
|
|
362
|
-
const addrAbbrev = this.contractAddress.toString().slice(0, 10);
|
|
363
|
-
const name = await this.contractStore.getDebugContractName(this.contractAddress);
|
|
364
|
-
const module = name ? `contract_log::${name}(${addrAbbrev})` : `contract_log::${addrAbbrev}`;
|
|
365
460
|
// Purpose of instanceId is to distinguish logs from different instances of the same component. It makes sense
|
|
366
461
|
// to re-use jobId as instanceId here as executions of different PXE jobs are isolated.
|
|
367
|
-
this.contractLogger =
|
|
462
|
+
this.contractLogger = await createContractLogger(
|
|
463
|
+
this.contractAddress,
|
|
464
|
+
addr => this.contractStore.getDebugContractName(addr),
|
|
465
|
+
'user',
|
|
466
|
+
{ instanceId: this.jobId },
|
|
467
|
+
);
|
|
368
468
|
}
|
|
369
469
|
return this.contractLogger;
|
|
370
470
|
}
|
|
371
471
|
|
|
372
|
-
|
|
472
|
+
/**
|
|
473
|
+
* Returns a per-contract logger whose output is prefixed with `aztecnr:<name>(<addrAbbrev>)`.
|
|
474
|
+
*/
|
|
475
|
+
async #getAztecnrLogger(): Promise<Logger> {
|
|
476
|
+
if (!this.aztecnrLogger) {
|
|
477
|
+
// Purpose of instanceId is to distinguish logs from different instances of the same component. It makes sense
|
|
478
|
+
// to re-use jobId as instanceId here as executions of different PXE jobs are isolated.
|
|
479
|
+
this.aztecnrLogger = await createContractLogger(
|
|
480
|
+
this.contractAddress,
|
|
481
|
+
addr => this.contractStore.getDebugContractName(addr),
|
|
482
|
+
'aztecnr',
|
|
483
|
+
{ instanceId: this.jobId },
|
|
484
|
+
);
|
|
485
|
+
}
|
|
486
|
+
return this.aztecnrLogger;
|
|
487
|
+
}
|
|
488
|
+
|
|
489
|
+
public async log(level: number, message: string, fields: Fr[]): Promise<void> {
|
|
373
490
|
if (!LogLevels[level]) {
|
|
374
|
-
throw new Error(`Invalid
|
|
491
|
+
throw new Error(`Invalid log level: ${level}`);
|
|
375
492
|
}
|
|
376
|
-
|
|
377
|
-
const
|
|
378
|
-
|
|
493
|
+
|
|
494
|
+
const { kind, message: strippedMessage } = stripAztecnrLogPrefix(message);
|
|
495
|
+
|
|
496
|
+
const logger = kind == 'aztecnr' ? await this.#getAztecnrLogger() : await this.#getContractLogger();
|
|
497
|
+
logContractMessage(logger, LogLevels[level], strippedMessage, fields);
|
|
379
498
|
}
|
|
380
499
|
|
|
381
|
-
public async
|
|
500
|
+
public async getPendingTaggedLogs(pendingTaggedLogArrayBaseSlot: Fr, scope: AztecAddress) {
|
|
382
501
|
const logService = new LogService(
|
|
383
502
|
this.aztecNode,
|
|
384
503
|
this.anchorBlockHeader,
|
|
385
504
|
this.keyStore,
|
|
386
|
-
this.
|
|
505
|
+
this.capsuleService,
|
|
387
506
|
this.recipientTaggingStore,
|
|
388
507
|
this.senderAddressBookStore,
|
|
389
508
|
this.addressStore,
|
|
390
509
|
this.jobId,
|
|
391
|
-
this.
|
|
510
|
+
this.logger.getBindings(),
|
|
392
511
|
);
|
|
393
512
|
|
|
394
|
-
await logService.fetchTaggedLogs(this.contractAddress, pendingTaggedLogArrayBaseSlot,
|
|
513
|
+
await logService.fetchTaggedLogs(this.contractAddress, pendingTaggedLogArrayBaseSlot, scope);
|
|
395
514
|
}
|
|
396
515
|
|
|
397
516
|
/**
|
|
@@ -404,10 +523,13 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
404
523
|
* @param noteValidationRequestsArrayBaseSlot - The base slot of capsule array containing note validation requests.
|
|
405
524
|
* @param eventValidationRequestsArrayBaseSlot - The base slot of capsule array containing event validation requests.
|
|
406
525
|
*/
|
|
407
|
-
public async
|
|
526
|
+
public async validateAndStoreEnqueuedNotesAndEvents(
|
|
408
527
|
contractAddress: AztecAddress,
|
|
409
528
|
noteValidationRequestsArrayBaseSlot: Fr,
|
|
410
529
|
eventValidationRequestsArrayBaseSlot: Fr,
|
|
530
|
+
maxNotePackedLen: number,
|
|
531
|
+
maxEventSerializedLen: number,
|
|
532
|
+
scope: AztecAddress,
|
|
411
533
|
) {
|
|
412
534
|
// TODO(#10727): allow other contracts to store notes
|
|
413
535
|
if (!this.contractAddress.equals(contractAddress)) {
|
|
@@ -417,12 +539,22 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
417
539
|
// We read all note and event validation requests and process them all concurrently. This makes the process much
|
|
418
540
|
// faster as we don't need to wait for the network round-trip.
|
|
419
541
|
const noteValidationRequests = (
|
|
420
|
-
await this.
|
|
421
|
-
|
|
542
|
+
await this.capsuleService.readCapsuleArray(
|
|
543
|
+
contractAddress,
|
|
544
|
+
noteValidationRequestsArrayBaseSlot,
|
|
545
|
+
this.jobId,
|
|
546
|
+
scope,
|
|
547
|
+
)
|
|
548
|
+
).map(fields => NoteValidationRequest.fromFields(fields, maxNotePackedLen));
|
|
422
549
|
|
|
423
550
|
const eventValidationRequests = (
|
|
424
|
-
await this.
|
|
425
|
-
|
|
551
|
+
await this.capsuleService.readCapsuleArray(
|
|
552
|
+
contractAddress,
|
|
553
|
+
eventValidationRequestsArrayBaseSlot,
|
|
554
|
+
this.jobId,
|
|
555
|
+
scope,
|
|
556
|
+
)
|
|
557
|
+
).map(fields => EventValidationRequest.fromFields(fields, maxEventSerializedLen));
|
|
426
558
|
|
|
427
559
|
const noteService = new NoteService(this.noteStore, this.aztecNode, this.anchorBlockHeader, this.jobId);
|
|
428
560
|
const noteStorePromises = noteValidationRequests.map(request =>
|
|
@@ -436,7 +568,7 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
436
568
|
request.noteHash,
|
|
437
569
|
request.nullifier,
|
|
438
570
|
request.txHash,
|
|
439
|
-
|
|
571
|
+
scope,
|
|
440
572
|
),
|
|
441
573
|
);
|
|
442
574
|
|
|
@@ -449,21 +581,34 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
449
581
|
request.serializedEvent,
|
|
450
582
|
request.eventCommitment,
|
|
451
583
|
request.txHash,
|
|
452
|
-
|
|
584
|
+
scope,
|
|
453
585
|
),
|
|
454
586
|
);
|
|
455
587
|
|
|
456
588
|
await Promise.all([...noteStorePromises, ...eventStorePromises]);
|
|
457
589
|
|
|
458
590
|
// Requests are cleared once we're done.
|
|
459
|
-
await this.
|
|
460
|
-
|
|
591
|
+
await this.capsuleService.setCapsuleArray(
|
|
592
|
+
contractAddress,
|
|
593
|
+
noteValidationRequestsArrayBaseSlot,
|
|
594
|
+
[],
|
|
595
|
+
this.jobId,
|
|
596
|
+
scope,
|
|
597
|
+
);
|
|
598
|
+
await this.capsuleService.setCapsuleArray(
|
|
599
|
+
contractAddress,
|
|
600
|
+
eventValidationRequestsArrayBaseSlot,
|
|
601
|
+
[],
|
|
602
|
+
this.jobId,
|
|
603
|
+
scope,
|
|
604
|
+
);
|
|
461
605
|
}
|
|
462
606
|
|
|
463
|
-
public async
|
|
607
|
+
public async getLogsByTag(
|
|
464
608
|
contractAddress: AztecAddress,
|
|
465
609
|
logRetrievalRequestsArrayBaseSlot: Fr,
|
|
466
610
|
logRetrievalResponsesArrayBaseSlot: Fr,
|
|
611
|
+
scope: AztecAddress,
|
|
467
612
|
) {
|
|
468
613
|
// TODO(#10727): allow other contracts to process partial notes
|
|
469
614
|
if (!this.contractAddress.equals(contractAddress)) {
|
|
@@ -473,101 +618,200 @@ export class UtilityExecutionOracle implements IMiscOracle, IUtilityExecutionOra
|
|
|
473
618
|
// We read all log retrieval requests and process them all concurrently. This makes the process much faster as we
|
|
474
619
|
// don't need to wait for the network round-trip.
|
|
475
620
|
const logRetrievalRequests = (
|
|
476
|
-
await this.
|
|
621
|
+
await this.capsuleService.readCapsuleArray(contractAddress, logRetrievalRequestsArrayBaseSlot, this.jobId, scope)
|
|
477
622
|
).map(LogRetrievalRequest.fromFields);
|
|
478
623
|
|
|
479
624
|
const logService = new LogService(
|
|
480
625
|
this.aztecNode,
|
|
481
626
|
this.anchorBlockHeader,
|
|
482
627
|
this.keyStore,
|
|
483
|
-
this.
|
|
628
|
+
this.capsuleService,
|
|
484
629
|
this.recipientTaggingStore,
|
|
485
630
|
this.senderAddressBookStore,
|
|
486
631
|
this.addressStore,
|
|
487
632
|
this.jobId,
|
|
488
|
-
this.
|
|
633
|
+
this.logger.getBindings(),
|
|
489
634
|
);
|
|
490
635
|
|
|
491
|
-
const maybeLogRetrievalResponses = await logService.
|
|
636
|
+
const maybeLogRetrievalResponses = await logService.fetchLogsByTag(contractAddress, logRetrievalRequests);
|
|
492
637
|
|
|
493
638
|
// Requests are cleared once we're done.
|
|
494
|
-
await this.
|
|
639
|
+
await this.capsuleService.setCapsuleArray(
|
|
640
|
+
contractAddress,
|
|
641
|
+
logRetrievalRequestsArrayBaseSlot,
|
|
642
|
+
[],
|
|
643
|
+
this.jobId,
|
|
644
|
+
scope,
|
|
645
|
+
);
|
|
495
646
|
|
|
496
647
|
// The responses are stored as Option<LogRetrievalResponse> in a second CapsuleArray.
|
|
497
|
-
await this.
|
|
648
|
+
await this.capsuleService.setCapsuleArray(
|
|
498
649
|
contractAddress,
|
|
499
650
|
logRetrievalResponsesArrayBaseSlot,
|
|
500
651
|
maybeLogRetrievalResponses.map(LogRetrievalResponse.toSerializedOption),
|
|
501
652
|
this.jobId,
|
|
653
|
+
scope,
|
|
502
654
|
);
|
|
503
655
|
}
|
|
504
656
|
|
|
505
|
-
public
|
|
657
|
+
public async getMessageContextsByTxHash(
|
|
658
|
+
contractAddress: AztecAddress,
|
|
659
|
+
messageContextRequestsArrayBaseSlot: Fr,
|
|
660
|
+
messageContextResponsesArrayBaseSlot: Fr,
|
|
661
|
+
scope: AztecAddress,
|
|
662
|
+
) {
|
|
663
|
+
try {
|
|
664
|
+
if (!this.contractAddress.equals(contractAddress)) {
|
|
665
|
+
throw new Error(`Got a message context request from ${contractAddress}, expected ${this.contractAddress}`);
|
|
666
|
+
}
|
|
667
|
+
|
|
668
|
+
// TODO(@mverzilli): this is a prime example of where using a volatile array would make much more sense, we don't
|
|
669
|
+
// need scopes here, we just need a bit of shared memory to cross boundaries between Noir and TS.
|
|
670
|
+
// At the same time, we don't want to allow any global scope access other than where backwards compatibility
|
|
671
|
+
// forces us to. Hence we need the scope here to be artificial.
|
|
672
|
+
const requestCapsules = await this.capsuleService.readCapsuleArray(
|
|
673
|
+
contractAddress,
|
|
674
|
+
messageContextRequestsArrayBaseSlot,
|
|
675
|
+
this.jobId,
|
|
676
|
+
scope,
|
|
677
|
+
);
|
|
678
|
+
|
|
679
|
+
const txHashes = requestCapsules.map((fields, i) => {
|
|
680
|
+
if (fields.length !== 1) {
|
|
681
|
+
throw new Error(
|
|
682
|
+
`Malformed message context request at index ${i}: expected 1 field (tx hash), got ${fields.length}`,
|
|
683
|
+
);
|
|
684
|
+
}
|
|
685
|
+
return fields[0];
|
|
686
|
+
});
|
|
687
|
+
|
|
688
|
+
const maybeMessageContexts = await this.messageContextService.getMessageContextsByTxHash(
|
|
689
|
+
txHashes,
|
|
690
|
+
this.anchorBlockHeader.getBlockNumber(),
|
|
691
|
+
);
|
|
692
|
+
|
|
693
|
+
// Leave response in response capsule array.
|
|
694
|
+
await this.capsuleService.setCapsuleArray(
|
|
695
|
+
contractAddress,
|
|
696
|
+
messageContextResponsesArrayBaseSlot,
|
|
697
|
+
maybeMessageContexts.map(MessageContext.toSerializedOption),
|
|
698
|
+
this.jobId,
|
|
699
|
+
scope,
|
|
700
|
+
);
|
|
701
|
+
} finally {
|
|
702
|
+
await this.capsuleService.setCapsuleArray(
|
|
703
|
+
contractAddress,
|
|
704
|
+
messageContextRequestsArrayBaseSlot,
|
|
705
|
+
[],
|
|
706
|
+
this.jobId,
|
|
707
|
+
scope,
|
|
708
|
+
);
|
|
709
|
+
}
|
|
710
|
+
}
|
|
711
|
+
|
|
712
|
+
public setCapsule(contractAddress: AztecAddress, slot: Fr, capsule: Fr[], scope: AztecAddress): void {
|
|
506
713
|
if (!contractAddress.equals(this.contractAddress)) {
|
|
507
714
|
// TODO(#10727): instead of this check that this.contractAddress is allowed to access the external DB
|
|
508
715
|
throw new Error(`Contract ${contractAddress} is not allowed to access ${this.contractAddress}'s PXE DB`);
|
|
509
716
|
}
|
|
510
|
-
this.
|
|
511
|
-
return Promise.resolve();
|
|
717
|
+
this.capsuleService.setCapsule(contractAddress, slot, capsule, this.jobId, scope);
|
|
512
718
|
}
|
|
513
719
|
|
|
514
|
-
public
|
|
720
|
+
public getCapsule(contractAddress: AztecAddress, slot: Fr, scope: AztecAddress): Promise<Fr[] | null> {
|
|
515
721
|
if (!contractAddress.equals(this.contractAddress)) {
|
|
516
722
|
// TODO(#10727): instead of this check that this.contractAddress is allowed to access the external DB
|
|
517
723
|
throw new Error(`Contract ${contractAddress} is not allowed to access ${this.contractAddress}'s PXE DB`);
|
|
518
724
|
}
|
|
519
|
-
return (
|
|
520
|
-
// TODO(#12425): On the following line, the pertinent capsule gets overshadowed by the transient one. Tackle this.
|
|
521
|
-
this.capsules.find(c => c.contractAddress.equals(contractAddress) && c.storageSlot.equals(slot))?.data ??
|
|
522
|
-
(await this.capsuleStore.loadCapsule(this.contractAddress, slot, this.jobId))
|
|
523
|
-
);
|
|
725
|
+
return this.capsuleService.getCapsule(contractAddress, slot, this.jobId, scope, this.capsules);
|
|
524
726
|
}
|
|
525
727
|
|
|
526
|
-
public
|
|
728
|
+
public deleteCapsule(contractAddress: AztecAddress, slot: Fr, scope: AztecAddress): void {
|
|
527
729
|
if (!contractAddress.equals(this.contractAddress)) {
|
|
528
730
|
// TODO(#10727): instead of this check that this.contractAddress is allowed to access the external DB
|
|
529
731
|
throw new Error(`Contract ${contractAddress} is not allowed to access ${this.contractAddress}'s PXE DB`);
|
|
530
732
|
}
|
|
531
|
-
this.
|
|
532
|
-
return Promise.resolve();
|
|
733
|
+
this.capsuleService.deleteCapsule(contractAddress, slot, this.jobId, scope);
|
|
533
734
|
}
|
|
534
735
|
|
|
535
|
-
public
|
|
736
|
+
public copyCapsule(
|
|
536
737
|
contractAddress: AztecAddress,
|
|
537
738
|
srcSlot: Fr,
|
|
538
739
|
dstSlot: Fr,
|
|
539
740
|
numEntries: number,
|
|
741
|
+
scope: AztecAddress,
|
|
540
742
|
): Promise<void> {
|
|
541
743
|
if (!contractAddress.equals(this.contractAddress)) {
|
|
542
744
|
// TODO(#10727): instead of this check that this.contractAddress is allowed to access the external DB
|
|
543
745
|
throw new Error(`Contract ${contractAddress} is not allowed to access ${this.contractAddress}'s PXE DB`);
|
|
544
746
|
}
|
|
545
|
-
return this.
|
|
747
|
+
return this.capsuleService.copyCapsule(contractAddress, srcSlot, dstSlot, numEntries, this.jobId, scope);
|
|
748
|
+
}
|
|
749
|
+
|
|
750
|
+
/**
|
|
751
|
+
* Clears cached sync state for a contract for a set of scopes, forcing re-sync on the next query so that newly
|
|
752
|
+
* stored notes or events are discovered.
|
|
753
|
+
*/
|
|
754
|
+
public setContractSyncCacheInvalid(contractAddress: AztecAddress, scopes: AztecAddress[]): void {
|
|
755
|
+
if (!contractAddress.equals(this.contractAddress)) {
|
|
756
|
+
throw new Error(`Contract ${this.contractAddress} cannot invalidate sync cache of ${contractAddress}`);
|
|
757
|
+
}
|
|
758
|
+
this.contractSyncService.invalidateContractForScopes(contractAddress, scopes);
|
|
546
759
|
}
|
|
547
760
|
|
|
548
761
|
// TODO(#11849): consider replacing this oracle with a pure Noir implementation of aes decryption.
|
|
549
|
-
public
|
|
762
|
+
public decryptAes128(ciphertext: Buffer, iv: Buffer, symKey: Buffer): Promise<Buffer> {
|
|
550
763
|
const aes128 = new Aes128();
|
|
551
764
|
return aes128.decryptBufferCBC(ciphertext, iv, symKey);
|
|
552
765
|
}
|
|
553
766
|
|
|
554
767
|
/**
|
|
555
|
-
* Retrieves the shared secret for a given address and ephemeral public key.
|
|
768
|
+
* Retrieves the app-siloed shared secret for a given address and ephemeral public key.
|
|
556
769
|
* @param address - The address to get the secret for.
|
|
557
770
|
* @param ephPk - The ephemeral public key to get the secret for.
|
|
558
|
-
* @
|
|
771
|
+
* @param contractAddress - The contract address for app-siloing (validated against execution context).
|
|
772
|
+
* @returns The app-siloed shared secret as a Field.
|
|
559
773
|
*/
|
|
560
|
-
public
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
|
|
565
|
-
|
|
774
|
+
public async getSharedSecret(address: AztecAddress, ephPk: Point, contractAddress: AztecAddress): Promise<Fr> {
|
|
775
|
+
if (!contractAddress.equals(this.contractAddress)) {
|
|
776
|
+
throw new Error(
|
|
777
|
+
`getSharedSecret called with contract address ${contractAddress}, expected ${this.contractAddress}`,
|
|
778
|
+
);
|
|
779
|
+
}
|
|
566
780
|
const recipientCompleteAddress = await this.getCompleteAddressOrFail(address);
|
|
567
781
|
const ivskM = await this.keyStore.getMasterSecretKey(
|
|
568
782
|
recipientCompleteAddress.publicKeys.masterIncomingViewingPublicKey,
|
|
569
783
|
);
|
|
570
784
|
const addressSecret = await computeAddressSecret(await recipientCompleteAddress.getPreaddress(), ivskM);
|
|
571
|
-
return
|
|
785
|
+
return deriveAppSiloedSharedSecret(addressSecret, ephPk, this.contractAddress);
|
|
786
|
+
}
|
|
787
|
+
|
|
788
|
+
public emitOffchainEffect(data: Fr[]): Promise<void> {
|
|
789
|
+
this.offchainEffects.push({ data, contractAddress: this.contractAddress });
|
|
790
|
+
return Promise.resolve();
|
|
791
|
+
}
|
|
792
|
+
|
|
793
|
+
/** Returns offchain effects collected during execution. */
|
|
794
|
+
public getOffchainEffects(): OffchainEffect[] {
|
|
795
|
+
return this.offchainEffects;
|
|
796
|
+
}
|
|
797
|
+
|
|
798
|
+
/** Runs a query concurrently with a validation that the block hash is not ahead of the anchor block. */
|
|
799
|
+
async #queryWithBlockHashNotAfterAnchor<T>(blockHash: BlockHash, query: () => Promise<T>): Promise<T> {
|
|
800
|
+
const [response] = await Promise.all([
|
|
801
|
+
query(),
|
|
802
|
+
(async () => {
|
|
803
|
+
const header = await this.aztecNode.getBlockHeader(blockHash);
|
|
804
|
+
if (!header) {
|
|
805
|
+
throw new Error(`Could not find block header for block hash ${blockHash}`);
|
|
806
|
+
}
|
|
807
|
+
|
|
808
|
+
if (header.getBlockNumber() > this.anchorBlockHeader.getBlockNumber()) {
|
|
809
|
+
throw new Error(
|
|
810
|
+
`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 ${await this.anchorBlockHeader.hash()}).`,
|
|
811
|
+
);
|
|
812
|
+
}
|
|
813
|
+
})(),
|
|
814
|
+
]);
|
|
815
|
+
return response;
|
|
572
816
|
}
|
|
573
817
|
}
|