@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.
Files changed (51) hide show
  1. package/dest/bin/index.js +1 -1
  2. package/dest/cmds/check_tx.d.ts +1 -1
  3. package/dest/cmds/check_tx.d.ts.map +1 -1
  4. package/dest/cmds/check_tx.js +41 -15
  5. package/dest/cmds/create_account.d.ts +4 -3
  6. package/dest/cmds/create_account.d.ts.map +1 -1
  7. package/dest/cmds/create_account.js +30 -28
  8. package/dest/cmds/deploy.d.ts +5 -4
  9. package/dest/cmds/deploy.d.ts.map +1 -1
  10. package/dest/cmds/deploy.js +43 -40
  11. package/dest/cmds/deploy_account.d.ts +4 -3
  12. package/dest/cmds/deploy_account.d.ts.map +1 -1
  13. package/dest/cmds/deploy_account.js +22 -22
  14. package/dest/cmds/get_fee_juice_balance.d.ts +5 -0
  15. package/dest/cmds/get_fee_juice_balance.d.ts.map +1 -0
  16. package/dest/cmds/get_fee_juice_balance.js +27 -0
  17. package/dest/cmds/import_test_accounts.d.ts +1 -1
  18. package/dest/cmds/import_test_accounts.d.ts.map +1 -1
  19. package/dest/cmds/import_test_accounts.js +1 -1
  20. package/dest/cmds/index.d.ts +1 -1
  21. package/dest/cmds/index.d.ts.map +1 -1
  22. package/dest/cmds/index.js +31 -15
  23. package/dest/cmds/send.d.ts +4 -3
  24. package/dest/cmds/send.d.ts.map +1 -1
  25. package/dest/cmds/send.js +26 -28
  26. package/dest/storage/wallet_db.d.ts +7 -2
  27. package/dest/storage/wallet_db.d.ts.map +1 -1
  28. package/dest/storage/wallet_db.js +14 -0
  29. package/dest/utils/constants.d.ts +2 -2
  30. package/dest/utils/constants.d.ts.map +1 -1
  31. package/dest/utils/constants.js +1 -0
  32. package/dest/utils/options/fees.d.ts +3 -2
  33. package/dest/utils/options/fees.d.ts.map +1 -1
  34. package/dest/utils/options/fees.js +33 -9
  35. package/dest/utils/wallet.d.ts +21 -8
  36. package/dest/utils/wallet.d.ts.map +1 -1
  37. package/dest/utils/wallet.js +133 -60
  38. package/package.json +16 -15
  39. package/src/bin/index.ts +1 -1
  40. package/src/cmds/check_tx.ts +46 -14
  41. package/src/cmds/create_account.ts +31 -20
  42. package/src/cmds/deploy.ts +40 -35
  43. package/src/cmds/deploy_account.ts +22 -14
  44. package/src/cmds/get_fee_juice_balance.ts +37 -0
  45. package/src/cmds/import_test_accounts.ts +5 -1
  46. package/src/cmds/index.ts +59 -5
  47. package/src/cmds/send.ts +23 -18
  48. package/src/storage/wallet_db.ts +14 -0
  49. package/src/utils/constants.ts +7 -1
  50. package/src/utils/options/fees.ts +35 -9
  51. package/src/utils/wallet.ts +144 -80
@@ -1,60 +1,109 @@
1
1
  import { EcdsaRAccountContract, EcdsaRSSHAccountContract } from '@aztec/accounts/ecdsa';
2
- import { SchnorrAccountContract } from '@aztec/accounts/schnorr';
3
- import { StubAccountContractArtifact, createStubAccount } from '@aztec/accounts/stub';
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 { SignerlessAccount } from '@aztec/aztec.js/account';
6
- import { getContractInstanceFromInstantiationParams, getGasLimits } from '@aztec/aztec.js/contracts';
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
- import { printGasEstimates } from './options/fees.js';
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
- return new CLIWallet(pxe, node, log, db);
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
- alias: value,
33
- item: AztecAddress.fromString(key)
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(from, executionPayload.feePayer, increasedFee.gasSettings);
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, this.scopesFrom(from));
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.equals(AztecAddress.ZERO)) {
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, salt);
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 stubAccount = createStubAccount(originalAddress);
140
- const instance = await getContractInstanceFromInstantiationParams(StubAccountContractArtifact, {
141
- salt: Fr.random()
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
- * Falls through to the standard entrypoint path for SignerlessAccount (ZERO address).
161
- */ async simulateViaEntrypoint(executionPayload, from, feeOptions, scopes, skipTxValidation, skipFeeEnforcement) {
162
- if (from.equals(AztecAddress.ZERO)) {
163
- return super.simulateViaEntrypoint(executionPayload, from, feeOptions, scopes, skipTxValidation, skipFeeEnforcement);
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
- const txRequest = await fromAccount.createTxExecutionRequest(finalExecutionPayload, feeOptions.gasSettings, chainInfo, executionOptions);
178
- return this.pxe.simulateTx(txRequest, {
179
- simulatePublic: true,
180
- skipFeeEnforcement: true,
181
- skipTxValidation: true,
182
- overrides: {
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
- scopes
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-private.20260318",
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=debug && node ./dest/bin",
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-private.20260318",
71
- "@aztec/aztec.js": "5.0.0-private.20260318",
72
- "@aztec/bb.js": "5.0.0-private.20260318",
73
- "@aztec/cli": "5.0.0-private.20260318",
74
- "@aztec/entrypoints": "5.0.0-private.20260318",
75
- "@aztec/ethereum": "5.0.0-private.20260318",
76
- "@aztec/foundation": "5.0.0-private.20260318",
77
- "@aztec/kv-store": "5.0.0-private.20260318",
78
- "@aztec/noir-contracts.js": "5.0.0-private.20260318",
79
- "@aztec/noir-noirc_abi": "5.0.0-private.20260318",
80
- "@aztec/pxe": "5.0.0-private.20260318",
81
- "@aztec/stdlib": "5.0.0-private.20260318",
82
- "@aztec/wallet-sdk": "5.0.0-private.20260318",
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() ?? '0.0.0';
76
+ const walletVersion = getPackageVersion();
77
77
 
78
78
  const db = WalletDB.getInstance();
79
79
  const walletAndNodeWrapper = new CliWalletAndNodeWrapper();
@@ -1,11 +1,15 @@
1
1
  import type { ContractArtifact } from '@aztec/aztec.js/abi';
2
- import type { AztecAddress } from '@aztec/aztec.js/addresses';
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 { siloNullifier } from '@aztec/stdlib/hash';
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 [receipt, effectsInBlock] = await Promise.all([aztecNode.getTxReceipt(txHash), aztecNode.getTxEffect(txHash)]);
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 (!effectsInBlock) {
47
+ if (!receipt.isMined() || !receipt.txEffect) {
44
48
  return;
45
49
  }
46
50
 
47
- const effects = effectsInBlock.data;
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: 'ALL_SCOPES' }))[0]
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
- const deployerAddress = ProtocolContractAddress.ContractInstanceRegistry;
148
- const classRegistryAddress = ProtocolContractAddress.ContractClassRegistry;
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
- for (const contract of knownContracts) {
153
- initNullifiers[(await siloNullifier(contract, contract.toField())).toString()] = contract;
154
- deployNullifiers[(await siloNullifier(deployerAddress, contract.toField())).toString()] = contract;
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 artifact of Object.values(artifactMap)) {
157
- classNullifiers[(await siloNullifier(classRegistryAddress, artifact.classId)).toString()] =
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 { AztecNode } from '@aztec/aztec.js/node';
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 { TxHash, TxReceipt } from '@aztec/stdlib/tx';
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
- Fr.ZERO,
44
+ salt,
42
45
  publicKey,
43
46
  );
44
- const { salt } = account.getInstance();
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 = 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: ${salt.toString()}`);
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
- if (!registerOnly) {
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 : AztecAddress.ZERO;
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
- fee: { ...deployAccountOpts.fee, estimateGas: true },
94
+ includeMetadata: true,
89
95
  });
90
- // estimateGas: true guarantees these fields are present
91
- const estimatedGas = sim.estimatedGas!;
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 { receipt } = await deployMethod.send({
126
- ...sendOpts,
127
- wait: { timeout: DEFAULT_TX_TIMEOUT_S, returnReceipt: true },
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
- } else {
136
- ({ txHash } = await deployMethod.send({ ...sendOpts, wait: NO_WAIT }));
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
  }