@aztec/pxe 0.0.1-commit.ff7989d6c → 0.0.1-commit.fff30aa
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/bin/check_oracle_version.js +4 -4
- package/dest/block_synchronizer/block_synchronizer.d.ts +3 -1
- package/dest/block_synchronizer/block_synchronizer.d.ts.map +1 -1
- package/dest/block_synchronizer/block_synchronizer.js +10 -0
- package/dest/config/package_info.js +1 -1
- package/dest/contract_function_simulator/contract_function_simulator.d.ts +13 -5
- package/dest/contract_function_simulator/contract_function_simulator.d.ts.map +1 -1
- package/dest/contract_function_simulator/contract_function_simulator.js +39 -12
- package/dest/contract_function_simulator/ephemeral_array_service.d.ts +28 -0
- package/dest/contract_function_simulator/ephemeral_array_service.d.ts.map +1 -0
- package/dest/contract_function_simulator/ephemeral_array_service.js +78 -0
- 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 +4 -6
- 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 +8 -10
- package/dest/contract_function_simulator/noir-structs/log_retrieval_request.d.ts +1 -1
- package/dest/contract_function_simulator/noir-structs/log_retrieval_request.js +1 -1
- package/dest/contract_function_simulator/noir-structs/log_retrieval_response.d.ts +2 -2
- 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 +2 -4
- package/dest/contract_function_simulator/noir-structs/note_validation_request.d.ts +4 -7
- 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 +6 -11
- package/dest/contract_function_simulator/oracle/interfaces.d.ts +61 -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 +47 -0
- package/dest/contract_function_simulator/oracle/oracle.d.ts +75 -44
- package/dest/contract_function_simulator/oracle/oracle.d.ts.map +1 -1
- package/dest/contract_function_simulator/oracle/oracle.js +281 -96
- package/dest/contract_function_simulator/oracle/private_execution.js +5 -3
- package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts +24 -50
- package/dest/contract_function_simulator/oracle/private_execution_oracle.d.ts.map +1 -1
- package/dest/contract_function_simulator/oracle/private_execution_oracle.js +53 -75
- package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts +94 -60
- package/dest/contract_function_simulator/oracle/utility_execution_oracle.d.ts.map +1 -1
- package/dest/contract_function_simulator/oracle/utility_execution_oracle.js +276 -110
- package/dest/contract_function_simulator/pick_notes.d.ts +1 -1
- package/dest/contract_function_simulator/pick_notes.d.ts.map +1 -1
- package/dest/contract_function_simulator/pick_notes.js +20 -3
- package/dest/contract_function_simulator/proxied_contract_data_source.d.ts +1 -1
- package/dest/contract_function_simulator/proxied_contract_data_source.d.ts.map +1 -1
- package/dest/contract_function_simulator/proxied_contract_data_source.js +3 -0
- package/dest/contract_logging.d.ts +9 -4
- package/dest/contract_logging.d.ts.map +1 -1
- package/dest/contract_logging.js +21 -6
- package/dest/contract_sync/contract_sync_service.d.ts +6 -5
- package/dest/contract_sync/contract_sync_service.d.ts.map +1 -1
- package/dest/contract_sync/contract_sync_service.js +62 -43
- package/dest/contract_sync/helpers.d.ts +2 -3
- package/dest/contract_sync/helpers.d.ts.map +1 -1
- package/dest/contract_sync/helpers.js +7 -2
- package/dest/debug/pxe_debug_utils.d.ts +3 -3
- package/dest/debug/pxe_debug_utils.d.ts.map +1 -1
- package/dest/entrypoints/client/bundle/index.d.ts +1 -2
- package/dest/entrypoints/client/bundle/index.d.ts.map +1 -1
- package/dest/entrypoints/client/bundle/index.js +0 -1
- package/dest/entrypoints/client/bundle/utils.d.ts +2 -2
- package/dest/entrypoints/client/bundle/utils.d.ts.map +1 -1
- package/dest/entrypoints/client/bundle/utils.js +2 -2
- package/dest/entrypoints/client/lazy/index.d.ts +1 -2
- package/dest/entrypoints/client/lazy/index.d.ts.map +1 -1
- package/dest/entrypoints/client/lazy/index.js +0 -1
- package/dest/entrypoints/client/lazy/utils.d.ts +2 -2
- package/dest/entrypoints/client/lazy/utils.d.ts.map +1 -1
- package/dest/entrypoints/client/lazy/utils.js +2 -2
- package/dest/entrypoints/pxe_creation_options.d.ts +3 -1
- package/dest/entrypoints/pxe_creation_options.d.ts.map +1 -1
- package/dest/entrypoints/pxe_creation_options.js +3 -1
- package/dest/entrypoints/server/index.d.ts +2 -3
- package/dest/entrypoints/server/index.d.ts.map +1 -1
- package/dest/entrypoints/server/index.js +1 -2
- package/dest/entrypoints/server/utils.d.ts +2 -2
- package/dest/entrypoints/server/utils.d.ts.map +1 -1
- package/dest/entrypoints/server/utils.js +2 -2
- 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/events/private_event_filter_validator.d.ts +3 -2
- package/dest/events/private_event_filter_validator.d.ts.map +1 -1
- package/dest/events/private_event_filter_validator.js +15 -0
- package/dest/logs/log_service.d.ts +7 -8
- package/dest/logs/log_service.d.ts.map +1 -1
- package/dest/logs/log_service.js +29 -39
- 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 +4 -5
- package/dest/notes/note_service.d.ts.map +1 -1
- package/dest/notes/note_service.js +14 -5
- package/dest/notes_filter.d.ts +2 -3
- package/dest/notes_filter.d.ts.map +1 -1
- package/dest/oracle_version.d.ts +4 -3
- package/dest/oracle_version.d.ts.map +1 -1
- package/dest/oracle_version.js +20 -10
- 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 +125 -64
- 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 +17 -12
- package/dest/private_kernel/private_kernel_oracle.d.ts +5 -5
- package/dest/private_kernel/private_kernel_oracle.d.ts.map +1 -1
- package/dest/private_kernel/private_kernel_oracle.js +12 -15
- package/dest/pxe.d.ts +10 -7
- package/dest/pxe.d.ts.map +1 -1
- package/dest/pxe.js +51 -28
- package/dest/storage/anchor_block_store/anchor_block_store.js +1 -1
- package/dest/storage/capsule_store/capsule_service.d.ts +21 -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 +36 -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 +1 -1
- package/dest/storage/contract_store/contract_store.d.ts.map +1 -1
- package/dest/storage/contract_store/contract_store.js +4 -2
- package/dest/storage/metadata.d.ts +1 -1
- package/dest/storage/metadata.js +1 -1
- package/dest/storage/note_store/note_store.d.ts +1 -1
- package/dest/storage/note_store/note_store.d.ts.map +1 -1
- package/dest/storage/note_store/note_store.js +2 -2
- package/dest/storage/private_event_store/private_event_store.d.ts +1 -1
- package/dest/storage/private_event_store/private_event_store.d.ts.map +1 -1
- package/dest/storage/private_event_store/private_event_store.js +3 -0
- 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/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 +4 -18
- 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/bin/check_oracle_version.ts +4 -4
- package/src/block_synchronizer/block_synchronizer.ts +12 -0
- package/src/config/package_info.ts +1 -1
- package/src/contract_function_simulator/contract_function_simulator.ts +60 -24
- package/src/contract_function_simulator/ephemeral_array_service.ts +110 -0
- 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 +9 -9
- package/src/contract_function_simulator/noir-structs/log_retrieval_request.ts +1 -1
- package/src/contract_function_simulator/noir-structs/log_retrieval_response.ts +2 -5
- package/src/contract_function_simulator/noir-structs/note_validation_request.ts +4 -10
- package/src/contract_function_simulator/oracle/interfaces.ts +82 -54
- package/src/contract_function_simulator/oracle/legacy_oracle_mappings.ts +104 -0
- package/src/contract_function_simulator/oracle/oracle.ts +363 -139
- package/src/contract_function_simulator/oracle/private_execution.ts +4 -4
- package/src/contract_function_simulator/oracle/private_execution_oracle.ts +65 -95
- package/src/contract_function_simulator/oracle/utility_execution_oracle.ts +451 -147
- package/src/contract_function_simulator/pick_notes.ts +22 -3
- package/src/contract_function_simulator/proxied_contract_data_source.ts +8 -1
- package/src/contract_logging.ts +18 -5
- package/src/contract_sync/contract_sync_service.ts +99 -75
- package/src/contract_sync/helpers.ts +4 -4
- package/src/debug/pxe_debug_utils.ts +3 -3
- package/src/entrypoints/client/bundle/index.ts +0 -1
- package/src/entrypoints/client/bundle/utils.ts +2 -3
- package/src/entrypoints/client/lazy/index.ts +0 -1
- package/src/entrypoints/client/lazy/utils.ts +2 -3
- package/src/entrypoints/pxe_creation_options.ts +7 -0
- package/src/entrypoints/server/index.ts +1 -2
- package/src/entrypoints/server/utils.ts +2 -3
- package/src/events/event_service.ts +17 -4
- package/src/events/private_event_filter_validator.ts +21 -1
- package/src/logs/log_service.ts +59 -75
- package/src/messages/message_context_service.ts +44 -0
- package/src/notes/note_service.ts +18 -8
- package/src/notes_filter.ts +1 -3
- package/src/oracle_version.ts +20 -10
- package/src/private_kernel/hints/private_kernel_reset_private_inputs_builder.ts +157 -110
- package/src/private_kernel/hints/test_utils.ts +325 -0
- package/src/private_kernel/private_kernel_execution_prover.ts +17 -15
- package/src/private_kernel/private_kernel_oracle.ts +14 -14
- package/src/pxe.ts +78 -33
- package/src/storage/anchor_block_store/anchor_block_store.ts +1 -1
- package/src/storage/capsule_store/capsule_service.ts +90 -0
- package/src/storage/capsule_store/capsule_store.ts +44 -26
- package/src/storage/capsule_store/index.ts +1 -0
- package/src/storage/contract_store/contract_store.ts +8 -6
- package/src/storage/metadata.ts +1 -1
- package/src/storage/note_store/note_store.ts +2 -5
- package/src/storage/private_event_store/private_event_store.ts +4 -0
- 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/index.ts +2 -2
- package/src/tagging/recipient_sync/load_private_logs_for_sender_recipient_pair.ts +8 -21
- 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/access_scopes.d.ts +0 -9
- package/dest/access_scopes.d.ts.map +0 -1
- package/dest/access_scopes.js +0 -6
- package/src/access_scopes.ts +0 -9
package/src/pxe.ts
CHANGED
|
@@ -18,6 +18,7 @@ import {
|
|
|
18
18
|
} from '@aztec/stdlib/abi';
|
|
19
19
|
import type { AuthWitness } from '@aztec/stdlib/auth-witness';
|
|
20
20
|
import type { AztecAddress } from '@aztec/stdlib/aztec-address';
|
|
21
|
+
import type { L2TipsProvider } from '@aztec/stdlib/block';
|
|
21
22
|
import {
|
|
22
23
|
CompleteAddress,
|
|
23
24
|
type ContractInstanceWithAddress,
|
|
@@ -52,7 +53,6 @@ import {
|
|
|
52
53
|
|
|
53
54
|
import { inspect } from 'util';
|
|
54
55
|
|
|
55
|
-
import type { AccessScopes } from './access_scopes.js';
|
|
56
56
|
import { BlockSynchronizer } from './block_synchronizer/index.js';
|
|
57
57
|
import type { PXEConfig } from './config/index.js';
|
|
58
58
|
import { BenchmarkedNodeFactory } from './contract_function_simulator/benchmarked_node.js';
|
|
@@ -68,6 +68,7 @@ import { PXEDebugUtils } from './debug/pxe_debug_utils.js';
|
|
|
68
68
|
import { enrichPublicSimulationError, enrichSimulationError } from './error_enriching.js';
|
|
69
69
|
import { PrivateEventFilterValidator } from './events/private_event_filter_validator.js';
|
|
70
70
|
import { JobCoordinator } from './job_coordinator/job_coordinator.js';
|
|
71
|
+
import { MessageContextService } from './messages/message_context_service.js';
|
|
71
72
|
import {
|
|
72
73
|
PrivateKernelExecutionProver,
|
|
73
74
|
type PrivateKernelExecutionProverConfig,
|
|
@@ -95,7 +96,7 @@ export type ProfileTxOpts = {
|
|
|
95
96
|
/** If true, proof generation is skipped during profiling. Defaults to true. */
|
|
96
97
|
skipProofGeneration?: boolean;
|
|
97
98
|
/** Addresses whose private state and keys are accessible during private execution. */
|
|
98
|
-
scopes:
|
|
99
|
+
scopes: AztecAddress[];
|
|
99
100
|
};
|
|
100
101
|
|
|
101
102
|
/** Options for PXE.simulateTx. */
|
|
@@ -106,10 +107,12 @@ export type SimulateTxOpts = {
|
|
|
106
107
|
skipTxValidation?: boolean;
|
|
107
108
|
/** If false, fees are enforced. */
|
|
108
109
|
skipFeeEnforcement?: boolean;
|
|
109
|
-
/**
|
|
110
|
+
/** If true, kernel logic is emulated in TS for simulation */
|
|
111
|
+
skipKernels?: boolean;
|
|
112
|
+
/** State overrides for the simulation, such as contract instances and artifacts. Requires skipKernels: true */
|
|
110
113
|
overrides?: SimulationOverrides;
|
|
111
114
|
/** Addresses whose private state and keys are accessible during private execution */
|
|
112
|
-
scopes:
|
|
115
|
+
scopes: AztecAddress[];
|
|
113
116
|
};
|
|
114
117
|
|
|
115
118
|
/** Options for PXE.executeUtility. */
|
|
@@ -117,7 +120,7 @@ export type ExecuteUtilityOpts = {
|
|
|
117
120
|
/** The authentication witnesses required for the function call. */
|
|
118
121
|
authwits?: AuthWitness[];
|
|
119
122
|
/** The accounts whose notes we can access in this call */
|
|
120
|
-
scopes:
|
|
123
|
+
scopes: AztecAddress[];
|
|
121
124
|
};
|
|
122
125
|
|
|
123
126
|
/** Args for PXE.create. */
|
|
@@ -157,6 +160,8 @@ export class PXE {
|
|
|
157
160
|
private addressStore: AddressStore,
|
|
158
161
|
private privateEventStore: PrivateEventStore,
|
|
159
162
|
private contractSyncService: ContractSyncService,
|
|
163
|
+
private messageContextService: MessageContextService,
|
|
164
|
+
private l2TipsStore: L2TipsProvider,
|
|
160
165
|
private simulator: CircuitSimulator,
|
|
161
166
|
private proverEnabled: boolean,
|
|
162
167
|
private proofCreator: PrivateKernelProver,
|
|
@@ -212,6 +217,8 @@ export class PXE {
|
|
|
212
217
|
noteStore,
|
|
213
218
|
createLogger('pxe:contract_sync', bindings),
|
|
214
219
|
);
|
|
220
|
+
const messageContextService = new MessageContextService(node);
|
|
221
|
+
|
|
215
222
|
const synchronizer = new BlockSynchronizer(
|
|
216
223
|
node,
|
|
217
224
|
store,
|
|
@@ -252,6 +259,8 @@ export class PXE {
|
|
|
252
259
|
addressStore,
|
|
253
260
|
privateEventStore,
|
|
254
261
|
contractSyncService,
|
|
262
|
+
messageContextService,
|
|
263
|
+
tipsStore,
|
|
255
264
|
simulator,
|
|
256
265
|
proverEnabled,
|
|
257
266
|
proofCreator,
|
|
@@ -286,6 +295,7 @@ export class PXE {
|
|
|
286
295
|
keyStore: this.keyStore,
|
|
287
296
|
addressStore: this.addressStore,
|
|
288
297
|
aztecNode: BenchmarkedNodeFactory.create(this.node),
|
|
298
|
+
l2TipsStore: this.l2TipsStore,
|
|
289
299
|
senderTaggingStore: this.senderTaggingStore,
|
|
290
300
|
recipientTaggingStore: this.recipientTaggingStore,
|
|
291
301
|
senderAddressBookStore: this.senderAddressBookStore,
|
|
@@ -293,6 +303,7 @@ export class PXE {
|
|
|
293
303
|
privateEventStore: this.privateEventStore,
|
|
294
304
|
simulator: this.simulator,
|
|
295
305
|
contractSyncService: this.contractSyncService,
|
|
306
|
+
messageContextService: this.messageContextService,
|
|
296
307
|
});
|
|
297
308
|
}
|
|
298
309
|
|
|
@@ -358,7 +369,7 @@ export class PXE {
|
|
|
358
369
|
async #executePrivate(
|
|
359
370
|
contractFunctionSimulator: ContractFunctionSimulator,
|
|
360
371
|
txRequest: TxExecutionRequest,
|
|
361
|
-
scopes:
|
|
372
|
+
scopes: AztecAddress[],
|
|
362
373
|
jobId: string,
|
|
363
374
|
): Promise<PrivateExecutionResult> {
|
|
364
375
|
const { origin: contractAddress, functionSelector } = txRequest;
|
|
@@ -407,12 +418,19 @@ export class PXE {
|
|
|
407
418
|
contractFunctionSimulator: ContractFunctionSimulator,
|
|
408
419
|
call: FunctionCall,
|
|
409
420
|
authWitnesses: AuthWitness[] | undefined,
|
|
410
|
-
scopes:
|
|
421
|
+
scopes: AztecAddress[],
|
|
411
422
|
jobId: string,
|
|
412
423
|
) {
|
|
413
424
|
try {
|
|
414
425
|
const anchorBlockHeader = await this.anchorBlockStore.getBlockHeader();
|
|
415
|
-
|
|
426
|
+
const { result, offchainEffects } = await contractFunctionSimulator.runUtility(
|
|
427
|
+
call,
|
|
428
|
+
authWitnesses ?? [],
|
|
429
|
+
anchorBlockHeader,
|
|
430
|
+
scopes,
|
|
431
|
+
jobId,
|
|
432
|
+
);
|
|
433
|
+
return { result, offchainEffects };
|
|
416
434
|
} catch (err) {
|
|
417
435
|
if (err instanceof SimulationError) {
|
|
418
436
|
await enrichSimulationError(err, this.contractStore, this.log);
|
|
@@ -466,8 +484,7 @@ export class PXE {
|
|
|
466
484
|
config: PrivateKernelExecutionProverConfig,
|
|
467
485
|
): Promise<PrivateKernelExecutionProofOutput<PrivateKernelTailCircuitPublicInputs>> {
|
|
468
486
|
const anchorBlockHeader = await this.anchorBlockStore.getBlockHeader();
|
|
469
|
-
const
|
|
470
|
-
const kernelOracle = new PrivateKernelOracle(this.contractStore, this.keyStore, this.node, anchorBlockHash);
|
|
487
|
+
const kernelOracle = new PrivateKernelOracle(this.contractStore, this.keyStore, this.node, anchorBlockHeader);
|
|
471
488
|
const kernelTraceProver = new PrivateKernelExecutionProver(
|
|
472
489
|
kernelOracle,
|
|
473
490
|
proofCreator,
|
|
@@ -485,7 +502,9 @@ export class PXE {
|
|
|
485
502
|
* @returns The synced block header
|
|
486
503
|
*/
|
|
487
504
|
public getSyncedBlockHeader(): Promise<BlockHeader> {
|
|
488
|
-
return this
|
|
505
|
+
return this.#putInJobQueue(() => {
|
|
506
|
+
return this.anchorBlockStore.getBlockHeader();
|
|
507
|
+
});
|
|
489
508
|
}
|
|
490
509
|
|
|
491
510
|
/**
|
|
@@ -542,6 +561,12 @@ export class PXE {
|
|
|
542
561
|
* TODO: It's strange that we return the address here and I (benesjan) think we should drop the return value.
|
|
543
562
|
*/
|
|
544
563
|
public async registerSender(sender: AztecAddress): Promise<AztecAddress> {
|
|
564
|
+
if (!(await sender.isValid())) {
|
|
565
|
+
throw new Error(
|
|
566
|
+
`Address ${sender} is not valid: it does not correspond to a point on the Grumpkin curve. Cannot register it as a sender.`,
|
|
567
|
+
);
|
|
568
|
+
}
|
|
569
|
+
|
|
545
570
|
const accounts = await this.keyStore.getAccounts();
|
|
546
571
|
if (accounts.includes(sender)) {
|
|
547
572
|
this.log.info(`Sender:\n "${sender.toString()}"\n already registered.`);
|
|
@@ -552,6 +577,9 @@ export class PXE {
|
|
|
552
577
|
|
|
553
578
|
if (wasAdded) {
|
|
554
579
|
this.log.info(`Added sender:\n ${sender.toString()}`);
|
|
580
|
+
// Wipe the entire sync cache: the new sender's tagged logs could contain notes/events for any contract, so
|
|
581
|
+
// all contracts must re-sync to discover them. Queued to avoid wiping while a job is in flight.
|
|
582
|
+
await this.#putInJobQueue(() => Promise.resolve(this.contractSyncService.wipe()));
|
|
555
583
|
} else {
|
|
556
584
|
this.log.info(`Sender:\n "${sender.toString()}"\n already registered.`);
|
|
557
585
|
}
|
|
@@ -764,17 +792,17 @@ export class PXE {
|
|
|
764
792
|
// transaction before this one is included in a block from this PXE, and that transaction contains a log with
|
|
765
793
|
// a tag derived from the same secret, we would reuse the tag and the transactions would be linked. Hence
|
|
766
794
|
// storing the tags here prevents linkage of txs sent from the same PXE.
|
|
767
|
-
const
|
|
768
|
-
if (
|
|
795
|
+
const taggingIndexRangesUsedInTheTx = privateExecutionResult.entrypoint.taggingIndexRanges;
|
|
796
|
+
if (taggingIndexRangesUsedInTheTx.length > 0) {
|
|
769
797
|
// TODO(benesjan): The following is an expensive operation. Figure out a way to avoid it.
|
|
770
798
|
const txHash = (await txProvingResult.toTx()).txHash;
|
|
771
799
|
|
|
772
|
-
await this.senderTaggingStore.storePendingIndexes(
|
|
773
|
-
this.log.debug(`Stored used
|
|
774
|
-
|
|
800
|
+
await this.senderTaggingStore.storePendingIndexes(taggingIndexRangesUsedInTheTx, txHash, jobId);
|
|
801
|
+
this.log.debug(`Stored used tagging index ranges as sender for the tx`, {
|
|
802
|
+
taggingIndexRangesUsedInTheTx,
|
|
775
803
|
});
|
|
776
804
|
} else {
|
|
777
|
-
this.log.debug(`No
|
|
805
|
+
this.log.debug(`No tagging index ranges used in the tx`);
|
|
778
806
|
}
|
|
779
807
|
|
|
780
808
|
return txProvingResult;
|
|
@@ -881,7 +909,14 @@ export class PXE {
|
|
|
881
909
|
*/
|
|
882
910
|
public simulateTx(
|
|
883
911
|
txRequest: TxExecutionRequest,
|
|
884
|
-
{
|
|
912
|
+
{
|
|
913
|
+
simulatePublic,
|
|
914
|
+
skipTxValidation = false,
|
|
915
|
+
skipFeeEnforcement = false,
|
|
916
|
+
skipKernels = true,
|
|
917
|
+
overrides,
|
|
918
|
+
scopes,
|
|
919
|
+
}: SimulateTxOpts,
|
|
885
920
|
): Promise<TxSimulationResult> {
|
|
886
921
|
// We disable concurrent simulations since those might execute oracles which read and write to the PXE stores (e.g.
|
|
887
922
|
// to the capsules), and we need to prevent concurrent runs from interfering with one another (e.g. attempting to
|
|
@@ -905,17 +940,20 @@ export class PXE {
|
|
|
905
940
|
await this.blockStateSynchronizer.sync();
|
|
906
941
|
const syncTime = syncTimer.ms();
|
|
907
942
|
|
|
908
|
-
const contractFunctionSimulator = this.#getSimulatorForTx(overrides);
|
|
909
|
-
// Temporary: in case there are overrides, we have to skip the kernels or validations
|
|
910
|
-
// will fail. Consider handing control to the user/wallet on whether they want to run them
|
|
911
|
-
// or not.
|
|
912
943
|
const overriddenContracts = overrides?.contracts ? new Set(Object.keys(overrides.contracts)) : undefined;
|
|
913
944
|
const hasOverriddenContracts = overriddenContracts !== undefined && overriddenContracts.size > 0;
|
|
914
|
-
const skipKernels = hasOverriddenContracts;
|
|
915
945
|
|
|
916
|
-
|
|
946
|
+
if (hasOverriddenContracts && !skipKernels) {
|
|
947
|
+
throw new Error(
|
|
948
|
+
'Simulating with overridden contracts is not compatible with kernel execution. Please set skipKernels to true when simulating with overridden contracts.',
|
|
949
|
+
);
|
|
950
|
+
}
|
|
951
|
+
const contractFunctionSimulator = this.#getSimulatorForTx(overrides);
|
|
952
|
+
|
|
917
953
|
if (hasOverriddenContracts) {
|
|
918
|
-
|
|
954
|
+
// Overridden contracts don't have a sync function, so calling sync on them would fail.
|
|
955
|
+
// We exclude them so the sync service skips them entirely.
|
|
956
|
+
this.contractSyncService.setExcludedFromSync(jobId, overriddenContracts);
|
|
919
957
|
}
|
|
920
958
|
|
|
921
959
|
// Execution of private functions only; no proving, and no kernel logic.
|
|
@@ -958,7 +996,8 @@ export class PXE {
|
|
|
958
996
|
const validationResult = await this.node.isValidTx(simulatedTx, { isSimulation: true, skipFeeEnforcement });
|
|
959
997
|
validationTime = validationTimer.ms();
|
|
960
998
|
if (validationResult.result === 'invalid') {
|
|
961
|
-
|
|
999
|
+
const reason = validationResult.reason.length > 0 ? ` Reason: ${validationResult.reason.join(', ')}` : '';
|
|
1000
|
+
throw new Error(`The simulated transaction is unable to be added to state and is invalid.${reason}`);
|
|
962
1001
|
}
|
|
963
1002
|
}
|
|
964
1003
|
|
|
@@ -1009,7 +1048,7 @@ export class PXE {
|
|
|
1009
1048
|
inspect(txRequest),
|
|
1010
1049
|
`simulatePublic=${simulatePublic}`,
|
|
1011
1050
|
`skipTxValidation=${skipTxValidation}`,
|
|
1012
|
-
`scopes=${scopes
|
|
1051
|
+
`scopes=${scopes.map(s => s.toString()).join(', ')}`,
|
|
1013
1052
|
);
|
|
1014
1053
|
}
|
|
1015
1054
|
});
|
|
@@ -1021,7 +1060,7 @@ export class PXE {
|
|
|
1021
1060
|
*/
|
|
1022
1061
|
public executeUtility(
|
|
1023
1062
|
call: FunctionCall,
|
|
1024
|
-
{ authwits, scopes }: ExecuteUtilityOpts = { scopes:
|
|
1063
|
+
{ authwits, scopes }: ExecuteUtilityOpts = { scopes: [] },
|
|
1025
1064
|
): Promise<UtilityExecutionResult> {
|
|
1026
1065
|
// We disable concurrent executions since those might execute oracles which read and write to the PXE stores (e.g.
|
|
1027
1066
|
// to the capsules), and we need to prevent concurrent runs from interfering with one another (e.g. attempting to
|
|
@@ -1046,7 +1085,7 @@ export class PXE {
|
|
|
1046
1085
|
scopes,
|
|
1047
1086
|
);
|
|
1048
1087
|
|
|
1049
|
-
const executionResult = await this.#executeUtility(
|
|
1088
|
+
const { result: executionResult, offchainEffects } = await this.#executeUtility(
|
|
1050
1089
|
contractFunctionSimulator,
|
|
1051
1090
|
call,
|
|
1052
1091
|
authwits ?? [],
|
|
@@ -1067,14 +1106,19 @@ export class PXE {
|
|
|
1067
1106
|
};
|
|
1068
1107
|
|
|
1069
1108
|
const simulationStats = contractFunctionSimulator.getStats();
|
|
1070
|
-
return {
|
|
1109
|
+
return {
|
|
1110
|
+
result: executionResult,
|
|
1111
|
+
offchainEffects,
|
|
1112
|
+
anchorBlockTimestamp: anchorBlockHeader.globalVariables.timestamp,
|
|
1113
|
+
stats: { timings, nodeRPCCalls: simulationStats.nodeRPCCalls },
|
|
1114
|
+
};
|
|
1071
1115
|
} catch (err: any) {
|
|
1072
1116
|
const { to, name, args } = call;
|
|
1073
1117
|
const stringifiedArgs = args.map(arg => arg.toString()).join(', ');
|
|
1074
1118
|
throw this.#contextualizeError(
|
|
1075
1119
|
err,
|
|
1076
1120
|
`executeUtility ${to}:${name}(${stringifiedArgs})`,
|
|
1077
|
-
`scopes=${scopes
|
|
1121
|
+
`scopes=${scopes.map(s => s.toString()).join(', ')}`,
|
|
1078
1122
|
);
|
|
1079
1123
|
}
|
|
1080
1124
|
});
|
|
@@ -1131,7 +1175,8 @@ export class PXE {
|
|
|
1131
1175
|
/**
|
|
1132
1176
|
* Stops the PXE's job queue.
|
|
1133
1177
|
*/
|
|
1134
|
-
public stop(): Promise<void> {
|
|
1135
|
-
|
|
1178
|
+
public async stop(): Promise<void> {
|
|
1179
|
+
await this.jobQueue.end();
|
|
1180
|
+
await this.blockStateSynchronizer.stop();
|
|
1136
1181
|
}
|
|
1137
1182
|
}
|
|
@@ -23,7 +23,7 @@ export class AnchorBlockStore {
|
|
|
23
23
|
}
|
|
24
24
|
|
|
25
25
|
async getBlockHeader(): Promise<BlockHeader> {
|
|
26
|
-
const headerBuffer = await this.#synchronizedHeader.getAsync();
|
|
26
|
+
const headerBuffer = await this.#store.transactionAsync(() => this.#synchronizedHeader.getAsync());
|
|
27
27
|
if (!headerBuffer) {
|
|
28
28
|
throw new Error(`Trying to get block header with a not-yet-synchronized PXE - this should never happen`);
|
|
29
29
|
}
|
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
import type { Fr } from '@aztec/foundation/curves/bn254';
|
|
2
|
+
import { AztecAddress } from '@aztec/stdlib/aztec-address';
|
|
3
|
+
import type { Capsule } from '@aztec/stdlib/tx';
|
|
4
|
+
|
|
5
|
+
import type { CapsuleStore } from './capsule_store.js';
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Wraps a CapsuleStore with scope-based access control. Each operation asserts that the requested scope is in the
|
|
9
|
+
* allowed scopes list before delegating to the underlying store.
|
|
10
|
+
*/
|
|
11
|
+
export class CapsuleService {
|
|
12
|
+
constructor(
|
|
13
|
+
private readonly capsuleStore: CapsuleStore,
|
|
14
|
+
private readonly allowedScopes: AztecAddress[],
|
|
15
|
+
) {}
|
|
16
|
+
|
|
17
|
+
setCapsule(contractAddress: AztecAddress, slot: Fr, capsule: Fr[], jobId: string, scope: AztecAddress) {
|
|
18
|
+
assertAllowedScope(scope, this.allowedScopes);
|
|
19
|
+
this.capsuleStore.setCapsule(contractAddress, slot, capsule, jobId, scope);
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
async getCapsule(
|
|
23
|
+
contractAddress: AztecAddress,
|
|
24
|
+
slot: Fr,
|
|
25
|
+
jobId: string,
|
|
26
|
+
scope: AztecAddress,
|
|
27
|
+
transientCapsules?: Capsule[],
|
|
28
|
+
): Promise<Fr[] | null> {
|
|
29
|
+
assertAllowedScope(scope, this.allowedScopes);
|
|
30
|
+
|
|
31
|
+
// TODO(#12425): On the following line, the pertinent capsule gets overshadowed by the transient one. Tackle this.
|
|
32
|
+
const maybeTransientCapsule = transientCapsules?.find(
|
|
33
|
+
c =>
|
|
34
|
+
c.contractAddress.equals(contractAddress) &&
|
|
35
|
+
c.storageSlot.equals(slot) &&
|
|
36
|
+
(c.scope ?? AztecAddress.ZERO).equals(scope),
|
|
37
|
+
)?.data;
|
|
38
|
+
|
|
39
|
+
return maybeTransientCapsule ?? (await this.capsuleStore.getCapsule(contractAddress, slot, jobId, scope));
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
deleteCapsule(contractAddress: AztecAddress, slot: Fr, jobId: string, scope: AztecAddress) {
|
|
43
|
+
assertAllowedScope(scope, this.allowedScopes);
|
|
44
|
+
this.capsuleStore.deleteCapsule(contractAddress, slot, jobId, scope);
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
copyCapsule(
|
|
48
|
+
contractAddress: AztecAddress,
|
|
49
|
+
srcSlot: Fr,
|
|
50
|
+
dstSlot: Fr,
|
|
51
|
+
numEntries: number,
|
|
52
|
+
jobId: string,
|
|
53
|
+
scope: AztecAddress,
|
|
54
|
+
): Promise<void> {
|
|
55
|
+
assertAllowedScope(scope, this.allowedScopes);
|
|
56
|
+
return this.capsuleStore.copyCapsule(contractAddress, srcSlot, dstSlot, numEntries, jobId, scope);
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
appendToCapsuleArray(
|
|
60
|
+
contractAddress: AztecAddress,
|
|
61
|
+
baseSlot: Fr,
|
|
62
|
+
content: Fr[][],
|
|
63
|
+
jobId: string,
|
|
64
|
+
scope: AztecAddress,
|
|
65
|
+
): Promise<void> {
|
|
66
|
+
assertAllowedScope(scope, this.allowedScopes);
|
|
67
|
+
return this.capsuleStore.appendToCapsuleArray(contractAddress, baseSlot, content, jobId, scope);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
readCapsuleArray(contractAddress: AztecAddress, baseSlot: Fr, jobId: string, scope: AztecAddress): Promise<Fr[][]> {
|
|
71
|
+
assertAllowedScope(scope, this.allowedScopes);
|
|
72
|
+
return this.capsuleStore.readCapsuleArray(contractAddress, baseSlot, jobId, scope);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
setCapsuleArray(contractAddress: AztecAddress, baseSlot: Fr, content: Fr[][], jobId: string, scope: AztecAddress) {
|
|
76
|
+
assertAllowedScope(scope, this.allowedScopes);
|
|
77
|
+
return this.capsuleStore.setCapsuleArray(contractAddress, baseSlot, content, jobId, scope);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
function assertAllowedScope(scope: AztecAddress, allowedScopes: AztecAddress[]) {
|
|
82
|
+
if (scope.equals(AztecAddress.ZERO)) {
|
|
83
|
+
return;
|
|
84
|
+
}
|
|
85
|
+
if (!allowedScopes.some((allowed: AztecAddress) => allowed.equals(scope))) {
|
|
86
|
+
throw new Error(
|
|
87
|
+
`Scope ${scope.toString()} is not in the allowed scopes list: [${allowedScopes.map((s: AztecAddress) => s.toString()).join(', ')}]. See https://docs.aztec.network/errors/10`,
|
|
88
|
+
);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
2
2
|
import { type Logger, createLogger } from '@aztec/foundation/log';
|
|
3
3
|
import type { AztecAsyncKVStore, AztecAsyncMap } from '@aztec/kv-store';
|
|
4
|
-
import
|
|
4
|
+
import { AztecAddress } from '@aztec/stdlib/aztec-address';
|
|
5
5
|
|
|
6
6
|
import type { StagedStore } from '../../job_coordinator/job_coordinator.js';
|
|
7
7
|
|
|
@@ -10,11 +10,12 @@ export class CapsuleStore implements StagedStore {
|
|
|
10
10
|
|
|
11
11
|
#store: AztecAsyncKVStore;
|
|
12
12
|
|
|
13
|
-
// Arbitrary data stored by contracts. Key is computed as `${contractAddress}:${key}
|
|
13
|
+
// Arbitrary data stored by contracts. Key is computed as `${contractAddress}:${scope}:${key}`, using the zero
|
|
14
|
+
// address for the global scope.
|
|
14
15
|
#capsules: AztecAsyncMap<string, Buffer>;
|
|
15
16
|
|
|
16
|
-
// jobId => `${contractAddress}:${key}` => capsule data
|
|
17
|
-
// when `#stagedCapsules.get('some-job-id').get('${some-contract-address
|
|
17
|
+
// jobId => `${contractAddress}:${scope}:${key}` => capsule data
|
|
18
|
+
// when `#stagedCapsules.get('some-job-id').get('${some-contract-address}:${some-scope}:${some-key}') === null`,
|
|
18
19
|
// it signals that the capsule was deleted during the job, so it needs to be deleted on commit
|
|
19
20
|
#stagedCapsules: Map<string, Map<string, Buffer | null>>;
|
|
20
21
|
|
|
@@ -134,8 +135,8 @@ export class CapsuleStore implements StagedStore {
|
|
|
134
135
|
* to public contract storage in that it's indexed by the contract address and storage slot but instead of the global
|
|
135
136
|
* network state it's backed by local PXE db.
|
|
136
137
|
*/
|
|
137
|
-
|
|
138
|
-
const dbSlotKey = dbSlotToKey(contractAddress, slot);
|
|
138
|
+
setCapsule(contractAddress: AztecAddress, slot: Fr, capsule: Fr[], jobId: string, scope: AztecAddress) {
|
|
139
|
+
const dbSlotKey = dbSlotToKey(contractAddress, slot, scope);
|
|
139
140
|
|
|
140
141
|
// A store overrides any pre-existing data on the slot
|
|
141
142
|
this.#setOnStage(jobId, dbSlotKey, Buffer.concat(capsule.map(value => value.toBuffer())));
|
|
@@ -147,8 +148,18 @@ export class CapsuleStore implements StagedStore {
|
|
|
147
148
|
* @param slot - The slot in the database to read.
|
|
148
149
|
* @returns The stored data or `null` if no data is stored under the slot.
|
|
149
150
|
*/
|
|
150
|
-
|
|
151
|
-
|
|
151
|
+
getCapsule(contractAddress: AztecAddress, slot: Fr, jobId: string, scope: AztecAddress): Promise<Fr[] | null> {
|
|
152
|
+
return this.#store.transactionAsync(() => this.#getCapsuleInternal(contractAddress, slot, jobId, scope));
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
/** Same as getCapsule but without its own transaction, for use inside an existing transactionAsync. */
|
|
156
|
+
async #getCapsuleInternal(
|
|
157
|
+
contractAddress: AztecAddress,
|
|
158
|
+
slot: Fr,
|
|
159
|
+
jobId: string,
|
|
160
|
+
scope: AztecAddress,
|
|
161
|
+
): Promise<Fr[] | null> {
|
|
162
|
+
const dataBuffer = await this.#getFromStage(jobId, dbSlotToKey(contractAddress, slot, scope));
|
|
152
163
|
if (!dataBuffer) {
|
|
153
164
|
this.logger.trace(`Data not found for contract ${contractAddress.toString()} and slot ${slot.toString()}`);
|
|
154
165
|
return null;
|
|
@@ -165,9 +176,9 @@ export class CapsuleStore implements StagedStore {
|
|
|
165
176
|
* @param contractAddress - The contract address under which the data is scoped.
|
|
166
177
|
* @param slot - The slot in the database to delete.
|
|
167
178
|
*/
|
|
168
|
-
deleteCapsule(contractAddress: AztecAddress, slot: Fr, jobId: string) {
|
|
179
|
+
deleteCapsule(contractAddress: AztecAddress, slot: Fr, jobId: string, scope: AztecAddress) {
|
|
169
180
|
// When we commit this, we will interpret null as a deletion, so we'll propagate the delete to the KV store
|
|
170
|
-
this.#deleteOnStage(jobId, dbSlotToKey(contractAddress, slot));
|
|
181
|
+
this.#deleteOnStage(jobId, dbSlotToKey(contractAddress, slot, scope));
|
|
171
182
|
}
|
|
172
183
|
|
|
173
184
|
/**
|
|
@@ -187,6 +198,7 @@ export class CapsuleStore implements StagedStore {
|
|
|
187
198
|
dstSlot: Fr,
|
|
188
199
|
numEntries: number,
|
|
189
200
|
jobId: string,
|
|
201
|
+
scope: AztecAddress,
|
|
190
202
|
): Promise<void> {
|
|
191
203
|
// This transactional context gives us "copy atomicity":
|
|
192
204
|
// there shouldn't be concurrent writes to what's being copied here.
|
|
@@ -203,8 +215,8 @@ export class CapsuleStore implements StagedStore {
|
|
|
203
215
|
}
|
|
204
216
|
|
|
205
217
|
for (const i of indexes) {
|
|
206
|
-
const currentSrcSlot = dbSlotToKey(contractAddress, srcSlot.add(new Fr(i)));
|
|
207
|
-
const currentDstSlot = dbSlotToKey(contractAddress, dstSlot.add(new Fr(i)));
|
|
218
|
+
const currentSrcSlot = dbSlotToKey(contractAddress, srcSlot.add(new Fr(i)), scope);
|
|
219
|
+
const currentDstSlot = dbSlotToKey(contractAddress, dstSlot.add(new Fr(i)), scope);
|
|
208
220
|
|
|
209
221
|
const toCopy = await this.#getFromStage(jobId, currentSrcSlot);
|
|
210
222
|
if (!toCopy) {
|
|
@@ -224,7 +236,13 @@ export class CapsuleStore implements StagedStore {
|
|
|
224
236
|
* @param baseSlot - The slot where the array length is stored
|
|
225
237
|
* @param content - Array of capsule data to append
|
|
226
238
|
*/
|
|
227
|
-
appendToCapsuleArray(
|
|
239
|
+
appendToCapsuleArray(
|
|
240
|
+
contractAddress: AztecAddress,
|
|
241
|
+
baseSlot: Fr,
|
|
242
|
+
content: Fr[][],
|
|
243
|
+
jobId: string,
|
|
244
|
+
scope: AztecAddress,
|
|
245
|
+
): Promise<void> {
|
|
228
246
|
// We wrap this in a transaction to serialize concurrent calls from Promise.all.
|
|
229
247
|
// Without this, concurrent appends to the same array could race: both read length=0,
|
|
230
248
|
// both write at the same slots, one overwrites the other.
|
|
@@ -232,22 +250,22 @@ export class CapsuleStore implements StagedStore {
|
|
|
232
250
|
// and not using a transaction here would heavily impact performance.
|
|
233
251
|
return this.#store.transactionAsync(async () => {
|
|
234
252
|
// Load current length, defaulting to 0 if not found
|
|
235
|
-
const lengthData = await this
|
|
253
|
+
const lengthData = await this.#getCapsuleInternal(contractAddress, baseSlot, jobId, scope);
|
|
236
254
|
const currentLength = lengthData ? lengthData[0].toNumber() : 0;
|
|
237
255
|
|
|
238
256
|
// Store each capsule at consecutive slots after baseSlot + 1 + currentLength
|
|
239
257
|
for (let i = 0; i < content.length; i++) {
|
|
240
258
|
const nextSlot = arraySlot(baseSlot, currentLength + i);
|
|
241
|
-
this.
|
|
259
|
+
this.setCapsule(contractAddress, nextSlot, content[i], jobId, scope);
|
|
242
260
|
}
|
|
243
261
|
|
|
244
262
|
// Update length to include all new capsules
|
|
245
263
|
const newLength = currentLength + content.length;
|
|
246
|
-
this.
|
|
264
|
+
this.setCapsule(contractAddress, baseSlot, [new Fr(newLength)], jobId, scope);
|
|
247
265
|
});
|
|
248
266
|
}
|
|
249
267
|
|
|
250
|
-
readCapsuleArray(contractAddress: AztecAddress, baseSlot: Fr, jobId: string): Promise<Fr[][]> {
|
|
268
|
+
readCapsuleArray(contractAddress: AztecAddress, baseSlot: Fr, jobId: string, scope: AztecAddress): Promise<Fr[][]> {
|
|
251
269
|
// I'm leaving this transactional context here though because I'm assuming this
|
|
252
270
|
// gives us "read array atomicity": there shouldn't be concurrent writes to what's being copied
|
|
253
271
|
// here.
|
|
@@ -255,14 +273,14 @@ export class CapsuleStore implements StagedStore {
|
|
|
255
273
|
// of jobs: different calls running concurrently on the same contract may cause trouble.
|
|
256
274
|
return this.#store.transactionAsync(async () => {
|
|
257
275
|
// Load length, defaulting to 0 if not found
|
|
258
|
-
const maybeLength = await this
|
|
276
|
+
const maybeLength = await this.#getCapsuleInternal(contractAddress, baseSlot, jobId, scope);
|
|
259
277
|
const length = maybeLength ? maybeLength[0].toBigInt() : 0n;
|
|
260
278
|
|
|
261
279
|
const values: Fr[][] = [];
|
|
262
280
|
|
|
263
281
|
// Read each capsule at consecutive slots after baseSlot
|
|
264
282
|
for (let i = 0; i < length; i++) {
|
|
265
|
-
const currentValue = await this
|
|
283
|
+
const currentValue = await this.#getCapsuleInternal(contractAddress, arraySlot(baseSlot, i), jobId, scope);
|
|
266
284
|
if (currentValue == undefined) {
|
|
267
285
|
throw new Error(
|
|
268
286
|
`Expected non-empty value at capsule array in base slot ${baseSlot} at index ${i} for contract ${contractAddress}`,
|
|
@@ -276,7 +294,7 @@ export class CapsuleStore implements StagedStore {
|
|
|
276
294
|
});
|
|
277
295
|
}
|
|
278
296
|
|
|
279
|
-
setCapsuleArray(contractAddress: AztecAddress, baseSlot: Fr, content: Fr[][], jobId: string) {
|
|
297
|
+
setCapsuleArray(contractAddress: AztecAddress, baseSlot: Fr, content: Fr[][], jobId: string, scope: AztecAddress) {
|
|
280
298
|
// This transactional context in theory isn't so critical now because we aren't
|
|
281
299
|
// writing to DB so if there's exceptions midway and it blows up, no visible impact
|
|
282
300
|
// to persistent storage will happen.
|
|
@@ -287,27 +305,27 @@ export class CapsuleStore implements StagedStore {
|
|
|
287
305
|
// of jobs: different calls running concurrently on the same contract may cause trouble.
|
|
288
306
|
return this.#store.transactionAsync(async () => {
|
|
289
307
|
// Load current length, defaulting to 0 if not found
|
|
290
|
-
const maybeLength = await this
|
|
308
|
+
const maybeLength = await this.#getCapsuleInternal(contractAddress, baseSlot, jobId, scope);
|
|
291
309
|
const originalLength = maybeLength ? maybeLength[0].toNumber() : 0;
|
|
292
310
|
|
|
293
311
|
// Set the new length
|
|
294
|
-
this.
|
|
312
|
+
this.setCapsule(contractAddress, baseSlot, [new Fr(content.length)], jobId, scope);
|
|
295
313
|
|
|
296
314
|
// Store the new content, possibly overwriting existing values
|
|
297
315
|
for (let i = 0; i < content.length; i++) {
|
|
298
|
-
this.
|
|
316
|
+
this.setCapsule(contractAddress, arraySlot(baseSlot, i), content[i], jobId, scope);
|
|
299
317
|
}
|
|
300
318
|
|
|
301
319
|
// Clear any stragglers
|
|
302
320
|
for (let i = content.length; i < originalLength; i++) {
|
|
303
|
-
this.deleteCapsule(contractAddress, arraySlot(baseSlot, i), jobId);
|
|
321
|
+
this.deleteCapsule(contractAddress, arraySlot(baseSlot, i), jobId, scope);
|
|
304
322
|
}
|
|
305
323
|
});
|
|
306
324
|
}
|
|
307
325
|
}
|
|
308
326
|
|
|
309
|
-
function dbSlotToKey(contractAddress: AztecAddress, slot: Fr): string {
|
|
310
|
-
return
|
|
327
|
+
function dbSlotToKey(contractAddress: AztecAddress, slot: Fr, scope: AztecAddress): string {
|
|
328
|
+
return [contractAddress.toString(), scope.toString(), slot.toString()].join(':');
|
|
311
329
|
}
|
|
312
330
|
|
|
313
331
|
function arraySlot(baseSlot: Fr, index: number) {
|
|
@@ -168,12 +168,14 @@ export class ContractStore {
|
|
|
168
168
|
}
|
|
169
169
|
|
|
170
170
|
async addContractInstance(contract: ContractInstanceWithAddress): Promise<void> {
|
|
171
|
-
this.#
|
|
171
|
+
await this.#store.transactionAsync(async () => {
|
|
172
|
+
await this.#contractInstances.set(
|
|
173
|
+
contract.address.toString(),
|
|
174
|
+
new SerializableContractInstance(contract).toBuffer(),
|
|
175
|
+
);
|
|
176
|
+
});
|
|
172
177
|
|
|
173
|
-
|
|
174
|
-
contract.address.toString(),
|
|
175
|
-
new SerializableContractInstance(contract).toBuffer(),
|
|
176
|
-
);
|
|
178
|
+
this.#contractClassIdMap.set(contract.address.toString(), contract.currentContractClassId);
|
|
177
179
|
}
|
|
178
180
|
|
|
179
181
|
// Private getters
|
|
@@ -246,7 +248,7 @@ export class ContractStore {
|
|
|
246
248
|
contractClassId: Fr,
|
|
247
249
|
): Promise<(ContractClassWithId & ContractClassIdPreimage) | undefined> {
|
|
248
250
|
const key = contractClassId.toString();
|
|
249
|
-
const buf = await this.#contractClassData.getAsync(key);
|
|
251
|
+
const buf = await this.#store.transactionAsync(() => this.#contractClassData.getAsync(key));
|
|
250
252
|
if (!buf) {
|
|
251
253
|
return undefined;
|
|
252
254
|
}
|
package/src/storage/metadata.ts
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
export const PXE_DATA_SCHEMA_VERSION =
|
|
1
|
+
export const PXE_DATA_SCHEMA_VERSION = 5;
|
|
@@ -106,7 +106,7 @@ export class NoteStore implements StagedStore {
|
|
|
106
106
|
* returned once if this is the case)
|
|
107
107
|
*/
|
|
108
108
|
getNotes(filter: NotesFilter, jobId: string): Promise<NoteDao[]> {
|
|
109
|
-
if (filter.scopes
|
|
109
|
+
if (filter.scopes.length === 0) {
|
|
110
110
|
return Promise.resolve([]);
|
|
111
111
|
}
|
|
112
112
|
|
|
@@ -180,10 +180,7 @@ export class NoteStore implements StagedStore {
|
|
|
180
180
|
continue;
|
|
181
181
|
}
|
|
182
182
|
|
|
183
|
-
if (
|
|
184
|
-
filter.scopes !== 'ALL_SCOPES' &&
|
|
185
|
-
note.scopes.intersection(new Set(filter.scopes.map(s => s.toString()))).size === 0
|
|
186
|
-
) {
|
|
183
|
+
if (note.scopes.intersection(new Set(filter.scopes.map(s => s.toString()))).size === 0) {
|
|
187
184
|
continue;
|
|
188
185
|
}
|
|
189
186
|
|
|
@@ -234,6 +234,10 @@ export class PrivateEventStore implements StagedStore {
|
|
|
234
234
|
* IMPORTANT: This method must be called within a transaction to ensure atomicity.
|
|
235
235
|
*/
|
|
236
236
|
public async rollback(blockNumber: number, synchedBlockNumber: number): Promise<void> {
|
|
237
|
+
if (this.#eventsForJob.size > 0) {
|
|
238
|
+
throw new Error('PXE private event store rollback is not allowed while jobs are running');
|
|
239
|
+
}
|
|
240
|
+
|
|
237
241
|
// First pass: collect all event IDs for all blocks, starting reads during iteration to keep tx alive.
|
|
238
242
|
const eventsByBlock: Map<number, { eventId: string; eventReadPromise: Promise<Buffer | undefined> }[]> = new Map();
|
|
239
243
|
|