@aztec/txe 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.
Files changed (58) hide show
  1. package/dest/index.d.ts +1 -1
  2. package/dest/index.d.ts.map +1 -1
  3. package/dest/index.js +88 -54
  4. package/dest/oracle/interfaces.d.ts +29 -28
  5. package/dest/oracle/interfaces.d.ts.map +1 -1
  6. package/dest/oracle/txe_oracle_public_context.d.ts +13 -13
  7. package/dest/oracle/txe_oracle_public_context.d.ts.map +1 -1
  8. package/dest/oracle/txe_oracle_public_context.js +12 -12
  9. package/dest/oracle/txe_oracle_top_level_context.d.ts +23 -23
  10. package/dest/oracle/txe_oracle_top_level_context.d.ts.map +1 -1
  11. package/dest/oracle/txe_oracle_top_level_context.js +112 -54
  12. package/dest/rpc_translator.d.ts +87 -82
  13. package/dest/rpc_translator.d.ts.map +1 -1
  14. package/dest/rpc_translator.js +285 -152
  15. package/dest/state_machine/archiver.d.ts +3 -3
  16. package/dest/state_machine/archiver.d.ts.map +1 -1
  17. package/dest/state_machine/archiver.js +7 -8
  18. package/dest/state_machine/dummy_p2p_client.d.ts +16 -12
  19. package/dest/state_machine/dummy_p2p_client.d.ts.map +1 -1
  20. package/dest/state_machine/dummy_p2p_client.js +28 -16
  21. package/dest/state_machine/global_variable_builder.d.ts +3 -3
  22. package/dest/state_machine/global_variable_builder.d.ts.map +1 -1
  23. package/dest/state_machine/global_variable_builder.js +1 -1
  24. package/dest/state_machine/index.d.ts +6 -3
  25. package/dest/state_machine/index.d.ts.map +1 -1
  26. package/dest/state_machine/index.js +9 -5
  27. package/dest/state_machine/mock_epoch_cache.d.ts +17 -3
  28. package/dest/state_machine/mock_epoch_cache.d.ts.map +1 -1
  29. package/dest/state_machine/mock_epoch_cache.js +32 -2
  30. package/dest/state_machine/synchronizer.d.ts +5 -5
  31. package/dest/state_machine/synchronizer.d.ts.map +1 -1
  32. package/dest/state_machine/synchronizer.js +3 -3
  33. package/dest/txe_session.d.ts +10 -6
  34. package/dest/txe_session.d.ts.map +1 -1
  35. package/dest/txe_session.js +90 -25
  36. package/dest/util/encoding.d.ts +69 -1
  37. package/dest/util/encoding.d.ts.map +1 -1
  38. package/dest/util/txe_public_contract_data_source.d.ts +2 -3
  39. package/dest/util/txe_public_contract_data_source.d.ts.map +1 -1
  40. package/dest/util/txe_public_contract_data_source.js +6 -25
  41. package/package.json +15 -15
  42. package/src/index.ts +89 -52
  43. package/src/oracle/interfaces.ts +32 -31
  44. package/src/oracle/txe_oracle_public_context.ts +12 -12
  45. package/src/oracle/txe_oracle_top_level_context.ts +118 -92
  46. package/src/rpc_translator.ts +332 -174
  47. package/src/state_machine/archiver.ts +6 -5
  48. package/src/state_machine/dummy_p2p_client.ts +39 -22
  49. package/src/state_machine/global_variable_builder.ts +7 -1
  50. package/src/state_machine/index.ts +10 -1
  51. package/src/state_machine/mock_epoch_cache.ts +42 -3
  52. package/src/state_machine/synchronizer.ts +4 -4
  53. package/src/txe_session.ts +102 -73
  54. package/src/util/txe_public_contract_data_source.ts +10 -38
  55. package/dest/util/txe_contract_store.d.ts +0 -12
  56. package/dest/util/txe_contract_store.d.ts.map +0 -1
  57. package/dest/util/txe_contract_store.js +0 -22
  58. package/src/util/txe_contract_store.ts +0 -36
@@ -4,7 +4,7 @@ import { Schnorr } from '@aztec/foundation/crypto/schnorr';
4
4
  import { Fr } from '@aztec/foundation/curves/bn254';
5
5
  import { LogLevels, applyStringFormatting, createLogger } from '@aztec/foundation/log';
6
6
  import { TestDateProvider } from '@aztec/foundation/timer';
7
- import { ORACLE_VERSION, enrichPublicSimulationError } from '@aztec/pxe/server';
7
+ import { CapsuleService, ORACLE_VERSION, enrichPublicSimulationError } from '@aztec/pxe/server';
8
8
  import { ExecutionNoteCache, ExecutionTaggingIndexCache, HashedValuesCache, Oracle, PrivateExecutionOracle, UtilityExecutionOracle, executePrivateFunction, generateSimulatedProvingResult } from '@aztec/pxe/simulator';
9
9
  import { ExecutionError, WASMSimulator, createSimulationError, extractCallStack, resolveAssertionMessageFromError, toACVMWitness, witnessMapToFields } from '@aztec/simulator/client';
10
10
  import { CppPublicTxSimulator, GuardedMerkleTreeOperations, PublicContractsDB, PublicProcessor } from '@aztec/simulator/server';
@@ -36,15 +36,15 @@ export class TXEOracleTopLevelContext {
36
36
  senderAddressBookStore;
37
37
  capsuleStore;
38
38
  privateEventStore;
39
- jobId;
40
39
  nextBlockTimestamp;
41
40
  version;
42
41
  chainId;
43
42
  authwits;
43
+ contractSyncService;
44
44
  isMisc;
45
45
  isTxe;
46
46
  logger;
47
- constructor(stateMachine, contractStore, noteStore, keyStore, addressStore, accountStore, senderTaggingStore, recipientTaggingStore, senderAddressBookStore, capsuleStore, privateEventStore, jobId, nextBlockTimestamp, version, chainId, authwits){
47
+ constructor(stateMachine, contractStore, noteStore, keyStore, addressStore, accountStore, senderTaggingStore, recipientTaggingStore, senderAddressBookStore, capsuleStore, privateEventStore, nextBlockTimestamp, version, chainId, authwits, contractSyncService){
48
48
  this.stateMachine = stateMachine;
49
49
  this.contractStore = contractStore;
50
50
  this.noteStore = noteStore;
@@ -56,30 +56,31 @@ export class TXEOracleTopLevelContext {
56
56
  this.senderAddressBookStore = senderAddressBookStore;
57
57
  this.capsuleStore = capsuleStore;
58
58
  this.privateEventStore = privateEventStore;
59
- this.jobId = jobId;
60
59
  this.nextBlockTimestamp = nextBlockTimestamp;
61
60
  this.version = version;
62
61
  this.chainId = chainId;
63
62
  this.authwits = authwits;
63
+ this.contractSyncService = contractSyncService;
64
64
  this.isMisc = true;
65
65
  this.isTxe = true;
66
66
  this.logger = createLogger('txe:top_level_context');
67
67
  this.logger.debug('Entering Top Level Context');
68
68
  }
69
- utilityAssertCompatibleOracleVersion(version) {
69
+ assertCompatibleOracleVersion(version) {
70
70
  if (version !== ORACLE_VERSION) {
71
- throw new Error(`Incompatible oracle version. TXE is using version '${ORACLE_VERSION}', but got a request for '${version}'.`);
71
+ const hint = version > ORACLE_VERSION ? 'The contract was compiled with a newer version of Aztec.nr than this aztec cli version supports. Upgrade your aztec cli version to a compatible version.' : 'The contract was compiled with an older version of Aztec.nr than this aztec cli version supports. Recompile the contract with a compatible version of Aztec.nr.';
72
+ throw new Error(`Incompatible aztec cli version: ${hint} See https://docs.aztec.network/errors/8 (expected oracle version ${ORACLE_VERSION}, got ${version})`);
72
73
  }
73
74
  }
74
75
  // This is typically only invoked in private contexts, but it is convenient to also have it in top-level for testing
75
76
  // setup.
76
- utilityGetRandomField() {
77
+ getRandomField() {
77
78
  return Fr.random();
78
79
  }
79
80
  // We instruct users to debug contracts via this oracle, so it makes sense that they'd expect it to also work in tests
80
- utilityDebugLog(level, message, fields) {
81
+ log(level, message, fields) {
81
82
  if (!LogLevels[level]) {
82
- throw new Error(`Invalid debug log level: ${level}`);
83
+ throw new Error(`Invalid log level: ${level}`);
83
84
  }
84
85
  const levelName = LogLevels[level];
85
86
  this.logger[levelName](`${applyStringFormatting(message, fields)}`, {
@@ -87,19 +88,19 @@ export class TXEOracleTopLevelContext {
87
88
  });
88
89
  return Promise.resolve();
89
90
  }
90
- txeGetDefaultAddress() {
91
+ getDefaultAddress() {
91
92
  return DEFAULT_ADDRESS;
92
93
  }
93
- async txeGetNextBlockNumber() {
94
+ async getNextBlockNumber() {
94
95
  return BlockNumber(await this.getLastBlockNumber() + 1);
95
96
  }
96
- txeGetNextBlockTimestamp() {
97
+ getNextBlockTimestamp() {
97
98
  return Promise.resolve(this.nextBlockTimestamp);
98
99
  }
99
- async txeGetLastBlockTimestamp() {
100
+ async getLastBlockTimestamp() {
100
101
  return (await this.stateMachine.node.getBlockHeader('latest')).globalVariables.timestamp;
101
102
  }
102
- async txeGetLastTxEffects() {
103
+ async getLastTxEffects() {
103
104
  const latestBlockNumber = await this.stateMachine.archiver.getBlockNumber();
104
105
  const block = await this.stateMachine.archiver.getBlock(latestBlockNumber);
105
106
  if (block.body.txEffects.length != 1) {
@@ -113,7 +114,19 @@ export class TXEOracleTopLevelContext {
113
114
  nullifiers: txEffects.nullifiers
114
115
  };
115
116
  }
116
- async txeGetPrivateEvents(selector, contractAddress, scope) {
117
+ async syncContractNonOracleMethod(contractAddress, scope, jobId) {
118
+ if (contractAddress.equals(DEFAULT_ADDRESS)) {
119
+ this.logger.debug(`Skipping sync in getPrivateEvents because the events correspond to the default address.`);
120
+ return;
121
+ }
122
+ const blockHeader = await this.stateMachine.anchorBlockStore.getBlockHeader();
123
+ await this.stateMachine.contractSyncService.ensureContractSynced(contractAddress, null, async (call, execScopes)=>{
124
+ await this.executeUtilityCall(call, execScopes, jobId);
125
+ }, blockHeader, jobId, [
126
+ scope
127
+ ]);
128
+ }
129
+ async getPrivateEvents(selector, contractAddress, scope) {
117
130
  return (await this.privateEventStore.getPrivateEvents(selector, {
118
131
  contractAddress,
119
132
  scopes: [
@@ -123,17 +136,17 @@ export class TXEOracleTopLevelContext {
123
136
  toBlock: await this.getLastBlockNumber() + 1
124
137
  })).map((e)=>e.packedEvent);
125
138
  }
126
- async txeAdvanceBlocksBy(blocks) {
139
+ async advanceBlocksBy(blocks) {
127
140
  this.logger.debug(`time traveling ${blocks} blocks`);
128
141
  for(let i = 0; i < blocks; i++){
129
142
  await this.mineBlock();
130
143
  }
131
144
  }
132
- txeAdvanceTimestampBy(duration) {
145
+ advanceTimestampBy(duration) {
133
146
  this.logger.debug(`time traveling ${duration} seconds`);
134
147
  this.nextBlockTimestamp += duration;
135
148
  }
136
- async txeDeploy(artifact, instance, secret) {
149
+ async deploy(artifact, instance, secret) {
137
150
  // Emit deployment nullifier
138
151
  await this.mineBlock({
139
152
  nullifiers: [
@@ -141,25 +154,25 @@ export class TXEOracleTopLevelContext {
141
154
  ]
142
155
  });
143
156
  if (!secret.equals(Fr.ZERO)) {
144
- await this.txeAddAccount(artifact, instance, secret);
157
+ await this.addAccount(artifact, instance, secret);
145
158
  } else {
146
159
  await this.contractStore.addContractInstance(instance);
147
- await this.contractStore.addContractArtifact(instance.currentContractClassId, artifact);
160
+ await this.contractStore.addContractArtifact(artifact);
148
161
  this.logger.debug(`Deployed ${artifact.name} at ${instance.address}`);
149
162
  }
150
163
  }
151
- async txeAddAccount(artifact, instance, secret) {
164
+ async addAccount(artifact, instance, secret) {
152
165
  const partialAddress = await computePartialAddress(instance);
153
166
  this.logger.debug(`Deployed ${artifact.name} at ${instance.address}`);
154
167
  await this.contractStore.addContractInstance(instance);
155
- await this.contractStore.addContractArtifact(instance.currentContractClassId, artifact);
168
+ await this.contractStore.addContractArtifact(artifact);
156
169
  const completeAddress = await this.keyStore.addAccount(secret, partialAddress);
157
170
  await this.accountStore.setAccount(completeAddress.address, completeAddress);
158
171
  await this.addressStore.addCompleteAddress(completeAddress);
159
172
  this.logger.debug(`Created account ${completeAddress.address}`);
160
173
  return completeAddress;
161
174
  }
162
- async txeCreateAccount(secret) {
175
+ async createAccount(secret) {
163
176
  // This is a foot gun !
164
177
  const completeAddress = await this.keyStore.addAccount(secret, secret);
165
178
  await this.accountStore.setAccount(completeAddress.address, completeAddress);
@@ -167,7 +180,7 @@ export class TXEOracleTopLevelContext {
167
180
  this.logger.debug(`Created account ${completeAddress.address}`);
168
181
  return completeAddress;
169
182
  }
170
- async txeAddAuthWitness(address, messageHash) {
183
+ async addAuthWitness(address, messageHash) {
171
184
  const account = await this.accountStore.getAccount(address);
172
185
  const privateKey = await this.keyStore.getMasterSecretKey(account.publicKeys.masterIncomingViewingPublicKey);
173
186
  const schnorr = new Schnorr();
@@ -178,7 +191,7 @@ export class TXEOracleTopLevelContext {
178
191
  this.authwits.set(authWitness.requestHash.toString(), authWitness);
179
192
  }
180
193
  async mineBlock(options = {}) {
181
- const blockNumber = await this.txeGetNextBlockNumber();
194
+ const blockNumber = await this.getNextBlockNumber();
182
195
  const txEffect = TxEffect.empty();
183
196
  txEffect.nullifiers = [
184
197
  getSingleTxBlockRequestHash(blockNumber),
@@ -200,26 +213,25 @@ export class TXEOracleTopLevelContext {
200
213
  this.logger.info(`Created block ${blockNumber} with timestamp ${block.header.globalVariables.timestamp}`);
201
214
  await this.stateMachine.handleL2Block(block);
202
215
  }
203
- async txePrivateCallNewFlow(from, targetContractAddress = AztecAddress.zero(), functionSelector = FunctionSelector.empty(), args, argsHash = Fr.zero(), isStaticCall = false) {
216
+ async privateCallNewFlow(from, targetContractAddress = AztecAddress.zero(), functionSelector = FunctionSelector.empty(), args, argsHash = Fr.zero(), isStaticCall = false, jobId) {
204
217
  this.logger.verbose(`Executing external function ${await this.contractStore.getDebugFunctionName(targetContractAddress, functionSelector)}@${targetContractAddress} isStaticCall=${isStaticCall}`);
205
218
  const artifact = await this.contractStore.getFunctionArtifact(targetContractAddress, functionSelector);
206
219
  if (!artifact) {
207
220
  const message = functionSelector.equals(await FunctionSelector.fromSignature('verify_private_authwit(Field)')) ? 'Found no account contract artifact for a private authwit check - use `create_contract_account` instead of `create_light_account` for authwit support.' : 'Function Artifact does not exist';
208
221
  throw new Error(message);
209
222
  }
210
- // When `from` is the zero address (used when creating a new contract account for example),
211
- // we disable scope filtering by setting effectiveScopes to undefined. This allows these operations
212
- // to proceed without requiring keys registered for the zero address.
213
- const effectiveScopes = from.isZero() ? undefined : [
223
+ // When `from` is the zero address (e.g. when deploying a new account contract), we return an
224
+ // empty scope list which acts as deny-all: no notes are visible and no keys are accessible.
225
+ const effectiveScopes = from.isZero() ? [] : [
214
226
  from
215
227
  ];
216
228
  // Sync notes before executing private function to discover notes from previous transactions
217
- const utilityExecutor = async (call)=>{
218
- await this.executeUtilityCall(call, effectiveScopes);
229
+ const utilityExecutor = async (call, execScopes)=>{
230
+ await this.executeUtilityCall(call, execScopes, jobId);
219
231
  };
220
232
  const blockHeader = await this.stateMachine.anchorBlockStore.getBlockHeader();
221
- await this.stateMachine.contractSyncService.ensureContractSynced(targetContractAddress, functionSelector, utilityExecutor, blockHeader, this.jobId);
222
- const blockNumber = await this.txeGetNextBlockNumber();
233
+ await this.stateMachine.contractSyncService.ensureContractSynced(targetContractAddress, functionSelector, utilityExecutor, blockHeader, jobId, effectiveScopes);
234
+ const blockNumber = await this.getNextBlockNumber();
223
235
  const callContext = new CallContext(from, targetContractAddress, functionSelector, isStaticCall);
224
236
  const gasLimits = new Gas(DEFAULT_DA_GAS_LIMIT, DEFAULT_L2_GAS_LIMIT);
225
237
  const teardownGasLimits = new Gas(DEFAULT_TEARDOWN_DA_GAS_LIMIT, DEFAULT_TEARDOWN_L2_GAS_LIMIT);
@@ -234,12 +246,40 @@ export class TXEOracleTopLevelContext {
234
246
  await noteCache.setMinRevertibleSideEffectCounter(minRevertibleSideEffectCounter);
235
247
  const taggingIndexCache = new ExecutionTaggingIndexCache();
236
248
  const simulator = new WASMSimulator();
237
- const privateExecutionOracle = new PrivateExecutionOracle(argsHash, txContext, callContext, /** Header of a block whose state is used during private execution (not the block the transaction is included in). */ blockHeader, utilityExecutor, /** List of transient auth witnesses to be used during this simulation */ Array.from(this.authwits.values()), /** List of transient auth witnesses to be used during this simulation */ [], HashedValuesCache.create([
238
- new HashedValues(args, argsHash)
239
- ]), noteCache, taggingIndexCache, this.contractStore, this.noteStore, this.keyStore, this.addressStore, this.stateMachine.node, this.senderTaggingStore, this.recipientTaggingStore, this.senderAddressBookStore, this.capsuleStore, this.privateEventStore, this.stateMachine.contractSyncService, this.jobId, 0, minRevertibleSideEffectCounter, undefined, effectiveScopes, /**
240
- * In TXE, the typical transaction entrypoint is skipped, so we need to simulate the actions that such a
241
- * contract would perform, including setting senderForTags.
242
- */ from, simulator);
249
+ const privateExecutionOracle = new PrivateExecutionOracle({
250
+ argsHash,
251
+ txContext,
252
+ callContext,
253
+ anchorBlockHeader: blockHeader,
254
+ utilityExecutor,
255
+ authWitnesses: Array.from(this.authwits.values()),
256
+ capsules: [],
257
+ executionCache: HashedValuesCache.create([
258
+ new HashedValues(args, argsHash)
259
+ ]),
260
+ noteCache,
261
+ taggingIndexCache,
262
+ contractStore: this.contractStore,
263
+ noteStore: this.noteStore,
264
+ keyStore: this.keyStore,
265
+ addressStore: this.addressStore,
266
+ aztecNode: this.stateMachine.node,
267
+ senderTaggingStore: this.senderTaggingStore,
268
+ recipientTaggingStore: this.recipientTaggingStore,
269
+ senderAddressBookStore: this.senderAddressBookStore,
270
+ capsuleService: new CapsuleService(this.capsuleStore, effectiveScopes),
271
+ privateEventStore: this.privateEventStore,
272
+ contractSyncService: this.stateMachine.contractSyncService,
273
+ jobId,
274
+ totalPublicCalldataCount: 0,
275
+ sideEffectCounter: minRevertibleSideEffectCounter,
276
+ scopes: effectiveScopes,
277
+ // In TXE, the typical transaction entrypoint is skipped, so we need to simulate the actions that such a
278
+ // contract would perform, including setting senderForTags.
279
+ senderForTags: from,
280
+ simulator,
281
+ messageContextService: this.stateMachine.messageContextService
282
+ });
243
283
  // Note: This is a slight modification of simulator.run without any of the checks. Maybe we should modify simulator.run with a boolean value to skip checks.
244
284
  let result;
245
285
  let executionResult;
@@ -251,7 +291,7 @@ export class TXEOracleTopLevelContext {
251
291
  r.publicInputs.publicTeardownCallRequest
252
292
  ]));
253
293
  const publicFunctionsCalldata = await Promise.all(publicCallRequests.map(async (r)=>{
254
- const calldata = await privateExecutionOracle.privateLoadFromExecutionCache(r.calldataHash);
294
+ const calldata = await privateExecutionOracle.getHashPreimage(r.calldataHash);
255
295
  return new HashedValues(calldata, r.calldataHash);
256
296
  }));
257
297
  noteCache.finish();
@@ -263,7 +303,7 @@ export class TXEOracleTopLevelContext {
263
303
  // According to the protocol rules, there must be at least one nullifier in the tx. The first nullifier is used as
264
304
  // the nonce generator for the note hashes.
265
305
  // We pass the non-zero minRevertibleSideEffectCounter to make sure the side effects are split correctly.
266
- const { publicInputs } = await generateSimulatedProvingResult(result, (addr, sel)=>this.contractStore.getDebugFunctionName(addr, sel), minRevertibleSideEffectCounter);
306
+ const { publicInputs } = await generateSimulatedProvingResult(result, (addr, sel)=>this.contractStore.getDebugFunctionName(addr, sel), this.stateMachine.node, minRevertibleSideEffectCounter);
267
307
  const globals = makeGlobalVariables();
268
308
  globals.blockNumber = blockNumber;
269
309
  globals.timestamp = this.nextBlockTimestamp;
@@ -331,9 +371,9 @@ export class TXEOracleTopLevelContext {
331
371
  await forkedWorldTrees.close();
332
372
  return executionResult.returnValues ?? [];
333
373
  }
334
- async txePublicCallNewFlow(from, targetContractAddress, calldata, isStaticCall) {
374
+ async publicCallNewFlow(from, targetContractAddress, calldata, isStaticCall) {
335
375
  this.logger.verbose(`Executing public function ${await this.contractStore.getDebugFunctionName(targetContractAddress, FunctionSelector.fromField(calldata[0]))}@${targetContractAddress} isStaticCall=${isStaticCall}`);
336
- const blockNumber = await this.txeGetNextBlockNumber();
376
+ const blockNumber = await this.getNextBlockNumber();
337
377
  const gasLimits = new Gas(DEFAULT_DA_GAS_LIMIT, DEFAULT_L2_GAS_LIMIT);
338
378
  const teardownGasLimits = new Gas(DEFAULT_TEARDOWN_DA_GAS_LIMIT, DEFAULT_TEARDOWN_L2_GAS_LIMIT);
339
379
  const gasSettings = new GasSettings(gasLimits, teardownGasLimits, GasFees.empty(), GasFees.empty());
@@ -372,7 +412,7 @@ export class TXEOracleTopLevelContext {
372
412
  revertibleAccumulatedData.publicCallRequests[0] = new PublicCallRequest(from, targetContractAddress, isStaticCall, calldataHash);
373
413
  const inputsForPublic = new PartialPrivateTailPublicInputsForPublic(nonRevertibleAccumulatedData, revertibleAccumulatedData, PublicCallRequest.empty());
374
414
  const constantData = new TxConstantData(anchorBlockHeader, txContext, Fr.zero(), Fr.zero());
375
- const txData = new PrivateKernelTailCircuitPublicInputs(constantData, /*gasUsed=*/ new Gas(0, 0), /*feePayer=*/ AztecAddress.zero(), /*includeByTimestamp=*/ 0n, inputsForPublic, undefined);
415
+ const txData = new PrivateKernelTailCircuitPublicInputs(constantData, /*gasUsed=*/ new Gas(0, 0), /*feePayer=*/ AztecAddress.zero(), /*expirationTimestamp=*/ 0n, inputsForPublic, undefined);
376
416
  const tx = await Tx.create({
377
417
  data: txData,
378
418
  chonkProof: ChonkProof.empty(),
@@ -425,16 +465,16 @@ export class TXEOracleTopLevelContext {
425
465
  await forkedWorldTrees.close();
426
466
  return returnValues ?? [];
427
467
  }
428
- async txeSimulateUtilityFunction(targetContractAddress, functionSelector, args) {
468
+ async executeUtilityFunction(targetContractAddress, functionSelector, args, jobId) {
429
469
  const artifact = await this.contractStore.getFunctionArtifact(targetContractAddress, functionSelector);
430
470
  if (!artifact) {
431
471
  throw new Error(`Cannot call ${functionSelector} as there is no artifact found at ${targetContractAddress}.`);
432
472
  }
433
473
  // Sync notes before executing utility function to discover notes from previous transactions
434
474
  const blockHeader = await this.stateMachine.anchorBlockStore.getBlockHeader();
435
- await this.stateMachine.contractSyncService.ensureContractSynced(targetContractAddress, functionSelector, async (call)=>{
436
- await this.executeUtilityCall(call);
437
- }, blockHeader, this.jobId);
475
+ await this.stateMachine.contractSyncService.ensureContractSynced(targetContractAddress, functionSelector, async (call, execScopes)=>{
476
+ await this.executeUtilityCall(call, execScopes, jobId);
477
+ }, blockHeader, jobId, 'ALL_SCOPES');
438
478
  const call = FunctionCall.from({
439
479
  name: artifact.name,
440
480
  to: targetContractAddress,
@@ -445,9 +485,9 @@ export class TXEOracleTopLevelContext {
445
485
  args,
446
486
  returnTypes: []
447
487
  });
448
- return this.executeUtilityCall(call);
488
+ return this.executeUtilityCall(call, 'ALL_SCOPES', jobId);
449
489
  }
450
- async executeUtilityCall(call, scopes) {
490
+ async executeUtilityCall(call, scopes, jobId) {
451
491
  const entryPointArtifact = await this.contractStore.getFunctionArtifactWithDebugMetadata(call.to, call.selector);
452
492
  if (entryPointArtifact.functionType !== FunctionType.UTILITY) {
453
493
  throw new Error(`Cannot run ${entryPointArtifact.functionType} function as utility`);
@@ -458,7 +498,25 @@ export class TXEOracleTopLevelContext {
458
498
  });
459
499
  try {
460
500
  const anchorBlockHeader = await this.stateMachine.anchorBlockStore.getBlockHeader();
461
- const oracle = new UtilityExecutionOracle(call.to, [], [], anchorBlockHeader, this.contractStore, this.noteStore, this.keyStore, this.addressStore, this.stateMachine.node, this.recipientTaggingStore, this.senderAddressBookStore, this.capsuleStore, this.privateEventStore, this.jobId, undefined, scopes);
501
+ const oracle = new UtilityExecutionOracle({
502
+ contractAddress: call.to,
503
+ authWitnesses: [],
504
+ capsules: [],
505
+ anchorBlockHeader,
506
+ contractStore: this.contractStore,
507
+ noteStore: this.noteStore,
508
+ keyStore: this.keyStore,
509
+ addressStore: this.addressStore,
510
+ aztecNode: this.stateMachine.node,
511
+ recipientTaggingStore: this.recipientTaggingStore,
512
+ senderAddressBookStore: this.senderAddressBookStore,
513
+ capsuleService: new CapsuleService(this.capsuleStore, scopes),
514
+ privateEventStore: this.privateEventStore,
515
+ messageContextService: this.stateMachine.messageContextService,
516
+ contractSyncService: this.contractSyncService,
517
+ jobId,
518
+ scopes
519
+ });
462
520
  const acirExecutionResult = await new WASMSimulator().executeUserCircuit(toACVMWitness(0, call.args), entryPointArtifact, new Oracle(oracle).toACIRCallback()).catch((err)=>{
463
521
  err.message = resolveAssertionMessageFromError(err, entryPointArtifact);
464
522
  throw new ExecutionError(err.message, {
@@ -468,10 +526,10 @@ export class TXEOracleTopLevelContext {
468
526
  cause: err
469
527
  });
470
528
  });
471
- this.logger.verbose(`Utility simulation for ${call.to}.${call.selector} completed`);
529
+ this.logger.verbose(`Utility execution for ${call.to}.${call.selector} completed`);
472
530
  return witnessMapToFields(acirExecutionResult.returnWitness);
473
531
  } catch (err) {
474
- throw createSimulationError(err instanceof Error ? err : new Error('Unknown error during utility simulation'));
532
+ throw createSimulationError(err instanceof Error ? err : new Error('Unknown error during utility execution'));
475
533
  }
476
534
  }
477
535
  close() {