@aztec/end-to-end 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/README.md +27 -0
- package/dest/bench/client_flows/client_flows_benchmark.d.ts +1 -1
- package/dest/bench/client_flows/client_flows_benchmark.d.ts.map +1 -1
- package/dest/bench/client_flows/client_flows_benchmark.js +18 -11
- package/dest/bench/utils.d.ts +1 -1
- package/dest/bench/utils.d.ts.map +1 -1
- package/dest/bench/utils.js +6 -3
- package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.d.ts +3 -2
- package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.d.ts.map +1 -1
- package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.js +15 -15
- package/dest/e2e_epochs/epochs_test.d.ts +3 -1
- package/dest/e2e_epochs/epochs_test.d.ts.map +1 -1
- package/dest/e2e_epochs/epochs_test.js +5 -2
- package/dest/e2e_fees/fees_test.d.ts +1 -1
- package/dest/e2e_fees/fees_test.d.ts.map +1 -1
- package/dest/e2e_fees/fees_test.js +14 -7
- package/dest/e2e_nested_contract/nested_contract_test.d.ts +1 -1
- package/dest/e2e_nested_contract/nested_contract_test.d.ts.map +1 -1
- package/dest/e2e_nested_contract/nested_contract_test.js +4 -6
- package/dest/e2e_p2p/inactivity_slash_test.d.ts +1 -1
- package/dest/e2e_p2p/inactivity_slash_test.d.ts.map +1 -1
- package/dest/e2e_p2p/inactivity_slash_test.js +1 -0
- package/dest/e2e_p2p/p2p_network.d.ts +3 -2
- package/dest/e2e_p2p/p2p_network.d.ts.map +1 -1
- package/dest/e2e_p2p/p2p_network.js +18 -15
- package/dest/e2e_p2p/shared.d.ts +33 -1
- package/dest/e2e_p2p/shared.d.ts.map +1 -1
- package/dest/e2e_p2p/shared.js +56 -2
- package/dest/e2e_token_contract/token_contract_test.d.ts +1 -1
- package/dest/e2e_token_contract/token_contract_test.d.ts.map +1 -1
- package/dest/e2e_token_contract/token_contract_test.js +11 -11
- package/dest/fixtures/authwit_proxy.d.ts +3 -3
- package/dest/fixtures/authwit_proxy.d.ts.map +1 -1
- package/dest/fixtures/authwit_proxy.js +4 -0
- package/dest/fixtures/e2e_prover_test.d.ts +1 -1
- package/dest/fixtures/e2e_prover_test.d.ts.map +1 -1
- package/dest/fixtures/e2e_prover_test.js +6 -6
- package/dest/fixtures/setup.d.ts +10 -7
- package/dest/fixtures/setup.d.ts.map +1 -1
- package/dest/fixtures/setup.js +16 -12
- package/dest/fixtures/token_utils.d.ts +1 -1
- package/dest/fixtures/token_utils.d.ts.map +1 -1
- package/dest/fixtures/token_utils.js +2 -2
- package/dest/forward-compatibility/wallet_rpc_client.d.ts +7 -0
- package/dest/forward-compatibility/wallet_rpc_client.d.ts.map +1 -0
- package/dest/forward-compatibility/wallet_rpc_client.js +15 -0
- package/dest/forward-compatibility/wallet_service.d.ts +3 -0
- package/dest/forward-compatibility/wallet_service.d.ts.map +1 -0
- package/dest/forward-compatibility/wallet_service.js +109 -0
- package/dest/install_legacy_contracts.d.cts +10 -0
- package/dest/install_legacy_contracts.d.cts.map +1 -0
- package/dest/legacy-jest-resolver.d.cts +3 -0
- package/dest/legacy-jest-resolver.d.cts.map +1 -0
- package/dest/shared/cross_chain_test_harness.d.ts +1 -1
- package/dest/shared/cross_chain_test_harness.d.ts.map +1 -1
- package/dest/shared/cross_chain_test_harness.js +13 -13
- package/dest/shared/gas_portal_test_harness.js +2 -2
- package/dest/shared/submit-transactions.js +1 -1
- package/dest/shared/uniswap_l1_l2.d.ts +1 -1
- package/dest/shared/uniswap_l1_l2.d.ts.map +1 -1
- package/dest/shared/uniswap_l1_l2.js +14 -17
- package/dest/simulators/lending_simulator.d.ts +1 -1
- package/dest/simulators/lending_simulator.d.ts.map +1 -1
- package/dest/simulators/lending_simulator.js +4 -4
- package/dest/simulators/token_simulator.d.ts +1 -1
- package/dest/simulators/token_simulator.d.ts.map +1 -1
- package/dest/simulators/token_simulator.js +2 -2
- package/dest/spartan/setup_test_wallets.d.ts +4 -2
- package/dest/spartan/setup_test_wallets.d.ts.map +1 -1
- package/dest/spartan/setup_test_wallets.js +28 -15
- package/dest/spartan/utils/index.d.ts +2 -1
- package/dest/spartan/utils/index.d.ts.map +1 -1
- package/dest/spartan/utils/index.js +2 -0
- package/dest/spartan/utils/nodes.d.ts +5 -4
- package/dest/spartan/utils/nodes.d.ts.map +1 -1
- package/dest/spartan/utils/nodes.js +5 -4
- package/dest/test-wallet/test_wallet.d.ts +24 -23
- package/dest/test-wallet/test_wallet.d.ts.map +1 -1
- package/dest/test-wallet/test_wallet.js +115 -80
- package/dest/test-wallet/wallet_worker_script.js +41 -33
- package/dest/test-wallet/worker_wallet.d.ts +4 -4
- package/dest/test-wallet/worker_wallet.d.ts.map +1 -1
- package/dest/test-wallet/worker_wallet.js +51 -3
- package/dest/test-wallet/worker_wallet_schema.d.ts +4 -4
- package/package.json +41 -40
- package/src/bench/client_flows/client_flows_benchmark.ts +40 -32
- package/src/bench/utils.ts +7 -2
- package/src/e2e_blacklist_token_contract/blacklist_token_contract_test.ts +21 -20
- package/src/e2e_epochs/epochs_test.ts +14 -2
- package/src/e2e_fees/fees_test.ts +14 -7
- package/src/e2e_nested_contract/nested_contract_test.ts +6 -4
- package/src/e2e_p2p/inactivity_slash_test.ts +1 -0
- package/src/e2e_p2p/p2p_network.ts +24 -20
- package/src/e2e_p2p/shared.ts +97 -2
- package/src/e2e_token_contract/token_contract_test.ts +16 -8
- package/src/fixtures/authwit_proxy.ts +4 -0
- package/src/fixtures/dumps/epoch_proof_result.json +1 -1
- package/src/fixtures/e2e_prover_test.ts +11 -5
- package/src/fixtures/setup.ts +27 -18
- package/src/fixtures/token_utils.ts +4 -2
- package/src/forward-compatibility/wallet_rpc_client.ts +14 -0
- package/src/forward-compatibility/wallet_service.ts +104 -0
- package/src/install_legacy_contracts.cjs +75 -0
- package/src/legacy-jest-resolver.cjs +101 -0
- package/src/shared/cross_chain_test_harness.ts +13 -9
- package/src/shared/gas_portal_test_harness.ts +1 -1
- package/src/shared/submit-transactions.ts +1 -1
- package/src/shared/uniswap_l1_l2.ts +35 -28
- package/src/simulators/lending_simulator.ts +8 -4
- package/src/simulators/token_simulator.ts +6 -2
- package/src/spartan/setup_test_wallets.ts +51 -19
- package/src/spartan/utils/index.ts +3 -0
- package/src/spartan/utils/nodes.ts +6 -3
- package/src/test-wallet/test_wallet.ts +144 -99
- package/src/test-wallet/wallet_worker_script.ts +47 -30
- package/src/test-wallet/worker_wallet.ts +54 -5
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import { generateSchnorrAccounts } from '@aztec/accounts/testing';
|
|
2
|
+
import { NO_FROM } from '@aztec/aztec.js/account';
|
|
2
3
|
import { AztecAddress } from '@aztec/aztec.js/addresses';
|
|
3
4
|
import { NO_WAIT } from '@aztec/aztec.js/contracts';
|
|
4
5
|
import { L1FeeJuicePortalManager } from '@aztec/aztec.js/ethereum';
|
|
@@ -89,7 +90,7 @@ export async function deploySponsoredTestAccountsWithTokens(
|
|
|
89
90
|
const paymentMethod = new SponsoredFeePaymentMethod(await getSponsoredFPCAddress());
|
|
90
91
|
const recipientDeployMethod = await recipientAccount.getDeployMethod();
|
|
91
92
|
await recipientDeployMethod.send({
|
|
92
|
-
from:
|
|
93
|
+
from: NO_FROM,
|
|
93
94
|
fee: { paymentMethod },
|
|
94
95
|
wait: { timeout: 2400 },
|
|
95
96
|
});
|
|
@@ -97,7 +98,7 @@ export async function deploySponsoredTestAccountsWithTokens(
|
|
|
97
98
|
fundedAccounts.map(async a => {
|
|
98
99
|
const deployMethod = await a.getDeployMethod();
|
|
99
100
|
await deployMethod.send({
|
|
100
|
-
from:
|
|
101
|
+
from: NO_FROM,
|
|
101
102
|
fee: { paymentMethod },
|
|
102
103
|
wait: { timeout: 2400 },
|
|
103
104
|
}); // increase timeout on purpose in order to account for two empty epochs
|
|
@@ -129,20 +130,28 @@ export async function deploySponsoredTestAccountsWithTokens(
|
|
|
129
130
|
}
|
|
130
131
|
|
|
131
132
|
async function deployAccountWithDiagnostics(
|
|
132
|
-
account: { getDeployMethod: () => Promise<{ send: (opts: any) => any }>; address: any },
|
|
133
|
+
account: { getDeployMethod: () => Promise<{ simulate: (opts: any) => any; send: (opts: any) => any }>; address: any },
|
|
133
134
|
paymentMethod: SponsoredFeePaymentMethod,
|
|
134
135
|
aztecNode: AztecNode,
|
|
135
136
|
logger: Logger,
|
|
136
137
|
accountLabel: string,
|
|
138
|
+
estimateGas?: boolean,
|
|
137
139
|
): Promise<void> {
|
|
138
140
|
const deployMethod = await account.getDeployMethod();
|
|
139
141
|
let txHash;
|
|
140
142
|
try {
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
fee: { paymentMethod }
|
|
143
|
+
let gasSettings;
|
|
144
|
+
if (estimateGas) {
|
|
145
|
+
const sim = await deployMethod.simulate({ from: NO_FROM, fee: { paymentMethod } });
|
|
146
|
+
gasSettings = sim.estimatedGas;
|
|
147
|
+
logger.info(`${accountLabel} estimated gas: DA=${gasSettings.gasLimits.daGas} L2=${gasSettings.gasLimits.l2Gas}`);
|
|
148
|
+
}
|
|
149
|
+
const deployResult = await deployMethod.send({
|
|
150
|
+
from: NO_FROM,
|
|
151
|
+
fee: { paymentMethod, gasSettings },
|
|
144
152
|
wait: NO_WAIT,
|
|
145
153
|
});
|
|
154
|
+
txHash = deployResult.txHash;
|
|
146
155
|
await waitForTx(aztecNode, txHash, { timeout: 2400 });
|
|
147
156
|
logger.info(`${accountLabel} deployed at ${account.address}`);
|
|
148
157
|
} catch (error) {
|
|
@@ -164,18 +173,29 @@ async function deployAccountWithDiagnostics(
|
|
|
164
173
|
}
|
|
165
174
|
|
|
166
175
|
async function deployAccountsInBatches(
|
|
167
|
-
accounts: {
|
|
176
|
+
accounts: {
|
|
177
|
+
getDeployMethod: () => Promise<{ simulate: (opts: any) => any; send: (opts: any) => any }>;
|
|
178
|
+
address: any;
|
|
179
|
+
}[],
|
|
168
180
|
paymentMethod: SponsoredFeePaymentMethod,
|
|
169
181
|
aztecNode: AztecNode,
|
|
170
182
|
logger: Logger,
|
|
171
183
|
labelPrefix: string,
|
|
172
184
|
batchSize = 2,
|
|
185
|
+
estimateGas?: boolean,
|
|
173
186
|
): Promise<void> {
|
|
174
187
|
for (let i = 0; i < accounts.length; i += batchSize) {
|
|
175
188
|
const batch = accounts.slice(i, i + batchSize);
|
|
176
189
|
await Promise.all(
|
|
177
190
|
batch.map((account, idx) =>
|
|
178
|
-
deployAccountWithDiagnostics(
|
|
191
|
+
deployAccountWithDiagnostics(
|
|
192
|
+
account,
|
|
193
|
+
paymentMethod,
|
|
194
|
+
aztecNode,
|
|
195
|
+
logger,
|
|
196
|
+
`${labelPrefix}${i + idx + 1}`,
|
|
197
|
+
estimateGas,
|
|
198
|
+
),
|
|
179
199
|
),
|
|
180
200
|
);
|
|
181
201
|
}
|
|
@@ -186,6 +206,7 @@ export async function deploySponsoredTestAccounts(
|
|
|
186
206
|
aztecNode: AztecNode,
|
|
187
207
|
logger: Logger,
|
|
188
208
|
numberOfFundedWallets = 1,
|
|
209
|
+
opts?: { estimateGas?: boolean },
|
|
189
210
|
): Promise<TestAccountsWithoutTokens> {
|
|
190
211
|
const [recipient, ...funded] = await generateSchnorrAccounts(numberOfFundedWallets + 1);
|
|
191
212
|
const recipientAccount = await wallet.createSchnorrAccount(recipient.secret, recipient.salt);
|
|
@@ -195,8 +216,23 @@ export async function deploySponsoredTestAccounts(
|
|
|
195
216
|
|
|
196
217
|
const paymentMethod = new SponsoredFeePaymentMethod(await getSponsoredFPCAddress());
|
|
197
218
|
|
|
198
|
-
await deployAccountWithDiagnostics(
|
|
199
|
-
|
|
219
|
+
await deployAccountWithDiagnostics(
|
|
220
|
+
recipientAccount,
|
|
221
|
+
paymentMethod,
|
|
222
|
+
aztecNode,
|
|
223
|
+
logger,
|
|
224
|
+
'Recipient account',
|
|
225
|
+
opts?.estimateGas,
|
|
226
|
+
);
|
|
227
|
+
await deployAccountsInBatches(
|
|
228
|
+
fundedAccounts,
|
|
229
|
+
paymentMethod,
|
|
230
|
+
aztecNode,
|
|
231
|
+
logger,
|
|
232
|
+
'Funded account ',
|
|
233
|
+
2,
|
|
234
|
+
opts?.estimateGas,
|
|
235
|
+
);
|
|
200
236
|
|
|
201
237
|
return {
|
|
202
238
|
aztecNode,
|
|
@@ -234,7 +270,7 @@ export async function deployTestAccountsWithTokens(
|
|
|
234
270
|
fundedAccounts.map(async (a, i) => {
|
|
235
271
|
const paymentMethod = new FeeJuicePaymentMethodWithClaim(a.address, claims[i]);
|
|
236
272
|
const deployMethod = await a.getDeployMethod();
|
|
237
|
-
await deployMethod.send({ from:
|
|
273
|
+
await deployMethod.send({ from: NO_FROM, fee: { paymentMethod } });
|
|
238
274
|
logger.info(`Account deployed at ${a.address}`);
|
|
239
275
|
}),
|
|
240
276
|
);
|
|
@@ -278,7 +314,7 @@ async function bridgeL1FeeJuice(
|
|
|
278
314
|
const claim = await portal.bridgeTokensPublic(recipient, amount, true /* mint */);
|
|
279
315
|
|
|
280
316
|
const isSynced = async () =>
|
|
281
|
-
(await aztecNode.
|
|
317
|
+
(await aztecNode.getL1ToL2MessageCheckpoint(Fr.fromHexString(claim.messageHash))) !== undefined;
|
|
282
318
|
await retryUntil(isSynced, `message ${claim.messageHash} sync`, 24, 0.5);
|
|
283
319
|
|
|
284
320
|
log.info(`Created a claim for ${amount} L1 fee juice to ${recipient}.`, claim);
|
|
@@ -310,13 +346,9 @@ async function deployTokenAndMint(
|
|
|
310
346
|
logger: Logger,
|
|
311
347
|
) {
|
|
312
348
|
logger.verbose(`Deploying TokenContract...`);
|
|
313
|
-
const {
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
TOKEN_NAME,
|
|
317
|
-
TOKEN_SYMBOL,
|
|
318
|
-
TOKEN_DECIMALS,
|
|
319
|
-
).send({
|
|
349
|
+
const {
|
|
350
|
+
receipt: { contract: tokenContract },
|
|
351
|
+
} = await TokenContract.deploy(wallet, admin, TOKEN_NAME, TOKEN_SYMBOL, TOKEN_DECIMALS).send({
|
|
320
352
|
from: admin,
|
|
321
353
|
fee: {
|
|
322
354
|
paymentMethod,
|
|
@@ -255,18 +255,21 @@ export async function initHADb(namespace: string): Promise<void> {
|
|
|
255
255
|
}
|
|
256
256
|
|
|
257
257
|
/**
|
|
258
|
-
*
|
|
259
|
-
*
|
|
258
|
+
* Enables or disables probabilistic transaction dropping on validators and waits for rollout.
|
|
259
|
+
* Wired to env vars P2P_DROP_TX and P2P_DROP_TX_CHANCE via Helm values.
|
|
260
260
|
*/
|
|
261
261
|
export async function setValidatorTxDrop({
|
|
262
262
|
namespace,
|
|
263
|
+
enabled,
|
|
263
264
|
probability,
|
|
264
265
|
logger: log,
|
|
265
266
|
}: {
|
|
266
267
|
namespace: string;
|
|
268
|
+
enabled: boolean;
|
|
267
269
|
probability: number;
|
|
268
270
|
logger: Logger;
|
|
269
271
|
}) {
|
|
272
|
+
const drop = enabled ? 'true' : 'false';
|
|
270
273
|
const prob = String(probability);
|
|
271
274
|
|
|
272
275
|
const selectors = ['app.kubernetes.io/name=validator', 'app.kubernetes.io/component=validator', 'app=validator'];
|
|
@@ -281,7 +284,7 @@ export async function setValidatorTxDrop({
|
|
|
281
284
|
if (names.length === 0) {
|
|
282
285
|
continue;
|
|
283
286
|
}
|
|
284
|
-
const cmd = `kubectl set env statefulset -l ${selector} -n ${namespace} P2P_DROP_TX_CHANCE=${prob}`;
|
|
287
|
+
const cmd = `kubectl set env statefulset -l ${selector} -n ${namespace} P2P_DROP_TX=${drop} P2P_DROP_TX_CHANCE=${prob}`;
|
|
285
288
|
log.info(`command: ${cmd}`);
|
|
286
289
|
await execAsync(cmd);
|
|
287
290
|
updated = true;
|
|
@@ -1,7 +1,9 @@
|
|
|
1
1
|
import { EcdsaKAccountContract, EcdsaRAccountContract } from '@aztec/accounts/ecdsa';
|
|
2
2
|
import { SchnorrAccountContract } from '@aztec/accounts/schnorr';
|
|
3
|
-
import {
|
|
4
|
-
import {
|
|
3
|
+
import { StubEcdsaAccountContractArtifact, createStubEcdsaAccount } from '@aztec/accounts/stub/ecdsa';
|
|
4
|
+
import { StubSchnorrAccountContractArtifact, createStubSchnorrAccount } from '@aztec/accounts/stub/schnorr';
|
|
5
|
+
import { type Account, type AccountContract, NO_FROM } from '@aztec/aztec.js/account';
|
|
6
|
+
import type { CompleteAddress } from '@aztec/aztec.js/addresses';
|
|
5
7
|
import {
|
|
6
8
|
type CallIntent,
|
|
7
9
|
type ContractFunctionInteractionCallIntent,
|
|
@@ -13,10 +15,12 @@ import {
|
|
|
13
15
|
} from '@aztec/aztec.js/authorization';
|
|
14
16
|
import type { AztecNode } from '@aztec/aztec.js/node';
|
|
15
17
|
import { AccountManager, type SendOptions } from '@aztec/aztec.js/wallet';
|
|
18
|
+
import { TxSimulationResultWithAppOffset } from '@aztec/aztec.js/wallet';
|
|
16
19
|
import type { DefaultAccountEntrypointOptions } from '@aztec/entrypoints/account';
|
|
20
|
+
import { DefaultEntrypoint } from '@aztec/entrypoints/default';
|
|
17
21
|
import { Fq, Fr } from '@aztec/foundation/curves/bn254';
|
|
18
22
|
import { GrumpkinScalar } from '@aztec/foundation/curves/grumpkin';
|
|
19
|
-
import type {
|
|
23
|
+
import type { NotesFilter } from '@aztec/pxe/client/lazy';
|
|
20
24
|
import { type PXEConfig, getPXEConfig } from '@aztec/pxe/config';
|
|
21
25
|
import { PXE, type PXECreationOptions, createPXE } from '@aztec/pxe/server';
|
|
22
26
|
import { AuthWitness } from '@aztec/stdlib/auth-witness';
|
|
@@ -24,9 +28,17 @@ import { AztecAddress } from '@aztec/stdlib/aztec-address';
|
|
|
24
28
|
import { getContractInstanceFromInstantiationParams } from '@aztec/stdlib/contract';
|
|
25
29
|
import { deriveSigningKey } from '@aztec/stdlib/keys';
|
|
26
30
|
import type { NoteDao } from '@aztec/stdlib/note';
|
|
27
|
-
import
|
|
31
|
+
import {
|
|
32
|
+
type BlockHeader,
|
|
33
|
+
type ContractOverrides,
|
|
34
|
+
SimulationOverrides,
|
|
35
|
+
type TxExecutionRequest,
|
|
36
|
+
type TxHash,
|
|
37
|
+
type TxReceipt,
|
|
38
|
+
} from '@aztec/stdlib/tx';
|
|
28
39
|
import { ExecutionPayload, mergeExecutionPayloads } from '@aztec/stdlib/tx';
|
|
29
|
-
import { BaseWallet, type
|
|
40
|
+
import { BaseWallet, type SimulateViaEntrypointOptions } from '@aztec/wallet-sdk/base-wallet';
|
|
41
|
+
import type { AccountType } from '@aztec/wallets/embedded';
|
|
30
42
|
|
|
31
43
|
import { AztecNodeProxy, ProvenTx } from './utils.js';
|
|
32
44
|
|
|
@@ -36,6 +48,7 @@ import { AztecNodeProxy, ProvenTx } from './utils.js';
|
|
|
36
48
|
export interface AccountData {
|
|
37
49
|
secret: Fr;
|
|
38
50
|
salt: Fr;
|
|
51
|
+
type?: AccountType;
|
|
39
52
|
contract: AccountContract;
|
|
40
53
|
}
|
|
41
54
|
|
|
@@ -76,70 +89,92 @@ export class TestWallet extends BaseWallet {
|
|
|
76
89
|
|
|
77
90
|
createSchnorrAccount(secret: Fr, salt: Fr, signingKey?: Fq): Promise<AccountManager> {
|
|
78
91
|
signingKey = signingKey ?? deriveSigningKey(secret);
|
|
79
|
-
|
|
80
|
-
secret,
|
|
81
|
-
salt,
|
|
82
|
-
contract: new SchnorrAccountContract(signingKey),
|
|
83
|
-
};
|
|
84
|
-
return this.createAccount(accountData);
|
|
92
|
+
return this.createAccount({ secret, salt, type: 'schnorr', contract: new SchnorrAccountContract(signingKey) });
|
|
85
93
|
}
|
|
86
94
|
|
|
87
95
|
createECDSARAccount(secret: Fr, salt: Fr, signingKey: Buffer): Promise<AccountManager> {
|
|
88
|
-
|
|
96
|
+
return this.createAccount({
|
|
89
97
|
secret,
|
|
90
98
|
salt,
|
|
99
|
+
type: 'ecdsasecp256r1',
|
|
91
100
|
contract: new EcdsaRAccountContract(signingKey),
|
|
92
|
-
};
|
|
93
|
-
return this.createAccount(accountData);
|
|
101
|
+
});
|
|
94
102
|
}
|
|
95
103
|
|
|
96
104
|
createECDSAKAccount(secret: Fr, salt: Fr, signingKey: Buffer): Promise<AccountManager> {
|
|
97
|
-
|
|
105
|
+
return this.createAccount({
|
|
98
106
|
secret,
|
|
99
107
|
salt,
|
|
108
|
+
type: 'ecdsasecp256k1',
|
|
100
109
|
contract: new EcdsaKAccountContract(signingKey),
|
|
101
|
-
};
|
|
102
|
-
return this.createAccount(accountData);
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
async getFakeAccountDataFor(address: AztecAddress) {
|
|
106
|
-
const originalAccount = await this.getAccountFromAddress(address);
|
|
107
|
-
if (originalAccount instanceof SignerlessAccount) {
|
|
108
|
-
throw new Error(`Cannot create fake account data for SignerlessAccount at address: ${address}`);
|
|
109
|
-
}
|
|
110
|
-
const originalAddress = (originalAccount as Account).getCompleteAddress();
|
|
111
|
-
const contractInstance = await this.pxe.getContractInstance(originalAddress.address);
|
|
112
|
-
if (!contractInstance) {
|
|
113
|
-
throw new Error(`No contract instance found for address: ${originalAddress.address}`);
|
|
114
|
-
}
|
|
115
|
-
const stubAccount = createStubAccount(originalAddress);
|
|
116
|
-
const instance = await getContractInstanceFromInstantiationParams(StubAccountContractArtifact, {
|
|
117
|
-
salt: Fr.random(),
|
|
118
110
|
});
|
|
119
|
-
return {
|
|
120
|
-
account: stubAccount,
|
|
121
|
-
instance,
|
|
122
|
-
artifact: StubAccountContractArtifact,
|
|
123
|
-
};
|
|
124
111
|
}
|
|
125
|
-
protected accounts: Map<string, Account> = new Map();
|
|
126
112
|
|
|
127
113
|
/**
|
|
128
|
-
*
|
|
129
|
-
*
|
|
130
|
-
* When this flag is true, simulateViaEntrypoint constructs a request using a fake account
|
|
131
|
-
* (and accepts contract overrides on the input) and the PXE emulates kernel effects without
|
|
132
|
-
* generating kernel witnesses. When false, simulateViaEntrypoint defers to the standard
|
|
133
|
-
* simulation path via the real account entrypoint.
|
|
114
|
+
* Builds contract overrides for all provided addresses by replacing their account contracts with stub implementations.
|
|
134
115
|
*/
|
|
135
|
-
|
|
116
|
+
protected async buildAccountOverrides(addresses: AztecAddress[]): Promise<ContractOverrides> {
|
|
117
|
+
const accounts = await this.getAccounts();
|
|
118
|
+
const contracts: ContractOverrides = {};
|
|
119
|
+
|
|
120
|
+
const filtered = accounts.filter(acc => addresses.some(addr => addr.equals(acc.item)));
|
|
121
|
+
|
|
122
|
+
for (const account of filtered) {
|
|
123
|
+
const address = account.item;
|
|
124
|
+
const originalAccount = await this.getAccountFromAddress(address);
|
|
125
|
+
const completeAddress = originalAccount.getCompleteAddress();
|
|
126
|
+
const contractInstance = await this.pxe.getContractInstance(completeAddress.address);
|
|
127
|
+
if (!contractInstance) {
|
|
128
|
+
throw new Error(
|
|
129
|
+
`No contract instance found for address: ${completeAddress.address} during account override building. This is a bug!`,
|
|
130
|
+
);
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
const stubArtifact = this.getStubArtifactFor(address);
|
|
134
|
+
const stubConstructorArgs =
|
|
135
|
+
this.getTypeFor(address) === 'schnorr' ? [Fr.ZERO, Fr.ZERO] : [Buffer.alloc(32), Buffer.alloc(32)];
|
|
136
|
+
const stubInstance = await getContractInstanceFromInstantiationParams(stubArtifact, {
|
|
137
|
+
salt: Fr.random(),
|
|
138
|
+
constructorArgs: stubConstructorArgs,
|
|
139
|
+
});
|
|
140
|
+
|
|
141
|
+
contracts[address.toString()] = {
|
|
142
|
+
instance: stubInstance,
|
|
143
|
+
artifact: stubArtifact,
|
|
144
|
+
};
|
|
145
|
+
}
|
|
136
146
|
|
|
137
|
-
|
|
138
|
-
this.simulatedSimulations = true;
|
|
147
|
+
return contracts;
|
|
139
148
|
}
|
|
140
149
|
|
|
141
|
-
|
|
142
|
-
|
|
150
|
+
protected accounts: Map<string, { account: Account; type: AccountType }> = new Map();
|
|
151
|
+
|
|
152
|
+
private getTypeFor(address: AztecAddress): AccountType {
|
|
153
|
+
return this.accounts.get(address.toString())?.type ?? 'schnorr';
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
private getStubArtifactFor(address: AztecAddress) {
|
|
157
|
+
return this.getTypeFor(address) === 'schnorr'
|
|
158
|
+
? StubSchnorrAccountContractArtifact
|
|
159
|
+
: StubEcdsaAccountContractArtifact;
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
private getStubAccountFor(address: AztecAddress, completeAddress: CompleteAddress) {
|
|
163
|
+
return this.getTypeFor(address) === 'schnorr'
|
|
164
|
+
? createStubSchnorrAccount(completeAddress)
|
|
165
|
+
: createStubEcdsaAccount(completeAddress);
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
/**
|
|
169
|
+
* Controls how the test wallet simulates transactions:
|
|
170
|
+
* - `kernelless`: Skips kernel circuits but uses the real account contract. Default.
|
|
171
|
+
* - `kernelless-override`: Skips kernels and replaces the account with a stub that doesn't do authwit validation.
|
|
172
|
+
* - `full`: Uses real kernel circuits and real account contracts. Slow!
|
|
173
|
+
*/
|
|
174
|
+
private simulationMode: 'kernelless' | 'kernelless-override' | 'full' = 'kernelless';
|
|
175
|
+
|
|
176
|
+
setSimulationMode(mode: 'kernelless' | 'kernelless-override' | 'full') {
|
|
177
|
+
this.simulationMode = mode;
|
|
143
178
|
}
|
|
144
179
|
|
|
145
180
|
setMinFeePadding(value?: number) {
|
|
@@ -147,27 +182,25 @@ export class TestWallet extends BaseWallet {
|
|
|
147
182
|
}
|
|
148
183
|
|
|
149
184
|
protected getAccountFromAddress(address: AztecAddress): Promise<Account> {
|
|
150
|
-
|
|
151
|
-
if (address.equals(AztecAddress.ZERO)) {
|
|
152
|
-
account = new SignerlessAccount();
|
|
153
|
-
} else {
|
|
154
|
-
account = this.accounts.get(address?.toString() ?? '');
|
|
155
|
-
}
|
|
185
|
+
const entry = this.accounts.get(address?.toString() ?? '');
|
|
156
186
|
|
|
157
|
-
if (!
|
|
187
|
+
if (!entry) {
|
|
158
188
|
throw new Error(`Account not found in wallet for address: ${address}`);
|
|
159
189
|
}
|
|
160
190
|
|
|
161
|
-
return Promise.resolve(account);
|
|
191
|
+
return Promise.resolve(entry.account);
|
|
162
192
|
}
|
|
163
193
|
|
|
164
194
|
getAccounts() {
|
|
165
|
-
return Promise.resolve(
|
|
195
|
+
return Promise.resolve(
|
|
196
|
+
Array.from(this.accounts.values()).map(entry => ({ alias: '', item: entry.account.getAddress() })),
|
|
197
|
+
);
|
|
166
198
|
}
|
|
167
199
|
|
|
168
200
|
async createAccount(accountData?: AccountData): Promise<AccountManager> {
|
|
169
201
|
const secret = accountData?.secret ?? Fr.random();
|
|
170
202
|
const salt = accountData?.salt ?? Fr.random();
|
|
203
|
+
const type = accountData?.type ?? 'schnorr';
|
|
171
204
|
const contract = accountData?.contract ?? new SchnorrAccountContract(GrumpkinScalar.random());
|
|
172
205
|
|
|
173
206
|
const accountManager = await AccountManager.create(this, secret, contract, salt);
|
|
@@ -177,7 +210,8 @@ export class TestWallet extends BaseWallet {
|
|
|
177
210
|
|
|
178
211
|
await this.registerContract(instance, artifact, secret);
|
|
179
212
|
|
|
180
|
-
|
|
213
|
+
const address = accountManager.address.toString();
|
|
214
|
+
this.accounts.set(address, { account: await accountManager.getAccount(), type });
|
|
181
215
|
|
|
182
216
|
return accountManager;
|
|
183
217
|
}
|
|
@@ -220,60 +254,71 @@ export class TestWallet extends BaseWallet {
|
|
|
220
254
|
return account.createAuthWit(intentInnerHash, chainInfo);
|
|
221
255
|
}
|
|
222
256
|
|
|
223
|
-
/**
|
|
224
|
-
* Override simulateViaEntrypoint to use fake accounts for kernelless simulation
|
|
225
|
-
* when simulatedSimulations is enabled. Otherwise falls through to the real entrypoint path.
|
|
226
|
-
*/
|
|
227
257
|
protected override async simulateViaEntrypoint(
|
|
228
258
|
executionPayload: ExecutionPayload,
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
if (!this.simulatedSimulations) {
|
|
236
|
-
return super.simulateViaEntrypoint(
|
|
237
|
-
executionPayload,
|
|
238
|
-
from,
|
|
239
|
-
feeOptions,
|
|
240
|
-
scopes,
|
|
241
|
-
skipTxValidation,
|
|
242
|
-
skipFeeEnforcement,
|
|
243
|
-
);
|
|
244
|
-
}
|
|
259
|
+
opts: SimulateViaEntrypointOptions,
|
|
260
|
+
): Promise<TxSimulationResultWithAppOffset> {
|
|
261
|
+
const { from, feeOptions, additionalScopes, skipTxValidation, skipFeeEnforcement } = opts;
|
|
262
|
+
const scopes = this.scopesFrom(from, additionalScopes);
|
|
263
|
+
const skipKernels = this.simulationMode !== 'full';
|
|
264
|
+
const useOverride = this.simulationMode === 'kernelless-override';
|
|
245
265
|
|
|
246
266
|
const feeExecutionPayload = await feeOptions.walletFeePaymentMethod?.getExecutionPayload();
|
|
247
|
-
const executionOptions: DefaultAccountEntrypointOptions = {
|
|
248
|
-
txNonce: Fr.random(),
|
|
249
|
-
cancellable: this.cancellableTransactions,
|
|
250
|
-
feePaymentMethodOptions: feeOptions.accountFeePaymentMethodOptions,
|
|
251
|
-
};
|
|
252
267
|
const finalExecutionPayload = feeExecutionPayload
|
|
253
268
|
? mergeExecutionPayloads([feeExecutionPayload, executionPayload])
|
|
254
269
|
: executionPayload;
|
|
255
|
-
const { account: fromAccount, instance, artifact } = await this.getFakeAccountDataFor(from);
|
|
256
270
|
const chainInfo = await this.getChainInfo();
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
271
|
+
|
|
272
|
+
let overrides: SimulationOverrides | undefined;
|
|
273
|
+
let txRequest: TxExecutionRequest;
|
|
274
|
+
if (useOverride) {
|
|
275
|
+
const accountOverrides = await this.buildAccountOverrides(scopes);
|
|
276
|
+
overrides = new SimulationOverrides(accountOverrides);
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
if (from === NO_FROM) {
|
|
280
|
+
const entrypoint = new DefaultEntrypoint();
|
|
281
|
+
txRequest = await entrypoint.createTxExecutionRequest(finalExecutionPayload, feeOptions.gasSettings, chainInfo);
|
|
282
|
+
} else {
|
|
283
|
+
let fromAccount: Account;
|
|
284
|
+
if (useOverride) {
|
|
285
|
+
const originalAccount = await this.getAccountFromAddress(from);
|
|
286
|
+
fromAccount = this.getStubAccountFor(from, originalAccount.getCompleteAddress());
|
|
287
|
+
} else {
|
|
288
|
+
fromAccount = await this.getAccountFromAddress(from);
|
|
289
|
+
}
|
|
290
|
+
const executionOptions: DefaultAccountEntrypointOptions = {
|
|
291
|
+
txNonce: Fr.random(),
|
|
292
|
+
cancellable: this.cancellableTransactions,
|
|
293
|
+
// If from is an address, feeOptions include the way the account contract should handle the fee payment
|
|
294
|
+
feePaymentMethodOptions: feeOptions.accountFeePaymentMethodOptions!,
|
|
295
|
+
};
|
|
296
|
+
txRequest = await fromAccount.createTxExecutionRequest(
|
|
297
|
+
finalExecutionPayload,
|
|
298
|
+
feeOptions.gasSettings,
|
|
299
|
+
chainInfo,
|
|
300
|
+
executionOptions,
|
|
301
|
+
);
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
const result = await this.pxe.simulateTx(txRequest, {
|
|
267
305
|
simulatePublic: true,
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
306
|
+
skipKernels,
|
|
307
|
+
skipFeeEnforcement,
|
|
308
|
+
skipTxValidation,
|
|
309
|
+
overrides,
|
|
271
310
|
scopes,
|
|
272
311
|
});
|
|
312
|
+
const appCallOffset = await this.computeAppCallOffset(from, feeOptions);
|
|
313
|
+
return TxSimulationResultWithAppOffset.fromResultAndOffset(result, appCallOffset);
|
|
273
314
|
}
|
|
274
315
|
|
|
275
316
|
async proveTx(exec: ExecutionPayload, opts: Omit<SendOptions, 'wait'>): Promise<ProvenTx> {
|
|
276
|
-
const fee = await this.completeFeeOptions(
|
|
317
|
+
const fee = await this.completeFeeOptions({
|
|
318
|
+
from: opts.from,
|
|
319
|
+
feePayer: exec.feePayer,
|
|
320
|
+
gasSettings: opts.fee?.gasSettings,
|
|
321
|
+
});
|
|
277
322
|
const txRequest = await this.createTxExecutionRequestFromPayloadAndFee(exec, opts.from, fee);
|
|
278
323
|
const txProvingResult = await this.pxe.proveTx(txRequest, this.scopesFrom(opts.from, opts.additionalScopes));
|
|
279
324
|
return new ProvenTx(
|
|
@@ -1,43 +1,60 @@
|
|
|
1
1
|
import { createAztecNodeClient } from '@aztec/aztec.js/node';
|
|
2
|
+
import type { SendOptions } from '@aztec/aztec.js/wallet';
|
|
2
3
|
import { jsonStringify } from '@aztec/foundation/json-rpc';
|
|
3
|
-
import
|
|
4
|
+
import { createLogger } from '@aztec/foundation/log';
|
|
5
|
+
import type { ApiSchema, Fr } from '@aztec/foundation/schemas';
|
|
4
6
|
import { parseWithOptionals, schemaHasMethod } from '@aztec/foundation/schemas';
|
|
5
7
|
import { NodeListener, TransportServer } from '@aztec/foundation/transport';
|
|
8
|
+
import { ExecutionPayload, Tx } from '@aztec/stdlib/tx';
|
|
6
9
|
|
|
7
10
|
import { workerData } from 'worker_threads';
|
|
8
11
|
|
|
9
12
|
import { TestWallet } from './test_wallet.js';
|
|
10
13
|
import { WorkerWalletSchema } from './worker_wallet_schema.js';
|
|
11
14
|
|
|
12
|
-
const
|
|
15
|
+
const logger = createLogger('e2e:test-wallet:worker');
|
|
13
16
|
|
|
14
|
-
|
|
15
|
-
const
|
|
17
|
+
try {
|
|
18
|
+
const { nodeUrl, pxeConfig } = workerData as { nodeUrl: string; pxeConfig?: Record<string, unknown> };
|
|
16
19
|
|
|
17
|
-
|
|
18
|
-
const
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
// ProvenTx has non-serializable fields (node proxy, etc.) — extract only Tx-compatible fields
|
|
22
|
-
const { data, chonkProof, contractClassLogFields, publicFunctionCalldata } = provenTx;
|
|
23
|
-
return { data, chonkProof, contractClassLogFields, publicFunctionCalldata };
|
|
24
|
-
},
|
|
25
|
-
registerAccount: async (secret, salt) => {
|
|
26
|
-
const manager = await wallet.createSchnorrAccount(secret, salt);
|
|
27
|
-
return manager.address;
|
|
28
|
-
},
|
|
29
|
-
};
|
|
20
|
+
logger.info('Initializing worker wallet', { nodeUrl });
|
|
21
|
+
const node = createAztecNodeClient(nodeUrl);
|
|
22
|
+
const wallet = await TestWallet.create(node, pxeConfig);
|
|
23
|
+
logger.info('Worker wallet initialized');
|
|
30
24
|
|
|
31
|
-
const
|
|
32
|
-
|
|
33
|
-
const
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
25
|
+
const customMethods = {
|
|
26
|
+
proveTx: async (exec: ExecutionPayload, opts: Omit<SendOptions, 'wait'>) => {
|
|
27
|
+
const provenTx = await wallet.proveTx(exec, opts);
|
|
28
|
+
return new Tx(
|
|
29
|
+
provenTx.getTxHash(),
|
|
30
|
+
provenTx.data,
|
|
31
|
+
provenTx.chonkProof,
|
|
32
|
+
provenTx.contractClassLogFields,
|
|
33
|
+
provenTx.publicFunctionCalldata,
|
|
34
|
+
);
|
|
35
|
+
},
|
|
36
|
+
registerAccount: async (secret: Fr, salt: Fr) => {
|
|
37
|
+
const manager = await wallet.createSchnorrAccount(secret, salt);
|
|
38
|
+
return manager.address;
|
|
39
|
+
},
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
const schema = WorkerWalletSchema as ApiSchema;
|
|
43
|
+
const listener = new NodeListener();
|
|
44
|
+
const server = new TransportServer<{ fn: string; args: string }>(listener, async msg => {
|
|
45
|
+
if (!schemaHasMethod(schema, msg.fn)) {
|
|
46
|
+
throw new Error(`Unknown method: ${msg.fn}`);
|
|
47
|
+
}
|
|
48
|
+
const jsonParams = JSON.parse(msg.args) as unknown[];
|
|
49
|
+
const args: any[] = await parseWithOptionals(jsonParams, schema[msg.fn].parameters());
|
|
50
|
+
// we have to erase the fn type in order to be able to spread ...args
|
|
51
|
+
const handler: ((...args: any[]) => Promise<any>) | undefined =
|
|
52
|
+
msg.fn in customMethods ? customMethods[msg.fn as keyof typeof customMethods] : undefined;
|
|
53
|
+
const result = handler ? await handler(...args) : await (wallet as any)[msg.fn](...args);
|
|
54
|
+
return jsonStringify(result);
|
|
55
|
+
});
|
|
56
|
+
server.start();
|
|
57
|
+
} catch (err: unknown) {
|
|
58
|
+
logger.error('Worker wallet initialization failed', { error: err instanceof Error ? err.stack : String(err) });
|
|
59
|
+
process.exit(1);
|
|
60
|
+
}
|