@aztec/cli-wallet 5.0.0-private.20260318 → 5.0.0-rc.1
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/index.js +1 -1
- package/dest/cmds/check_tx.d.ts +1 -1
- package/dest/cmds/check_tx.d.ts.map +1 -1
- package/dest/cmds/check_tx.js +41 -15
- package/dest/cmds/create_account.d.ts +4 -3
- package/dest/cmds/create_account.d.ts.map +1 -1
- package/dest/cmds/create_account.js +30 -28
- package/dest/cmds/deploy.d.ts +5 -4
- package/dest/cmds/deploy.d.ts.map +1 -1
- package/dest/cmds/deploy.js +43 -40
- package/dest/cmds/deploy_account.d.ts +4 -3
- package/dest/cmds/deploy_account.d.ts.map +1 -1
- package/dest/cmds/deploy_account.js +22 -22
- package/dest/cmds/get_fee_juice_balance.d.ts +5 -0
- package/dest/cmds/get_fee_juice_balance.d.ts.map +1 -0
- package/dest/cmds/get_fee_juice_balance.js +27 -0
- package/dest/cmds/import_test_accounts.d.ts +1 -1
- package/dest/cmds/import_test_accounts.d.ts.map +1 -1
- package/dest/cmds/import_test_accounts.js +1 -1
- package/dest/cmds/index.d.ts +1 -1
- package/dest/cmds/index.d.ts.map +1 -1
- package/dest/cmds/index.js +31 -15
- package/dest/cmds/send.d.ts +4 -3
- package/dest/cmds/send.d.ts.map +1 -1
- package/dest/cmds/send.js +26 -28
- package/dest/storage/wallet_db.d.ts +7 -2
- package/dest/storage/wallet_db.d.ts.map +1 -1
- package/dest/storage/wallet_db.js +14 -0
- package/dest/utils/constants.d.ts +2 -2
- package/dest/utils/constants.d.ts.map +1 -1
- package/dest/utils/constants.js +1 -0
- package/dest/utils/options/fees.d.ts +3 -2
- package/dest/utils/options/fees.d.ts.map +1 -1
- package/dest/utils/options/fees.js +33 -9
- package/dest/utils/wallet.d.ts +21 -8
- package/dest/utils/wallet.d.ts.map +1 -1
- package/dest/utils/wallet.js +133 -60
- package/package.json +16 -15
- package/src/bin/index.ts +1 -1
- package/src/cmds/check_tx.ts +46 -14
- package/src/cmds/create_account.ts +31 -20
- package/src/cmds/deploy.ts +40 -35
- package/src/cmds/deploy_account.ts +22 -14
- package/src/cmds/get_fee_juice_balance.ts +37 -0
- package/src/cmds/import_test_accounts.ts +5 -1
- package/src/cmds/index.ts +59 -5
- package/src/cmds/send.ts +23 -18
- package/src/storage/wallet_db.ts +14 -0
- package/src/utils/constants.ts +7 -1
- package/src/utils/options/fees.ts +35 -9
- package/src/utils/wallet.ts +144 -80
package/dest/utils/wallet.js
CHANGED
|
@@ -1,60 +1,109 @@
|
|
|
1
1
|
import { EcdsaRAccountContract, EcdsaRSSHAccountContract } from '@aztec/accounts/ecdsa';
|
|
2
|
-
import {
|
|
3
|
-
import {
|
|
2
|
+
import { StubEcdsaAccountContractArtifact, createStubEcdsaAccount } from '@aztec/accounts/ecdsa/stub';
|
|
3
|
+
import { SchnorrAccountContract, SchnorrInitializerlessAccountContract } from '@aztec/accounts/schnorr';
|
|
4
|
+
import { StubSchnorrAccountContractArtifact, createStubSchnorrAccount } from '@aztec/accounts/schnorr/stub';
|
|
4
5
|
import { getIdentities } from '@aztec/accounts/utils';
|
|
5
|
-
import {
|
|
6
|
-
import {
|
|
6
|
+
import { NO_FROM } from '@aztec/aztec.js/account';
|
|
7
|
+
import { ContractFunctionInteraction, getContractClassFromArtifact } from '@aztec/aztec.js/contracts';
|
|
7
8
|
import { AccountManager } from '@aztec/aztec.js/wallet';
|
|
9
|
+
import { TxSimulationResultWithAppOffset } from '@aztec/aztec.js/wallet';
|
|
10
|
+
import { DefaultEntrypoint } from '@aztec/entrypoints/default';
|
|
8
11
|
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
9
12
|
import { createPXE, getPXEConfig } from '@aztec/pxe/server';
|
|
13
|
+
import { getStandardAuthRegistry } from '@aztec/standard-contracts/auth-registry';
|
|
10
14
|
import { AztecAddress } from '@aztec/stdlib/aztec-address';
|
|
11
15
|
import { deriveSigningKey } from '@aztec/stdlib/keys';
|
|
12
16
|
import { ExecutionPayload, mergeExecutionPayloads } from '@aztec/stdlib/tx';
|
|
13
|
-
import { BaseWallet } from '@aztec/wallet-sdk/base-wallet';
|
|
17
|
+
import { BaseWallet, getGasLimits } from '@aztec/wallet-sdk/base-wallet';
|
|
14
18
|
import { extractECDSAPublicKeyFromBase64String } from './ecdsa.js';
|
|
15
|
-
|
|
19
|
+
/** Padding the CLI wallet applies to simulated gas usage when deriving declared gas limits. */ const DEFAULT_ESTIMATED_GAS_PADDING = 0.1;
|
|
16
20
|
export class CLIWallet extends BaseWallet {
|
|
17
21
|
userLog;
|
|
18
22
|
db;
|
|
19
23
|
accountCache;
|
|
24
|
+
// Stub class ids, populated on wallet startup
|
|
25
|
+
// to avoid redundant work per simulation
|
|
26
|
+
stubClassIds;
|
|
20
27
|
constructor(pxe, node, userLog, db){
|
|
21
|
-
super(pxe, node), this.userLog = userLog, this.db = db, this.accountCache = new Map();
|
|
28
|
+
super(pxe, node), this.userLog = userLog, this.db = db, this.accountCache = new Map(), this.stubClassIds = new Map();
|
|
22
29
|
this.cancellableTransactions = true;
|
|
23
30
|
}
|
|
24
31
|
static async create(node, log, db, overridePXEConfig) {
|
|
25
32
|
const pxeConfig = Object.assign(getPXEConfig(), overridePXEConfig);
|
|
26
33
|
const pxe = await createPXE(node, pxeConfig);
|
|
27
|
-
|
|
34
|
+
const wallet = new CLIWallet(pxe, node, log, db);
|
|
35
|
+
await wallet.initStubClasses();
|
|
36
|
+
await wallet.registerAuthRegistry();
|
|
37
|
+
return wallet;
|
|
38
|
+
}
|
|
39
|
+
async registerAuthRegistry() {
|
|
40
|
+
const { instance, artifact } = await getStandardAuthRegistry();
|
|
41
|
+
await this.pxe.registerContract({
|
|
42
|
+
instance,
|
|
43
|
+
artifact
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Hashes and registers the stub class for every supported account type with PXE, populating
|
|
48
|
+
* stubClassIds. Called on wallet initialization.
|
|
49
|
+
*/ async initStubClasses() {
|
|
50
|
+
const { id: schnorrClassId } = await getContractClassFromArtifact(StubSchnorrAccountContractArtifact);
|
|
51
|
+
await this.pxe.registerContractClass(StubSchnorrAccountContractArtifact);
|
|
52
|
+
// ecdsa stubs share the same class id
|
|
53
|
+
const { id: ecdsaClassId } = await getContractClassFromArtifact(StubEcdsaAccountContractArtifact);
|
|
54
|
+
await this.pxe.registerContractClass(StubEcdsaAccountContractArtifact);
|
|
55
|
+
this.stubClassIds.set('schnorr', schnorrClassId);
|
|
56
|
+
this.stubClassIds.set('schnorr_initializerless', schnorrClassId);
|
|
57
|
+
this.stubClassIds.set('ecdsasecp256k1', ecdsaClassId);
|
|
58
|
+
this.stubClassIds.set('ecdsasecp256r1', ecdsaClassId);
|
|
59
|
+
this.stubClassIds.set('ecdsasecp256r1ssh', ecdsaClassId);
|
|
28
60
|
}
|
|
29
61
|
async getAccounts() {
|
|
30
62
|
const accounts = await this.db?.listAliases('accounts') ?? [];
|
|
31
|
-
return Promise.resolve(accounts.map(({ key, value })=>
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
63
|
+
return Promise.resolve(accounts.map(({ key, value })=>{
|
|
64
|
+
const alias = key.includes(':') ? key.slice(key.indexOf(':') + 1) : key;
|
|
65
|
+
return {
|
|
66
|
+
alias,
|
|
67
|
+
item: AztecAddress.fromString(value)
|
|
68
|
+
};
|
|
69
|
+
}));
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* Derives suggested total and teardown gas limits from simulated gas usage, padded and clamped to the
|
|
73
|
+
* network's per-tx admission limits.
|
|
74
|
+
* @param gasUsed - The gas consumed during simulation (from a `simulate({ includeMetadata: true })` result).
|
|
75
|
+
*/ async estimateGasLimits(gasUsed) {
|
|
76
|
+
const maxTxGasLimits = await this.getMaxTxGasLimits();
|
|
77
|
+
return getGasLimits(gasUsed, maxTxGasLimits, DEFAULT_ESTIMATED_GAS_PADDING);
|
|
35
78
|
}
|
|
36
79
|
async createCancellationTxExecutionRequest(from, txNonce, increasedFee) {
|
|
37
80
|
const executionPayload = ExecutionPayload.empty();
|
|
38
|
-
const feeOptions = await this.completeFeeOptions(
|
|
81
|
+
const feeOptions = await this.completeFeeOptions({
|
|
82
|
+
from,
|
|
83
|
+
feePayer: executionPayload.feePayer,
|
|
84
|
+
gasSettings: increasedFee.gasSettings
|
|
85
|
+
});
|
|
39
86
|
const feeExecutionPayload = await feeOptions.walletFeePaymentMethod?.getExecutionPayload();
|
|
40
87
|
const fromAccount = await this.getAccountFromAddress(from);
|
|
41
88
|
const chainInfo = await this.getChainInfo();
|
|
42
89
|
const executionOptions = {
|
|
43
90
|
txNonce,
|
|
44
91
|
cancellable: this.cancellableTransactions,
|
|
92
|
+
// If from is an address, feeOptions include the way the account contract should handle the fee payment
|
|
45
93
|
feePaymentMethodOptions: feeOptions.accountFeePaymentMethodOptions
|
|
46
94
|
};
|
|
47
95
|
return await fromAccount.createTxExecutionRequest(feeExecutionPayload ?? executionPayload, feeOptions.gasSettings, chainInfo, executionOptions);
|
|
48
96
|
}
|
|
49
97
|
async proveCancellationTx(from, txNonce, increasedFee) {
|
|
50
98
|
const cancellationTxRequest = await this.createCancellationTxExecutionRequest(from, txNonce, increasedFee);
|
|
51
|
-
return await this.pxe.proveTx(cancellationTxRequest,
|
|
99
|
+
return await this.pxe.proveTx(cancellationTxRequest, {
|
|
100
|
+
scopes: this.scopesFrom(from),
|
|
101
|
+
senderForTags: from
|
|
102
|
+
});
|
|
52
103
|
}
|
|
53
104
|
async getAccountFromAddress(address) {
|
|
54
105
|
let account;
|
|
55
|
-
if (address.
|
|
56
|
-
account = new SignerlessAccount();
|
|
57
|
-
} else if (this.accountCache.has(address.toString())) {
|
|
106
|
+
if (this.accountCache.has(address.toString())) {
|
|
58
107
|
return this.accountCache.get(address.toString());
|
|
59
108
|
} else {
|
|
60
109
|
const accountManager = await this.createOrRetrieveAccount(address);
|
|
@@ -66,11 +115,30 @@ export class CLIWallet extends BaseWallet {
|
|
|
66
115
|
return account;
|
|
67
116
|
}
|
|
68
117
|
async createAccount(secret, salt, contract) {
|
|
69
|
-
const accountManager = await AccountManager.create(this, secret, contract,
|
|
118
|
+
const accountManager = await AccountManager.create(this, secret, contract, {
|
|
119
|
+
salt
|
|
120
|
+
});
|
|
70
121
|
const instance = accountManager.getInstance();
|
|
71
122
|
const artifact = await contract.getContractArtifact();
|
|
72
123
|
await this.registerContract(instance, artifact, secret);
|
|
73
124
|
this.accountCache.set(accountManager.address.toString(), await accountManager.getAccount());
|
|
125
|
+
// Initializerless accounts have no deployment tx; their address commits to the signing public key
|
|
126
|
+
// (via the contract's immutablesHash, resolved by AccountManager.create) and the constructor's
|
|
127
|
+
// storage writes are materialized locally via a simulated "store" call here.
|
|
128
|
+
if (contract instanceof SchnorrInitializerlessAccountContract) {
|
|
129
|
+
const constructorAbi = artifact.functions.find((f)=>f.name === 'constructor');
|
|
130
|
+
if (!constructorAbi) {
|
|
131
|
+
throw new Error('Could not create SchnorrInitializerlessAccount: constructor ABI not found');
|
|
132
|
+
}
|
|
133
|
+
const { x, y } = await contract.getSigningPublicKey();
|
|
134
|
+
const storeCall = new ContractFunctionInteraction(this, instance.address, constructorAbi, [
|
|
135
|
+
x,
|
|
136
|
+
y
|
|
137
|
+
]);
|
|
138
|
+
await storeCall.simulate({
|
|
139
|
+
from: instance.address
|
|
140
|
+
});
|
|
141
|
+
}
|
|
74
142
|
return accountManager;
|
|
75
143
|
}
|
|
76
144
|
async createOrRetrieveAccount(address, secretKey, type = 'schnorr', salt, publicKey) {
|
|
@@ -88,6 +156,11 @@ export class CLIWallet extends BaseWallet {
|
|
|
88
156
|
account = await this.createAccount(secretKey, salt, new SchnorrAccountContract(deriveSigningKey(secretKey)));
|
|
89
157
|
break;
|
|
90
158
|
}
|
|
159
|
+
case 'schnorr_initializerless':
|
|
160
|
+
{
|
|
161
|
+
account = await this.createAccount(secretKey, salt, new SchnorrInitializerlessAccountContract(deriveSigningKey(secretKey)));
|
|
162
|
+
break;
|
|
163
|
+
}
|
|
91
164
|
case 'ecdsasecp256r1':
|
|
92
165
|
{
|
|
93
166
|
account = await this.createAccount(secretKey, salt, new EcdsaRAccountContract(deriveSigningKey(secretKey).toBuffer()));
|
|
@@ -125,70 +198,70 @@ export class CLIWallet extends BaseWallet {
|
|
|
125
198
|
* @returns The stub account, contract instance, and artifact for simulation
|
|
126
199
|
*/ async getFakeAccountDataFor(address) {
|
|
127
200
|
const originalAccount = await this.getAccountFromAddress(address);
|
|
128
|
-
// Account contracts can only be overridden if they have an associated address
|
|
129
|
-
// Overwriting SignerlessAccount is not supported, and does not really make sense
|
|
130
|
-
// since it has no authorization mechanism.
|
|
131
|
-
if (originalAccount instanceof SignerlessAccount) {
|
|
132
|
-
throw new Error(`Cannot create fake account data for SignerlessAccount at address: ${address}`);
|
|
133
|
-
}
|
|
134
201
|
const originalAddress = originalAccount.getCompleteAddress();
|
|
135
202
|
const contractInstance = await this.pxe.getContractInstance(originalAddress.address);
|
|
136
203
|
if (!contractInstance) {
|
|
137
204
|
throw new Error(`No contract instance found for address: ${originalAddress.address}`);
|
|
138
205
|
}
|
|
139
|
-
const
|
|
140
|
-
const
|
|
141
|
-
|
|
142
|
-
|
|
206
|
+
const { type } = await this.db.retrieveAccount(address);
|
|
207
|
+
const stubAccount = type === 'schnorr' || type === 'schnorr_initializerless' ? createStubSchnorrAccount(originalAddress) : createStubEcdsaAccount(originalAddress);
|
|
208
|
+
const stubClassId = this.stubClassIds.get(type);
|
|
209
|
+
if (!stubClassId) {
|
|
210
|
+
throw new Error(`Stub class for account type '${type}' was not registered at wallet init. This is a bug — initStubClasses should cover every supported AccountType.`);
|
|
211
|
+
}
|
|
212
|
+
const instance = {
|
|
213
|
+
...contractInstance,
|
|
214
|
+
currentContractClassId: stubClassId
|
|
215
|
+
};
|
|
143
216
|
return {
|
|
144
217
|
account: stubAccount,
|
|
145
|
-
instance
|
|
146
|
-
artifact: StubAccountContractArtifact
|
|
218
|
+
instance
|
|
147
219
|
};
|
|
148
220
|
}
|
|
149
|
-
async simulateTx(executionPayload, opts) {
|
|
150
|
-
const simulationResults = await super.simulateTx(executionPayload, opts);
|
|
151
|
-
if (opts.fee?.estimateGas) {
|
|
152
|
-
const feeOptions = await this.completeFeeOptions(opts.from, executionPayload.feePayer, opts.fee?.gasSettings);
|
|
153
|
-
const limits = getGasLimits(simulationResults, opts.fee?.estimatedGasPadding);
|
|
154
|
-
printGasEstimates(feeOptions, limits, this.userLog);
|
|
155
|
-
}
|
|
156
|
-
return simulationResults;
|
|
157
|
-
}
|
|
158
221
|
/**
|
|
159
222
|
* Uses a stub account for kernelless simulation, bypassing real account authorization.
|
|
160
|
-
*
|
|
161
|
-
*/ async simulateViaEntrypoint(executionPayload,
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
}
|
|
223
|
+
* Uses DefaultEntrypoint directly for NO_FROM transactions.
|
|
224
|
+
*/ async simulateViaEntrypoint(executionPayload, opts) {
|
|
225
|
+
const { from, feeOptions, additionalScopes, sendMessagesAs } = opts;
|
|
226
|
+
const scopes = this.scopesFrom(from, additionalScopes);
|
|
165
227
|
const feeExecutionPayload = await feeOptions.walletFeePaymentMethod?.getExecutionPayload();
|
|
166
|
-
const executionOptions = {
|
|
167
|
-
txNonce: Fr.random(),
|
|
168
|
-
cancellable: this.cancellableTransactions,
|
|
169
|
-
feePaymentMethodOptions: feeOptions.accountFeePaymentMethodOptions
|
|
170
|
-
};
|
|
171
228
|
const finalExecutionPayload = feeExecutionPayload ? mergeExecutionPayloads([
|
|
172
229
|
feeExecutionPayload,
|
|
173
230
|
executionPayload
|
|
174
231
|
]) : executionPayload;
|
|
175
|
-
const { account: fromAccount, instance, artifact } = await this.getFakeAccountDataFor(from);
|
|
176
232
|
const chainInfo = await this.getChainInfo();
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
233
|
+
let overrides;
|
|
234
|
+
let txRequest;
|
|
235
|
+
if (from === NO_FROM) {
|
|
236
|
+
const entrypoint = new DefaultEntrypoint();
|
|
237
|
+
txRequest = await entrypoint.createTxExecutionRequest(finalExecutionPayload, feeOptions.gasSettings, chainInfo);
|
|
238
|
+
} else {
|
|
239
|
+
const { account, instance } = await this.getFakeAccountDataFor(from);
|
|
240
|
+
overrides = {
|
|
183
241
|
contracts: {
|
|
184
242
|
[from.toString()]: {
|
|
185
|
-
instance
|
|
186
|
-
artifact
|
|
243
|
+
instance
|
|
187
244
|
}
|
|
188
245
|
}
|
|
189
|
-
}
|
|
190
|
-
|
|
246
|
+
};
|
|
247
|
+
const executionOptions = {
|
|
248
|
+
txNonce: Fr.random(),
|
|
249
|
+
cancellable: this.cancellableTransactions,
|
|
250
|
+
// If from is an address, feeOptions include the way the account contract should handle the fee payment
|
|
251
|
+
feePaymentMethodOptions: feeOptions.accountFeePaymentMethodOptions
|
|
252
|
+
};
|
|
253
|
+
txRequest = await account.createTxExecutionRequest(finalExecutionPayload, feeOptions.gasSettings, chainInfo, executionOptions);
|
|
254
|
+
}
|
|
255
|
+
const result = await this.pxe.simulateTx(txRequest, {
|
|
256
|
+
simulatePublic: true,
|
|
257
|
+
skipFeeEnforcement: true,
|
|
258
|
+
skipTxValidation: true,
|
|
259
|
+
overrides,
|
|
260
|
+
scopes,
|
|
261
|
+
senderForTags: this.senderForTagsFrom(from, sendMessagesAs)
|
|
191
262
|
});
|
|
263
|
+
const appCallOffset = await this.computeAppCallOffset(from, feeOptions);
|
|
264
|
+
return TxSimulationResultWithAppOffset.fromResultAndOffset(result, appCallOffset);
|
|
192
265
|
}
|
|
193
266
|
// Exposed because of the `aztec-wallet get-tx` command. It has been decided that it's fine to keep around because
|
|
194
267
|
// this is just a CLI wallet.
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@aztec/cli-wallet",
|
|
3
|
-
"version": "5.0.0-
|
|
3
|
+
"version": "5.0.0-rc.1",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"exports": {
|
|
6
6
|
".": "./dest/cmds/index.js",
|
|
@@ -19,7 +19,7 @@
|
|
|
19
19
|
"scripts": {
|
|
20
20
|
"start": "node --no-warnings ./dest/bin",
|
|
21
21
|
"start:debug": "node --inspect=0.0.0.0:9221 --no-warnings ./dest/bin",
|
|
22
|
-
"dev": "LOG_LEVEL
|
|
22
|
+
"dev": "LOG_LEVEL=\"debug; info: json-rpc, simulator\" node ./dest/bin",
|
|
23
23
|
"build": "yarn clean && ../scripts/tsc.sh",
|
|
24
24
|
"build:dev": "../scripts/tsc.sh --watch",
|
|
25
25
|
"clean": "rm -rf ./dest .tsbuildinfo",
|
|
@@ -67,19 +67,20 @@
|
|
|
67
67
|
]
|
|
68
68
|
},
|
|
69
69
|
"dependencies": {
|
|
70
|
-
"@aztec/accounts": "5.0.0-
|
|
71
|
-
"@aztec/aztec.js": "5.0.0-
|
|
72
|
-
"@aztec/bb.js": "5.0.0-
|
|
73
|
-
"@aztec/cli": "5.0.0-
|
|
74
|
-
"@aztec/entrypoints": "5.0.0-
|
|
75
|
-
"@aztec/ethereum": "5.0.0-
|
|
76
|
-
"@aztec/foundation": "5.0.0-
|
|
77
|
-
"@aztec/kv-store": "5.0.0-
|
|
78
|
-
"@aztec/noir-contracts.js": "5.0.0-
|
|
79
|
-
"@aztec/noir-noirc_abi": "5.0.0-
|
|
80
|
-
"@aztec/pxe": "5.0.0-
|
|
81
|
-
"@aztec/
|
|
82
|
-
"@aztec/
|
|
70
|
+
"@aztec/accounts": "5.0.0-rc.1",
|
|
71
|
+
"@aztec/aztec.js": "5.0.0-rc.1",
|
|
72
|
+
"@aztec/bb.js": "5.0.0-rc.1",
|
|
73
|
+
"@aztec/cli": "5.0.0-rc.1",
|
|
74
|
+
"@aztec/entrypoints": "5.0.0-rc.1",
|
|
75
|
+
"@aztec/ethereum": "5.0.0-rc.1",
|
|
76
|
+
"@aztec/foundation": "5.0.0-rc.1",
|
|
77
|
+
"@aztec/kv-store": "5.0.0-rc.1",
|
|
78
|
+
"@aztec/noir-contracts.js": "5.0.0-rc.1",
|
|
79
|
+
"@aztec/noir-noirc_abi": "5.0.0-rc.1",
|
|
80
|
+
"@aztec/pxe": "5.0.0-rc.1",
|
|
81
|
+
"@aztec/standard-contracts": "5.0.0-rc.1",
|
|
82
|
+
"@aztec/stdlib": "5.0.0-rc.1",
|
|
83
|
+
"@aztec/wallet-sdk": "5.0.0-rc.1",
|
|
83
84
|
"commander": "^12.1.0",
|
|
84
85
|
"inquirer": "^10.1.8",
|
|
85
86
|
"source-map-support": "^0.5.21",
|
package/src/bin/index.ts
CHANGED
|
@@ -73,7 +73,7 @@ function injectInternalCommands(program: Command, log: LogFn, db: WalletDB) {
|
|
|
73
73
|
|
|
74
74
|
/** CLI wallet main entrypoint */
|
|
75
75
|
async function main() {
|
|
76
|
-
const walletVersion = getPackageVersion()
|
|
76
|
+
const walletVersion = getPackageVersion();
|
|
77
77
|
|
|
78
78
|
const db = WalletDB.getInstance();
|
|
79
79
|
const walletAndNodeWrapper = new CliWalletAndNodeWrapper();
|
package/src/cmds/check_tx.ts
CHANGED
|
@@ -1,11 +1,15 @@
|
|
|
1
1
|
import type { ContractArtifact } from '@aztec/aztec.js/abi';
|
|
2
|
-
import
|
|
2
|
+
import { AztecAddress } from '@aztec/aztec.js/addresses';
|
|
3
3
|
import { Fr } from '@aztec/aztec.js/fields';
|
|
4
4
|
import type { AztecNode } from '@aztec/aztec.js/node';
|
|
5
5
|
import { ProtocolContractAddress } from '@aztec/aztec.js/protocol';
|
|
6
6
|
import type { TxHash } from '@aztec/aztec.js/tx';
|
|
7
7
|
import type { LogFn } from '@aztec/foundation/log';
|
|
8
|
-
import {
|
|
8
|
+
import {
|
|
9
|
+
computeSiloedPrivateInitializationNullifier,
|
|
10
|
+
computeSiloedPublicInitializationNullifier,
|
|
11
|
+
siloNullifier,
|
|
12
|
+
} from '@aztec/stdlib/hash';
|
|
9
13
|
import { NoteDao } from '@aztec/stdlib/note';
|
|
10
14
|
|
|
11
15
|
import type { CLIWallet } from '../utils/wallet.js';
|
|
@@ -29,7 +33,7 @@ export async function checkTx(
|
|
|
29
33
|
// dependency when dropping PXE JSON RPC Server.
|
|
30
34
|
|
|
31
35
|
async function inspectTx(wallet: CLIWallet, aztecNode: AztecNode, txHash: TxHash, log: LogFn) {
|
|
32
|
-
const
|
|
36
|
+
const receipt = await aztecNode.getTxReceipt(txHash, { includeTxEffect: true });
|
|
33
37
|
// Base tx data
|
|
34
38
|
log(`Tx ${txHash.toString()}`);
|
|
35
39
|
log(` Status: ${receipt.status}`);
|
|
@@ -40,11 +44,11 @@ async function inspectTx(wallet: CLIWallet, aztecNode: AztecNode, txHash: TxHash
|
|
|
40
44
|
log(` Error: ${receipt.error}`);
|
|
41
45
|
}
|
|
42
46
|
|
|
43
|
-
if (!
|
|
47
|
+
if (!receipt.isMined() || !receipt.txEffect) {
|
|
44
48
|
return;
|
|
45
49
|
}
|
|
46
50
|
|
|
47
|
-
const effects =
|
|
51
|
+
const effects = receipt.txEffect;
|
|
48
52
|
const artifactMap = await getKnownArtifacts(wallet);
|
|
49
53
|
|
|
50
54
|
log(` Block: ${receipt.blockNumber} (${receipt.blockHash?.toString()})`);
|
|
@@ -83,12 +87,13 @@ async function inspectTx(wallet: CLIWallet, aztecNode: AztecNode, txHash: TxHash
|
|
|
83
87
|
// Nullifiers
|
|
84
88
|
const nullifierCount = effects.nullifiers.length;
|
|
85
89
|
const { deployNullifiers, initNullifiers, classNullifiers } = await getKnownNullifiers(wallet, artifactMap);
|
|
90
|
+
const accounts = (await wallet.getAccounts()).map(a => a.item);
|
|
86
91
|
if (nullifierCount > 0) {
|
|
87
92
|
log(' Nullifiers:');
|
|
88
93
|
for (const nullifier of effects.nullifiers) {
|
|
89
94
|
const deployed = deployNullifiers[nullifier.toString()];
|
|
90
95
|
const note = deployed
|
|
91
|
-
? (await wallet.getNotes({ siloedNullifier: nullifier, contractAddress: deployed, scopes:
|
|
96
|
+
? (await wallet.getNotes({ siloedNullifier: nullifier, contractAddress: deployed, scopes: accounts }))[0]
|
|
92
97
|
: undefined;
|
|
93
98
|
const initialized = initNullifiers[nullifier.toString()];
|
|
94
99
|
const registered = classNullifiers[nullifier.toString()];
|
|
@@ -144,22 +149,49 @@ function toFriendlyAddress(address: AztecAddress, artifactMap: ArtifactMap) {
|
|
|
144
149
|
|
|
145
150
|
async function getKnownNullifiers(wallet: CLIWallet, artifactMap: ArtifactMap) {
|
|
146
151
|
const knownContracts = await wallet.getContracts();
|
|
147
|
-
|
|
148
|
-
const
|
|
152
|
+
|
|
153
|
+
const [contractResults, classResults] = await Promise.all([
|
|
154
|
+
Promise.all(knownContracts.map(contract => getContractNullifiers(wallet, contract))),
|
|
155
|
+
Promise.all(Object.values(artifactMap).map(artifact => getClassNullifier(artifact))),
|
|
156
|
+
]);
|
|
157
|
+
|
|
149
158
|
const initNullifiers: Record<string, AztecAddress> = {};
|
|
150
159
|
const deployNullifiers: Record<string, AztecAddress> = {};
|
|
151
160
|
const classNullifiers: Record<string, string> = {};
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
deployNullifiers[
|
|
161
|
+
|
|
162
|
+
for (const { contract, deployNullifier, privateInitNullifier, publicInitNullifier } of contractResults) {
|
|
163
|
+
deployNullifiers[deployNullifier.toString()] = contract;
|
|
164
|
+
if (privateInitNullifier) {
|
|
165
|
+
initNullifiers[privateInitNullifier.toString()] = contract;
|
|
166
|
+
}
|
|
167
|
+
initNullifiers[publicInitNullifier.toString()] = contract;
|
|
155
168
|
}
|
|
156
|
-
for (const
|
|
157
|
-
classNullifiers[
|
|
158
|
-
`${artifact.name}Class<${artifact.classId}>`;
|
|
169
|
+
for (const { nullifier, label } of classResults) {
|
|
170
|
+
classNullifiers[nullifier.toString()] = label;
|
|
159
171
|
}
|
|
172
|
+
|
|
160
173
|
return { initNullifiers, deployNullifiers, classNullifiers };
|
|
161
174
|
}
|
|
162
175
|
|
|
176
|
+
async function getContractNullifiers(wallet: CLIWallet, contract: AztecAddress) {
|
|
177
|
+
const deployerAddress = ProtocolContractAddress.ContractInstanceRegistry;
|
|
178
|
+
const deployNullifier = await siloNullifier(deployerAddress, contract.toField());
|
|
179
|
+
|
|
180
|
+
const metadata = await wallet.getContractMetadata(contract);
|
|
181
|
+
const privateInitNullifier = metadata.instance
|
|
182
|
+
? await computeSiloedPrivateInitializationNullifier(contract, metadata.instance.initializationHash)
|
|
183
|
+
: undefined;
|
|
184
|
+
const publicInitNullifier = await computeSiloedPublicInitializationNullifier(contract);
|
|
185
|
+
|
|
186
|
+
return { contract, deployNullifier, privateInitNullifier, publicInitNullifier };
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
async function getClassNullifier(artifact: ContractArtifactWithClassId) {
|
|
190
|
+
const classRegistryAddress = ProtocolContractAddress.ContractClassRegistry;
|
|
191
|
+
const nullifier = await siloNullifier(classRegistryAddress, artifact.classId);
|
|
192
|
+
return { nullifier, label: `${artifact.name}Class<${artifact.classId}>` };
|
|
193
|
+
}
|
|
194
|
+
|
|
163
195
|
type ArtifactMap = Record<string, ContractArtifactWithClassId>;
|
|
164
196
|
type ContractArtifactWithClassId = ContractArtifact & { classId: Fr };
|
|
165
197
|
|
|
@@ -1,11 +1,12 @@
|
|
|
1
|
+
import { NO_FROM } from '@aztec/aztec.js/account';
|
|
1
2
|
import { AztecAddress } from '@aztec/aztec.js/addresses';
|
|
2
3
|
import { NO_WAIT } from '@aztec/aztec.js/contracts';
|
|
3
|
-
import type
|
|
4
|
+
import { type AztecNode, waitForTx } from '@aztec/aztec.js/node';
|
|
4
5
|
import type { DeployAccountOptions } from '@aztec/aztec.js/wallet';
|
|
5
6
|
import { prettyPrintJSON } from '@aztec/cli/cli-utils';
|
|
6
7
|
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
7
8
|
import type { LogFn, Logger } from '@aztec/foundation/log';
|
|
8
|
-
import type
|
|
9
|
+
import { type TxHash, type TxReceipt, TxStatus } from '@aztec/stdlib/tx';
|
|
9
10
|
|
|
10
11
|
import { DEFAULT_TX_TIMEOUT_S } from '../utils/cli_wallet_and_node_wrapper.js';
|
|
11
12
|
import type { AccountType } from '../utils/constants.js';
|
|
@@ -18,6 +19,7 @@ export async function createAccount(
|
|
|
18
19
|
aztecNode: AztecNode,
|
|
19
20
|
accountType: AccountType,
|
|
20
21
|
secretKey: Fr | undefined,
|
|
22
|
+
salt: Fr | undefined,
|
|
21
23
|
publicKey: string | undefined,
|
|
22
24
|
alias: string | undefined,
|
|
23
25
|
deployer: AztecAddress | undefined,
|
|
@@ -27,6 +29,7 @@ export async function createAccount(
|
|
|
27
29
|
registerClass: boolean,
|
|
28
30
|
wait: boolean,
|
|
29
31
|
feeOpts: CLIFeeArgs,
|
|
32
|
+
waitForStatus: TxStatus,
|
|
30
33
|
json: boolean,
|
|
31
34
|
verbose: boolean,
|
|
32
35
|
debugLogger: Logger,
|
|
@@ -38,10 +41,10 @@ export async function createAccount(
|
|
|
38
41
|
undefined /* address, we don't have it yet */,
|
|
39
42
|
secretKey,
|
|
40
43
|
accountType,
|
|
41
|
-
|
|
44
|
+
salt,
|
|
42
45
|
publicKey,
|
|
43
46
|
);
|
|
44
|
-
const
|
|
47
|
+
const instanceSalt = account.getInstance().salt;
|
|
45
48
|
const { address, publicKeys, partialAddress } = await account.getCompleteAddress();
|
|
46
49
|
|
|
47
50
|
const out: Record<string, any> = {};
|
|
@@ -52,7 +55,7 @@ export async function createAccount(
|
|
|
52
55
|
out.secretKey = secretKey;
|
|
53
56
|
}
|
|
54
57
|
out.partialAddress = partialAddress;
|
|
55
|
-
out.salt =
|
|
58
|
+
out.salt = instanceSalt;
|
|
56
59
|
out.initHash = account.getInstance().initializationHash;
|
|
57
60
|
} else {
|
|
58
61
|
log(`\nNew account:\n`);
|
|
@@ -62,17 +65,19 @@ export async function createAccount(
|
|
|
62
65
|
log(`Secret key: ${secretKey.toString()}`);
|
|
63
66
|
}
|
|
64
67
|
log(`Partial address: ${partialAddress.toString()}`);
|
|
65
|
-
log(`Salt: ${
|
|
68
|
+
log(`Salt: ${instanceSalt.toString()}`);
|
|
66
69
|
log(`Init hash: ${account.getInstance().initializationHash.toString()}`);
|
|
67
70
|
}
|
|
68
71
|
|
|
69
72
|
let txHash: TxHash | undefined;
|
|
70
73
|
let txReceipt: TxReceipt | undefined;
|
|
71
|
-
|
|
74
|
+
// Initializerless accounts have no deployment tx — creating one only registers it locally — so there is
|
|
75
|
+
// nothing to deploy on-chain.
|
|
76
|
+
if (!registerOnly && accountType !== 'schnorr_initializerless') {
|
|
72
77
|
const { paymentMethod, gasSettings } = await feeOpts.toUserFeeOptions(aztecNode, wallet, address);
|
|
73
78
|
|
|
74
79
|
const delegatedDeployment = deployer && !account.address.equals(deployer);
|
|
75
|
-
const from = delegatedDeployment ? deployer :
|
|
80
|
+
const from = delegatedDeployment ? deployer : NO_FROM;
|
|
76
81
|
|
|
77
82
|
const deployAccountOpts: DeployAccountOptions = {
|
|
78
83
|
skipClassPublication: !registerClass,
|
|
@@ -82,13 +87,14 @@ export async function createAccount(
|
|
|
82
87
|
fee: { paymentMethod, gasSettings },
|
|
83
88
|
};
|
|
84
89
|
|
|
90
|
+
const localStart = performance.now();
|
|
85
91
|
const deployMethod = await account.getDeployMethod();
|
|
86
92
|
const sim = await deployMethod.simulate({
|
|
87
93
|
...deployAccountOpts,
|
|
88
|
-
|
|
94
|
+
includeMetadata: true,
|
|
89
95
|
});
|
|
90
|
-
//
|
|
91
|
-
const estimatedGas = sim.
|
|
96
|
+
// includeMetadata: true guarantees these fields are present
|
|
97
|
+
const estimatedGas = await wallet.estimateGasLimits(sim.gasUsed!);
|
|
92
98
|
const stats = sim.stats!;
|
|
93
99
|
|
|
94
100
|
if (feeOpts.estimateOnly) {
|
|
@@ -121,19 +127,24 @@ export async function createAccount(
|
|
|
121
127
|
}
|
|
122
128
|
: undefined,
|
|
123
129
|
};
|
|
130
|
+
|
|
131
|
+
({ txHash } = await deployMethod.send({ ...sendOpts, wait: NO_WAIT }));
|
|
132
|
+
const localTimeMs = performance.now() - localStart;
|
|
133
|
+
|
|
124
134
|
if (wait) {
|
|
125
|
-
const
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
txReceipt = receipt;
|
|
130
|
-
txHash = receipt.txHash;
|
|
135
|
+
const nodeStart = performance.now();
|
|
136
|
+
txReceipt = await waitForTx(aztecNode, txHash, { timeout: DEFAULT_TX_TIMEOUT_S, waitForStatus });
|
|
137
|
+
const nodeTimeMs = performance.now() - nodeStart;
|
|
138
|
+
|
|
131
139
|
out.txReceipt = {
|
|
132
140
|
status: txReceipt.status,
|
|
133
141
|
transactionFee: txReceipt.transactionFee,
|
|
134
142
|
};
|
|
135
|
-
|
|
136
|
-
(
|
|
143
|
+
|
|
144
|
+
if (!json) {
|
|
145
|
+
log(` Local processing time: ${(localTimeMs / 1000).toFixed(1)}s`);
|
|
146
|
+
log(` Node inclusion time: ${(nodeTimeMs / 1000).toFixed(1)}s`);
|
|
147
|
+
}
|
|
137
148
|
}
|
|
138
149
|
debugLogger.debug(`Account contract tx sent with hash ${txHash.toString()}`);
|
|
139
150
|
out.txHash = txHash;
|
|
@@ -151,5 +162,5 @@ export async function createAccount(
|
|
|
151
162
|
}
|
|
152
163
|
}
|
|
153
164
|
|
|
154
|
-
return { alias, address, secretKey, salt };
|
|
165
|
+
return { alias, address, secretKey, salt: instanceSalt };
|
|
155
166
|
}
|