@aztec/txe 0.0.1-commit.f295ac2 → 0.0.1-commit.f504929
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/index.d.ts +1 -1
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +88 -54
- package/dest/oracle/interfaces.d.ts +29 -28
- package/dest/oracle/interfaces.d.ts.map +1 -1
- package/dest/oracle/txe_oracle_public_context.d.ts +15 -15
- package/dest/oracle/txe_oracle_public_context.d.ts.map +1 -1
- package/dest/oracle/txe_oracle_public_context.js +16 -16
- package/dest/oracle/txe_oracle_top_level_context.d.ts +22 -23
- package/dest/oracle/txe_oracle_top_level_context.d.ts.map +1 -1
- package/dest/oracle/txe_oracle_top_level_context.js +125 -54
- package/dest/rpc_translator.d.ts +87 -81
- package/dest/rpc_translator.d.ts.map +1 -1
- package/dest/rpc_translator.js +287 -170
- package/dest/state_machine/archiver.d.ts +2 -2
- package/dest/state_machine/archiver.d.ts.map +1 -1
- package/dest/state_machine/archiver.js +7 -6
- package/dest/state_machine/dummy_p2p_client.d.ts +16 -12
- package/dest/state_machine/dummy_p2p_client.d.ts.map +1 -1
- package/dest/state_machine/dummy_p2p_client.js +28 -16
- package/dest/state_machine/index.d.ts +7 -7
- package/dest/state_machine/index.d.ts.map +1 -1
- package/dest/state_machine/index.js +31 -17
- package/dest/state_machine/mock_epoch_cache.d.ts +8 -6
- package/dest/state_machine/mock_epoch_cache.d.ts.map +1 -1
- package/dest/state_machine/mock_epoch_cache.js +9 -6
- package/dest/state_machine/synchronizer.d.ts +3 -3
- package/dest/state_machine/synchronizer.d.ts.map +1 -1
- package/dest/txe_session.d.ts +9 -6
- package/dest/txe_session.d.ts.map +1 -1
- package/dest/txe_session.js +86 -26
- package/dest/util/txe_public_contract_data_source.d.ts +2 -3
- package/dest/util/txe_public_contract_data_source.d.ts.map +1 -1
- package/dest/util/txe_public_contract_data_source.js +5 -22
- package/dest/utils/block_creation.d.ts +5 -5
- package/dest/utils/block_creation.d.ts.map +1 -1
- package/dest/utils/block_creation.js +7 -5
- package/package.json +15 -15
- package/src/index.ts +89 -52
- package/src/oracle/interfaces.ts +32 -31
- package/src/oracle/txe_oracle_public_context.ts +18 -20
- package/src/oracle/txe_oracle_top_level_context.ts +155 -102
- package/src/rpc_translator.ts +303 -177
- package/src/state_machine/archiver.ts +6 -8
- package/src/state_machine/dummy_p2p_client.ts +40 -22
- package/src/state_machine/index.ts +49 -19
- package/src/state_machine/mock_epoch_cache.ts +10 -11
- package/src/state_machine/synchronizer.ts +2 -2
- package/src/txe_session.ts +101 -85
- package/src/util/txe_public_contract_data_source.ts +10 -36
- package/src/utils/block_creation.ts +8 -6
- package/dest/util/txe_contract_store.d.ts +0 -12
- package/dest/util/txe_contract_store.d.ts.map +0 -1
- package/dest/util/txe_contract_store.js +0 -22
- package/src/util/txe_contract_store.ts +0 -36
|
@@ -36,7 +36,6 @@ export class TXEOracleTopLevelContext {
|
|
|
36
36
|
senderAddressBookStore;
|
|
37
37
|
capsuleStore;
|
|
38
38
|
privateEventStore;
|
|
39
|
-
jobId;
|
|
40
39
|
nextBlockTimestamp;
|
|
41
40
|
version;
|
|
42
41
|
chainId;
|
|
@@ -44,7 +43,7 @@ export class TXEOracleTopLevelContext {
|
|
|
44
43
|
isMisc;
|
|
45
44
|
isTxe;
|
|
46
45
|
logger;
|
|
47
|
-
constructor(stateMachine, contractStore, noteStore, keyStore, addressStore, accountStore, senderTaggingStore, recipientTaggingStore, senderAddressBookStore, capsuleStore, privateEventStore,
|
|
46
|
+
constructor(stateMachine, contractStore, noteStore, keyStore, addressStore, accountStore, senderTaggingStore, recipientTaggingStore, senderAddressBookStore, capsuleStore, privateEventStore, nextBlockTimestamp, version, chainId, authwits){
|
|
48
47
|
this.stateMachine = stateMachine;
|
|
49
48
|
this.contractStore = contractStore;
|
|
50
49
|
this.noteStore = noteStore;
|
|
@@ -56,7 +55,6 @@ export class TXEOracleTopLevelContext {
|
|
|
56
55
|
this.senderAddressBookStore = senderAddressBookStore;
|
|
57
56
|
this.capsuleStore = capsuleStore;
|
|
58
57
|
this.privateEventStore = privateEventStore;
|
|
59
|
-
this.jobId = jobId;
|
|
60
58
|
this.nextBlockTimestamp = nextBlockTimestamp;
|
|
61
59
|
this.version = version;
|
|
62
60
|
this.chainId = chainId;
|
|
@@ -66,39 +64,40 @@ export class TXEOracleTopLevelContext {
|
|
|
66
64
|
this.logger = createLogger('txe:top_level_context');
|
|
67
65
|
this.logger.debug('Entering Top Level Context');
|
|
68
66
|
}
|
|
69
|
-
|
|
67
|
+
assertCompatibleOracleVersion(version) {
|
|
70
68
|
if (version !== ORACLE_VERSION) {
|
|
71
69
|
throw new Error(`Incompatible oracle version. TXE is using version '${ORACLE_VERSION}', but got a request for '${version}'.`);
|
|
72
70
|
}
|
|
73
71
|
}
|
|
74
72
|
// This is typically only invoked in private contexts, but it is convenient to also have it in top-level for testing
|
|
75
73
|
// setup.
|
|
76
|
-
|
|
74
|
+
getRandomField() {
|
|
77
75
|
return Fr.random();
|
|
78
76
|
}
|
|
79
77
|
// We instruct users to debug contracts via this oracle, so it makes sense that they'd expect it to also work in tests
|
|
80
|
-
|
|
78
|
+
log(level, message, fields) {
|
|
81
79
|
if (!LogLevels[level]) {
|
|
82
|
-
throw new Error(`Invalid
|
|
80
|
+
throw new Error(`Invalid log level: ${level}`);
|
|
83
81
|
}
|
|
84
82
|
const levelName = LogLevels[level];
|
|
85
83
|
this.logger[levelName](`${applyStringFormatting(message, fields)}`, {
|
|
86
84
|
module: `${this.logger.module}:debug_log`
|
|
87
85
|
});
|
|
86
|
+
return Promise.resolve();
|
|
88
87
|
}
|
|
89
|
-
|
|
88
|
+
getDefaultAddress() {
|
|
90
89
|
return DEFAULT_ADDRESS;
|
|
91
90
|
}
|
|
92
|
-
async
|
|
91
|
+
async getNextBlockNumber() {
|
|
93
92
|
return BlockNumber(await this.getLastBlockNumber() + 1);
|
|
94
93
|
}
|
|
95
|
-
|
|
94
|
+
getNextBlockTimestamp() {
|
|
96
95
|
return Promise.resolve(this.nextBlockTimestamp);
|
|
97
96
|
}
|
|
98
|
-
async
|
|
97
|
+
async getLastBlockTimestamp() {
|
|
99
98
|
return (await this.stateMachine.node.getBlockHeader('latest')).globalVariables.timestamp;
|
|
100
99
|
}
|
|
101
|
-
async
|
|
100
|
+
async getLastTxEffects() {
|
|
102
101
|
const latestBlockNumber = await this.stateMachine.archiver.getBlockNumber();
|
|
103
102
|
const block = await this.stateMachine.archiver.getBlock(latestBlockNumber);
|
|
104
103
|
if (block.body.txEffects.length != 1) {
|
|
@@ -112,7 +111,19 @@ export class TXEOracleTopLevelContext {
|
|
|
112
111
|
nullifiers: txEffects.nullifiers
|
|
113
112
|
};
|
|
114
113
|
}
|
|
115
|
-
async
|
|
114
|
+
async syncContractNonOracleMethod(contractAddress, scope, jobId) {
|
|
115
|
+
if (contractAddress.equals(DEFAULT_ADDRESS)) {
|
|
116
|
+
this.logger.debug(`Skipping sync in getPrivateEvents because the events correspond to the default address.`);
|
|
117
|
+
return;
|
|
118
|
+
}
|
|
119
|
+
const blockHeader = await this.stateMachine.anchorBlockStore.getBlockHeader();
|
|
120
|
+
await this.stateMachine.contractSyncService.ensureContractSynced(contractAddress, null, async (call, execScopes)=>{
|
|
121
|
+
await this.executeUtilityCall(call, execScopes, jobId);
|
|
122
|
+
}, blockHeader, jobId, [
|
|
123
|
+
scope
|
|
124
|
+
]);
|
|
125
|
+
}
|
|
126
|
+
async getPrivateEvents(selector, contractAddress, scope) {
|
|
116
127
|
return (await this.privateEventStore.getPrivateEvents(selector, {
|
|
117
128
|
contractAddress,
|
|
118
129
|
scopes: [
|
|
@@ -122,17 +133,17 @@ export class TXEOracleTopLevelContext {
|
|
|
122
133
|
toBlock: await this.getLastBlockNumber() + 1
|
|
123
134
|
})).map((e)=>e.packedEvent);
|
|
124
135
|
}
|
|
125
|
-
async
|
|
136
|
+
async advanceBlocksBy(blocks) {
|
|
126
137
|
this.logger.debug(`time traveling ${blocks} blocks`);
|
|
127
138
|
for(let i = 0; i < blocks; i++){
|
|
128
139
|
await this.mineBlock();
|
|
129
140
|
}
|
|
130
141
|
}
|
|
131
|
-
|
|
142
|
+
advanceTimestampBy(duration) {
|
|
132
143
|
this.logger.debug(`time traveling ${duration} seconds`);
|
|
133
144
|
this.nextBlockTimestamp += duration;
|
|
134
145
|
}
|
|
135
|
-
async
|
|
146
|
+
async deploy(artifact, instance, secret) {
|
|
136
147
|
// Emit deployment nullifier
|
|
137
148
|
await this.mineBlock({
|
|
138
149
|
nullifiers: [
|
|
@@ -140,25 +151,25 @@ export class TXEOracleTopLevelContext {
|
|
|
140
151
|
]
|
|
141
152
|
});
|
|
142
153
|
if (!secret.equals(Fr.ZERO)) {
|
|
143
|
-
await this.
|
|
154
|
+
await this.addAccount(artifact, instance, secret);
|
|
144
155
|
} else {
|
|
145
156
|
await this.contractStore.addContractInstance(instance);
|
|
146
|
-
await this.contractStore.addContractArtifact(
|
|
157
|
+
await this.contractStore.addContractArtifact(artifact);
|
|
147
158
|
this.logger.debug(`Deployed ${artifact.name} at ${instance.address}`);
|
|
148
159
|
}
|
|
149
160
|
}
|
|
150
|
-
async
|
|
161
|
+
async addAccount(artifact, instance, secret) {
|
|
151
162
|
const partialAddress = await computePartialAddress(instance);
|
|
152
163
|
this.logger.debug(`Deployed ${artifact.name} at ${instance.address}`);
|
|
153
164
|
await this.contractStore.addContractInstance(instance);
|
|
154
|
-
await this.contractStore.addContractArtifact(
|
|
165
|
+
await this.contractStore.addContractArtifact(artifact);
|
|
155
166
|
const completeAddress = await this.keyStore.addAccount(secret, partialAddress);
|
|
156
167
|
await this.accountStore.setAccount(completeAddress.address, completeAddress);
|
|
157
168
|
await this.addressStore.addCompleteAddress(completeAddress);
|
|
158
169
|
this.logger.debug(`Created account ${completeAddress.address}`);
|
|
159
170
|
return completeAddress;
|
|
160
171
|
}
|
|
161
|
-
async
|
|
172
|
+
async createAccount(secret) {
|
|
162
173
|
// This is a foot gun !
|
|
163
174
|
const completeAddress = await this.keyStore.addAccount(secret, secret);
|
|
164
175
|
await this.accountStore.setAccount(completeAddress.address, completeAddress);
|
|
@@ -166,7 +177,7 @@ export class TXEOracleTopLevelContext {
|
|
|
166
177
|
this.logger.debug(`Created account ${completeAddress.address}`);
|
|
167
178
|
return completeAddress;
|
|
168
179
|
}
|
|
169
|
-
async
|
|
180
|
+
async addAuthWitness(address, messageHash) {
|
|
170
181
|
const account = await this.accountStore.getAccount(address);
|
|
171
182
|
const privateKey = await this.keyStore.getMasterSecretKey(account.publicKeys.masterIncomingViewingPublicKey);
|
|
172
183
|
const schnorr = new Schnorr();
|
|
@@ -177,7 +188,7 @@ export class TXEOracleTopLevelContext {
|
|
|
177
188
|
this.authwits.set(authWitness.requestHash.toString(), authWitness);
|
|
178
189
|
}
|
|
179
190
|
async mineBlock(options = {}) {
|
|
180
|
-
const blockNumber = await this.
|
|
191
|
+
const blockNumber = await this.getNextBlockNumber();
|
|
181
192
|
const txEffect = TxEffect.empty();
|
|
182
193
|
txEffect.nullifiers = [
|
|
183
194
|
getSingleTxBlockRequestHash(blockNumber),
|
|
@@ -199,25 +210,30 @@ export class TXEOracleTopLevelContext {
|
|
|
199
210
|
this.logger.info(`Created block ${blockNumber} with timestamp ${block.header.globalVariables.timestamp}`);
|
|
200
211
|
await this.stateMachine.handleL2Block(block);
|
|
201
212
|
}
|
|
202
|
-
async
|
|
213
|
+
async privateCallNewFlow(from, targetContractAddress = AztecAddress.zero(), functionSelector = FunctionSelector.empty(), args, argsHash = Fr.zero(), isStaticCall = false, jobId) {
|
|
203
214
|
this.logger.verbose(`Executing external function ${await this.contractStore.getDebugFunctionName(targetContractAddress, functionSelector)}@${targetContractAddress} isStaticCall=${isStaticCall}`);
|
|
204
215
|
const artifact = await this.contractStore.getFunctionArtifact(targetContractAddress, functionSelector);
|
|
205
216
|
if (!artifact) {
|
|
206
217
|
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';
|
|
207
218
|
throw new Error(message);
|
|
208
219
|
}
|
|
220
|
+
// When `from` is the zero address (e.g. when deploying a new account contract), we return an
|
|
221
|
+
// empty scope list which acts as deny-all: no notes are visible and no keys are accessible.
|
|
222
|
+
const effectiveScopes = from.isZero() ? [] : [
|
|
223
|
+
from
|
|
224
|
+
];
|
|
209
225
|
// Sync notes before executing private function to discover notes from previous transactions
|
|
210
|
-
const utilityExecutor = async (call)=>{
|
|
211
|
-
await this.executeUtilityCall(call);
|
|
226
|
+
const utilityExecutor = async (call, execScopes)=>{
|
|
227
|
+
await this.executeUtilityCall(call, execScopes, jobId);
|
|
212
228
|
};
|
|
213
|
-
await this.
|
|
214
|
-
|
|
229
|
+
const blockHeader = await this.stateMachine.anchorBlockStore.getBlockHeader();
|
|
230
|
+
await this.stateMachine.contractSyncService.ensureContractSynced(targetContractAddress, functionSelector, utilityExecutor, blockHeader, jobId, effectiveScopes);
|
|
231
|
+
const blockNumber = await this.getNextBlockNumber();
|
|
215
232
|
const callContext = new CallContext(from, targetContractAddress, functionSelector, isStaticCall);
|
|
216
233
|
const gasLimits = new Gas(DEFAULT_DA_GAS_LIMIT, DEFAULT_L2_GAS_LIMIT);
|
|
217
234
|
const teardownGasLimits = new Gas(DEFAULT_TEARDOWN_DA_GAS_LIMIT, DEFAULT_TEARDOWN_L2_GAS_LIMIT);
|
|
218
235
|
const gasSettings = new GasSettings(gasLimits, teardownGasLimits, GasFees.empty(), GasFees.empty());
|
|
219
236
|
const txContext = new TxContext(this.chainId, this.version, gasSettings);
|
|
220
|
-
const blockHeader = await this.stateMachine.anchorBlockStore.getBlockHeader();
|
|
221
237
|
const protocolNullifier = await computeProtocolNullifier(getSingleTxBlockRequestHash(blockNumber));
|
|
222
238
|
const noteCache = new ExecutionNoteCache(protocolNullifier);
|
|
223
239
|
// In production, the account contract sets the min revertible counter before calling the app function.
|
|
@@ -227,12 +243,39 @@ export class TXEOracleTopLevelContext {
|
|
|
227
243
|
await noteCache.setMinRevertibleSideEffectCounter(minRevertibleSideEffectCounter);
|
|
228
244
|
const taggingIndexCache = new ExecutionTaggingIndexCache();
|
|
229
245
|
const simulator = new WASMSimulator();
|
|
230
|
-
const privateExecutionOracle = new PrivateExecutionOracle(
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
246
|
+
const privateExecutionOracle = new PrivateExecutionOracle({
|
|
247
|
+
argsHash,
|
|
248
|
+
txContext,
|
|
249
|
+
callContext,
|
|
250
|
+
anchorBlockHeader: blockHeader,
|
|
251
|
+
utilityExecutor,
|
|
252
|
+
authWitnesses: Array.from(this.authwits.values()),
|
|
253
|
+
capsules: [],
|
|
254
|
+
executionCache: HashedValuesCache.create([
|
|
255
|
+
new HashedValues(args, argsHash)
|
|
256
|
+
]),
|
|
257
|
+
noteCache,
|
|
258
|
+
taggingIndexCache,
|
|
259
|
+
contractStore: this.contractStore,
|
|
260
|
+
noteStore: this.noteStore,
|
|
261
|
+
keyStore: this.keyStore,
|
|
262
|
+
addressStore: this.addressStore,
|
|
263
|
+
aztecNode: this.stateMachine.node,
|
|
264
|
+
senderTaggingStore: this.senderTaggingStore,
|
|
265
|
+
recipientTaggingStore: this.recipientTaggingStore,
|
|
266
|
+
senderAddressBookStore: this.senderAddressBookStore,
|
|
267
|
+
capsuleStore: this.capsuleStore,
|
|
268
|
+
privateEventStore: this.privateEventStore,
|
|
269
|
+
contractSyncService: this.stateMachine.contractSyncService,
|
|
270
|
+
jobId,
|
|
271
|
+
totalPublicCalldataCount: 0,
|
|
272
|
+
sideEffectCounter: minRevertibleSideEffectCounter,
|
|
273
|
+
scopes: effectiveScopes,
|
|
274
|
+
// In TXE, the typical transaction entrypoint is skipped, so we need to simulate the actions that such a
|
|
275
|
+
// contract would perform, including setting senderForTags.
|
|
276
|
+
senderForTags: from,
|
|
277
|
+
simulator
|
|
278
|
+
});
|
|
236
279
|
// 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.
|
|
237
280
|
let result;
|
|
238
281
|
let executionResult;
|
|
@@ -244,7 +287,7 @@ export class TXEOracleTopLevelContext {
|
|
|
244
287
|
r.publicInputs.publicTeardownCallRequest
|
|
245
288
|
]));
|
|
246
289
|
const publicFunctionsCalldata = await Promise.all(publicCallRequests.map(async (r)=>{
|
|
247
|
-
const calldata = await privateExecutionOracle.
|
|
290
|
+
const calldata = await privateExecutionOracle.loadFromExecutionCache(r.calldataHash);
|
|
248
291
|
return new HashedValues(calldata, r.calldataHash);
|
|
249
292
|
}));
|
|
250
293
|
noteCache.finish();
|
|
@@ -256,7 +299,7 @@ export class TXEOracleTopLevelContext {
|
|
|
256
299
|
// According to the protocol rules, there must be at least one nullifier in the tx. The first nullifier is used as
|
|
257
300
|
// the nonce generator for the note hashes.
|
|
258
301
|
// We pass the non-zero minRevertibleSideEffectCounter to make sure the side effects are split correctly.
|
|
259
|
-
const { publicInputs } = await generateSimulatedProvingResult(result, this.contractStore, minRevertibleSideEffectCounter);
|
|
302
|
+
const { publicInputs } = await generateSimulatedProvingResult(result, (addr, sel)=>this.contractStore.getDebugFunctionName(addr, sel), this.stateMachine.node, minRevertibleSideEffectCounter);
|
|
260
303
|
const globals = makeGlobalVariables();
|
|
261
304
|
globals.blockNumber = blockNumber;
|
|
262
305
|
globals.timestamp = this.nextBlockTimestamp;
|
|
@@ -264,7 +307,8 @@ export class TXEOracleTopLevelContext {
|
|
|
264
307
|
globals.version = this.version;
|
|
265
308
|
globals.gasFees = GasFees.empty();
|
|
266
309
|
const forkedWorldTrees = await this.stateMachine.synchronizer.nativeWorldStateService.fork();
|
|
267
|
-
const
|
|
310
|
+
const bindings = this.logger.getBindings();
|
|
311
|
+
const contractsDB = new PublicContractsDB(new TXEPublicContractDataSource(blockNumber, this.contractStore), bindings);
|
|
268
312
|
const guardedMerkleTrees = new GuardedMerkleTreeOperations(forkedWorldTrees);
|
|
269
313
|
const config = PublicSimulatorConfig.from({
|
|
270
314
|
skipFeeEnforcement: true,
|
|
@@ -273,7 +317,7 @@ export class TXEOracleTopLevelContext {
|
|
|
273
317
|
collectStatistics: false,
|
|
274
318
|
collectCallMetadata: true
|
|
275
319
|
});
|
|
276
|
-
const processor = new PublicProcessor(globals, guardedMerkleTrees, contractsDB, new CppPublicTxSimulator(guardedMerkleTrees, contractsDB, globals, config), new TestDateProvider());
|
|
320
|
+
const processor = new PublicProcessor(globals, guardedMerkleTrees, contractsDB, new CppPublicTxSimulator(guardedMerkleTrees, contractsDB, globals, config, bindings), new TestDateProvider(), undefined, createLogger('simulator:public-processor', bindings));
|
|
277
321
|
const tx = await Tx.create({
|
|
278
322
|
data: publicInputs,
|
|
279
323
|
chonkProof: ChonkProof.empty(),
|
|
@@ -323,9 +367,9 @@ export class TXEOracleTopLevelContext {
|
|
|
323
367
|
await forkedWorldTrees.close();
|
|
324
368
|
return executionResult.returnValues ?? [];
|
|
325
369
|
}
|
|
326
|
-
async
|
|
370
|
+
async publicCallNewFlow(from, targetContractAddress, calldata, isStaticCall) {
|
|
327
371
|
this.logger.verbose(`Executing public function ${await this.contractStore.getDebugFunctionName(targetContractAddress, FunctionSelector.fromField(calldata[0]))}@${targetContractAddress} isStaticCall=${isStaticCall}`);
|
|
328
|
-
const blockNumber = await this.
|
|
372
|
+
const blockNumber = await this.getNextBlockNumber();
|
|
329
373
|
const gasLimits = new Gas(DEFAULT_DA_GAS_LIMIT, DEFAULT_L2_GAS_LIMIT);
|
|
330
374
|
const teardownGasLimits = new Gas(DEFAULT_TEARDOWN_DA_GAS_LIMIT, DEFAULT_TEARDOWN_L2_GAS_LIMIT);
|
|
331
375
|
const gasSettings = new GasSettings(gasLimits, teardownGasLimits, GasFees.empty(), GasFees.empty());
|
|
@@ -340,7 +384,8 @@ export class TXEOracleTopLevelContext {
|
|
|
340
384
|
globals.version = this.version;
|
|
341
385
|
globals.gasFees = GasFees.empty();
|
|
342
386
|
const forkedWorldTrees = await this.stateMachine.synchronizer.nativeWorldStateService.fork();
|
|
343
|
-
const
|
|
387
|
+
const bindings2 = this.logger.getBindings();
|
|
388
|
+
const contractsDB = new PublicContractsDB(new TXEPublicContractDataSource(blockNumber, this.contractStore), bindings2);
|
|
344
389
|
const guardedMerkleTrees = new GuardedMerkleTreeOperations(forkedWorldTrees);
|
|
345
390
|
const config = PublicSimulatorConfig.from({
|
|
346
391
|
skipFeeEnforcement: true,
|
|
@@ -349,8 +394,8 @@ export class TXEOracleTopLevelContext {
|
|
|
349
394
|
collectStatistics: false,
|
|
350
395
|
collectCallMetadata: true
|
|
351
396
|
});
|
|
352
|
-
const simulator = new CppPublicTxSimulator(guardedMerkleTrees, contractsDB, globals, config);
|
|
353
|
-
const processor = new PublicProcessor(globals, guardedMerkleTrees, contractsDB, simulator, new TestDateProvider());
|
|
397
|
+
const simulator = new CppPublicTxSimulator(guardedMerkleTrees, contractsDB, globals, config, bindings2);
|
|
398
|
+
const processor = new PublicProcessor(globals, guardedMerkleTrees, contractsDB, simulator, new TestDateProvider(), undefined, createLogger('simulator:public-processor', bindings2));
|
|
354
399
|
// We're simulating a scenario in which private execution immediately enqueues a public call and halts. The private
|
|
355
400
|
// kernel init would in this case inject a nullifier with the transaction request hash as a non-revertible
|
|
356
401
|
// side-effect, which the AVM then expects to exist in order to use it as the nonce generator when siloing notes as
|
|
@@ -363,7 +408,7 @@ export class TXEOracleTopLevelContext {
|
|
|
363
408
|
revertibleAccumulatedData.publicCallRequests[0] = new PublicCallRequest(from, targetContractAddress, isStaticCall, calldataHash);
|
|
364
409
|
const inputsForPublic = new PartialPrivateTailPublicInputsForPublic(nonRevertibleAccumulatedData, revertibleAccumulatedData, PublicCallRequest.empty());
|
|
365
410
|
const constantData = new TxConstantData(anchorBlockHeader, txContext, Fr.zero(), Fr.zero());
|
|
366
|
-
const txData = new PrivateKernelTailCircuitPublicInputs(constantData, /*gasUsed=*/ new Gas(0, 0), /*feePayer=*/ AztecAddress.zero(), /*
|
|
411
|
+
const txData = new PrivateKernelTailCircuitPublicInputs(constantData, /*gasUsed=*/ new Gas(0, 0), /*feePayer=*/ AztecAddress.zero(), /*expirationTimestamp=*/ 0n, inputsForPublic, undefined);
|
|
367
412
|
const tx = await Tx.create({
|
|
368
413
|
data: txData,
|
|
369
414
|
chonkProof: ChonkProof.empty(),
|
|
@@ -416,19 +461,29 @@ export class TXEOracleTopLevelContext {
|
|
|
416
461
|
await forkedWorldTrees.close();
|
|
417
462
|
return returnValues ?? [];
|
|
418
463
|
}
|
|
419
|
-
async
|
|
464
|
+
async executeUtilityFunction(targetContractAddress, functionSelector, args, jobId) {
|
|
420
465
|
const artifact = await this.contractStore.getFunctionArtifact(targetContractAddress, functionSelector);
|
|
421
466
|
if (!artifact) {
|
|
422
467
|
throw new Error(`Cannot call ${functionSelector} as there is no artifact found at ${targetContractAddress}.`);
|
|
423
468
|
}
|
|
424
469
|
// Sync notes before executing utility function to discover notes from previous transactions
|
|
425
|
-
await this.
|
|
426
|
-
|
|
470
|
+
const blockHeader = await this.stateMachine.anchorBlockStore.getBlockHeader();
|
|
471
|
+
await this.stateMachine.contractSyncService.ensureContractSynced(targetContractAddress, functionSelector, async (call, execScopes)=>{
|
|
472
|
+
await this.executeUtilityCall(call, execScopes, jobId);
|
|
473
|
+
}, blockHeader, jobId, 'ALL_SCOPES');
|
|
474
|
+
const call = FunctionCall.from({
|
|
475
|
+
name: artifact.name,
|
|
476
|
+
to: targetContractAddress,
|
|
477
|
+
selector: functionSelector,
|
|
478
|
+
type: FunctionType.UTILITY,
|
|
479
|
+
hideMsgSender: false,
|
|
480
|
+
isStatic: false,
|
|
481
|
+
args,
|
|
482
|
+
returnTypes: []
|
|
427
483
|
});
|
|
428
|
-
|
|
429
|
-
return this.executeUtilityCall(call);
|
|
484
|
+
return this.executeUtilityCall(call, 'ALL_SCOPES', jobId);
|
|
430
485
|
}
|
|
431
|
-
async executeUtilityCall(call) {
|
|
486
|
+
async executeUtilityCall(call, scopes, jobId) {
|
|
432
487
|
const entryPointArtifact = await this.contractStore.getFunctionArtifactWithDebugMetadata(call.to, call.selector);
|
|
433
488
|
if (entryPointArtifact.functionType !== FunctionType.UTILITY) {
|
|
434
489
|
throw new Error(`Cannot run ${entryPointArtifact.functionType} function as utility`);
|
|
@@ -439,7 +494,23 @@ export class TXEOracleTopLevelContext {
|
|
|
439
494
|
});
|
|
440
495
|
try {
|
|
441
496
|
const anchorBlockHeader = await this.stateMachine.anchorBlockStore.getBlockHeader();
|
|
442
|
-
const oracle = new UtilityExecutionOracle(
|
|
497
|
+
const oracle = new UtilityExecutionOracle({
|
|
498
|
+
contractAddress: call.to,
|
|
499
|
+
authWitnesses: [],
|
|
500
|
+
capsules: [],
|
|
501
|
+
anchorBlockHeader,
|
|
502
|
+
contractStore: this.contractStore,
|
|
503
|
+
noteStore: this.noteStore,
|
|
504
|
+
keyStore: this.keyStore,
|
|
505
|
+
addressStore: this.addressStore,
|
|
506
|
+
aztecNode: this.stateMachine.node,
|
|
507
|
+
recipientTaggingStore: this.recipientTaggingStore,
|
|
508
|
+
senderAddressBookStore: this.senderAddressBookStore,
|
|
509
|
+
capsuleStore: this.capsuleStore,
|
|
510
|
+
privateEventStore: this.privateEventStore,
|
|
511
|
+
jobId,
|
|
512
|
+
scopes
|
|
513
|
+
});
|
|
443
514
|
const acirExecutionResult = await new WASMSimulator().executeUserCircuit(toACVMWitness(0, call.args), entryPointArtifact, new Oracle(oracle).toACIRCallback()).catch((err)=>{
|
|
444
515
|
err.message = resolveAssertionMessageFromError(err, entryPointArtifact);
|
|
445
516
|
throw new ExecutionError(err.message, {
|
|
@@ -449,10 +520,10 @@ export class TXEOracleTopLevelContext {
|
|
|
449
520
|
cause: err
|
|
450
521
|
});
|
|
451
522
|
});
|
|
452
|
-
this.logger.verbose(`Utility
|
|
523
|
+
this.logger.verbose(`Utility execution for ${call.to}.${call.selector} completed`);
|
|
453
524
|
return witnessMapToFields(acirExecutionResult.returnWitness);
|
|
454
525
|
} catch (err) {
|
|
455
|
-
throw createSimulationError(err instanceof Error ? err : new Error('Unknown error during utility
|
|
526
|
+
throw createSimulationError(err instanceof Error ? err : new Error('Unknown error during utility execution'));
|
|
456
527
|
}
|
|
457
528
|
}
|
|
458
529
|
close() {
|