@aztec/aztec 0.0.1-fake-ceab37513c → 0.0.6-commit.a2d1860fe9

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 (142) hide show
  1. package/README.md +1 -1
  2. package/dest/bin/index.d.ts +1 -1
  3. package/dest/bin/index.js +14 -9
  4. package/dest/cli/admin_api_key_store.d.ts +45 -0
  5. package/dest/cli/admin_api_key_store.d.ts.map +1 -0
  6. package/dest/cli/admin_api_key_store.js +98 -0
  7. package/dest/cli/aztec_start_action.d.ts +1 -1
  8. package/dest/cli/aztec_start_action.d.ts.map +1 -1
  9. package/dest/cli/aztec_start_action.js +59 -34
  10. package/dest/cli/aztec_start_options.d.ts +1 -1
  11. package/dest/cli/aztec_start_options.d.ts.map +1 -1
  12. package/dest/cli/aztec_start_options.js +46 -45
  13. package/dest/cli/cli.d.ts +1 -1
  14. package/dest/cli/cli.js +7 -7
  15. package/dest/cli/cmds/compile.d.ts +4 -0
  16. package/dest/cli/cmds/compile.d.ts.map +1 -0
  17. package/dest/cli/cmds/compile.js +95 -0
  18. package/dest/cli/cmds/migrate_ha_db.d.ts +3 -0
  19. package/dest/cli/cmds/migrate_ha_db.d.ts.map +1 -0
  20. package/dest/cli/cmds/migrate_ha_db.js +27 -0
  21. package/dest/cli/cmds/start_archiver.d.ts +1 -1
  22. package/dest/cli/cmds/start_archiver.d.ts.map +1 -1
  23. package/dest/cli/cmds/start_archiver.js +12 -14
  24. package/dest/cli/cmds/start_bot.d.ts +4 -7
  25. package/dest/cli/cmds/start_bot.d.ts.map +1 -1
  26. package/dest/cli/cmds/start_bot.js +18 -18
  27. package/dest/cli/cmds/start_node.d.ts +1 -1
  28. package/dest/cli/cmds/start_node.d.ts.map +1 -1
  29. package/dest/cli/cmds/start_node.js +71 -27
  30. package/dest/cli/cmds/start_p2p_bootstrap.d.ts +2 -2
  31. package/dest/cli/cmds/start_p2p_bootstrap.d.ts.map +1 -1
  32. package/dest/cli/cmds/start_p2p_bootstrap.js +2 -3
  33. package/dest/cli/cmds/start_prover_agent.d.ts +1 -1
  34. package/dest/cli/cmds/start_prover_agent.d.ts.map +1 -1
  35. package/dest/cli/cmds/start_prover_agent.js +4 -4
  36. package/dest/cli/cmds/start_prover_broker.d.ts +1 -1
  37. package/dest/cli/cmds/start_prover_broker.d.ts.map +1 -1
  38. package/dest/cli/cmds/start_prover_broker.js +4 -4
  39. package/dest/cli/cmds/start_txe.d.ts +1 -1
  40. package/dest/cli/index.d.ts +1 -1
  41. package/dest/cli/preload_crs.d.ts +1 -1
  42. package/dest/cli/release_version.d.ts +1 -1
  43. package/dest/cli/util.d.ts +12 -19
  44. package/dest/cli/util.d.ts.map +1 -1
  45. package/dest/cli/util.js +16 -11
  46. package/dest/cli/versioning.d.ts +1 -1
  47. package/dest/cli/versioning.js +3 -3
  48. package/dest/examples/token.d.ts +1 -1
  49. package/dest/examples/token.js +21 -19
  50. package/dest/examples/util.d.ts +3 -3
  51. package/dest/examples/util.d.ts.map +1 -1
  52. package/dest/examples/util.js +1 -1
  53. package/dest/index.d.ts +2 -2
  54. package/dest/index.d.ts.map +1 -1
  55. package/dest/index.js +1 -1
  56. package/dest/local-network/banana_fpc.d.ts +10 -0
  57. package/dest/local-network/banana_fpc.d.ts.map +1 -0
  58. package/dest/{sandbox → local-network}/banana_fpc.js +17 -21
  59. package/dest/local-network/index.d.ts +4 -0
  60. package/dest/local-network/index.d.ts.map +1 -0
  61. package/dest/local-network/index.js +3 -0
  62. package/dest/local-network/local-network.d.ts +73 -0
  63. package/dest/local-network/local-network.d.ts.map +1 -0
  64. package/dest/{sandbox/sandbox.js → local-network/local-network.js} +83 -66
  65. package/dest/local-network/sponsored_fpc.d.ts +5 -0
  66. package/dest/local-network/sponsored_fpc.d.ts.map +1 -0
  67. package/dest/{sandbox → local-network}/sponsored_fpc.js +7 -8
  68. package/dest/mnemonic.d.ts +1 -1
  69. package/dest/splash.d.ts +1 -1
  70. package/dest/testing/anvil_test_watcher.d.ts +12 -4
  71. package/dest/testing/anvil_test_watcher.d.ts.map +1 -1
  72. package/dest/testing/anvil_test_watcher.js +69 -31
  73. package/dest/testing/cheat_codes.d.ts +7 -8
  74. package/dest/testing/cheat_codes.d.ts.map +1 -1
  75. package/dest/testing/cheat_codes.js +9 -10
  76. package/dest/testing/epoch_test_settler.d.ts +19 -0
  77. package/dest/testing/epoch_test_settler.d.ts.map +1 -0
  78. package/dest/testing/epoch_test_settler.js +62 -0
  79. package/dest/testing/index.d.ts +2 -2
  80. package/dest/testing/index.d.ts.map +1 -1
  81. package/dest/testing/index.js +1 -1
  82. package/package.json +41 -37
  83. package/scripts/aztec.sh +62 -0
  84. package/scripts/extract_function.js +47 -0
  85. package/scripts/flamegraph.sh +59 -0
  86. package/scripts/init.sh +35 -0
  87. package/scripts/new.sh +59 -0
  88. package/scripts/setup_project.sh +31 -0
  89. package/src/bin/index.ts +15 -9
  90. package/src/cli/admin_api_key_store.ts +128 -0
  91. package/src/cli/aztec_start_action.ts +62 -26
  92. package/src/cli/aztec_start_options.ts +47 -43
  93. package/src/cli/cli.ts +11 -11
  94. package/src/cli/cmds/compile.ts +107 -0
  95. package/src/cli/cmds/migrate_ha_db.ts +43 -0
  96. package/src/cli/cmds/start_archiver.ts +8 -19
  97. package/src/cli/cmds/start_bot.ts +27 -15
  98. package/src/cli/cmds/start_node.ts +64 -23
  99. package/src/cli/cmds/start_p2p_bootstrap.ts +3 -3
  100. package/src/cli/cmds/start_prover_agent.ts +4 -12
  101. package/src/cli/cmds/start_prover_broker.ts +7 -3
  102. package/src/cli/util.ts +23 -26
  103. package/src/cli/versioning.ts +3 -3
  104. package/src/examples/token.ts +20 -19
  105. package/src/examples/util.ts +2 -2
  106. package/src/index.ts +5 -5
  107. package/src/{sandbox → local-network}/banana_fpc.ts +24 -25
  108. package/src/local-network/index.ts +7 -0
  109. package/src/local-network/local-network.ts +267 -0
  110. package/src/local-network/sponsored_fpc.ts +26 -0
  111. package/src/testing/anvil_test_watcher.ts +77 -34
  112. package/src/testing/cheat_codes.ts +13 -13
  113. package/src/testing/epoch_test_settler.ts +71 -0
  114. package/src/testing/index.ts +1 -1
  115. package/dest/cli/cmds/start_blob_sink.d.ts +0 -3
  116. package/dest/cli/cmds/start_blob_sink.d.ts.map +0 -1
  117. package/dest/cli/cmds/start_blob_sink.js +0 -33
  118. package/dest/cli/cmds/start_prover_node.d.ts +0 -7
  119. package/dest/cli/cmds/start_prover_node.d.ts.map +0 -1
  120. package/dest/cli/cmds/start_prover_node.js +0 -108
  121. package/dest/cli/cmds/start_pxe.d.ts +0 -16
  122. package/dest/cli/cmds/start_pxe.d.ts.map +0 -1
  123. package/dest/cli/cmds/start_pxe.js +0 -31
  124. package/dest/sandbox/banana_fpc.d.ts +0 -11
  125. package/dest/sandbox/banana_fpc.d.ts.map +0 -1
  126. package/dest/sandbox/index.d.ts +0 -4
  127. package/dest/sandbox/index.d.ts.map +0 -1
  128. package/dest/sandbox/index.js +0 -3
  129. package/dest/sandbox/sandbox.d.ts +0 -84
  130. package/dest/sandbox/sandbox.d.ts.map +0 -1
  131. package/dest/sandbox/sponsored_fpc.d.ts +0 -4
  132. package/dest/sandbox/sponsored_fpc.d.ts.map +0 -1
  133. package/dest/testing/aztec_cheat_codes.d.ts +0 -59
  134. package/dest/testing/aztec_cheat_codes.d.ts.map +0 -1
  135. package/dest/testing/aztec_cheat_codes.js +0 -62
  136. package/src/cli/cmds/start_blob_sink.ts +0 -57
  137. package/src/cli/cmds/start_prover_node.ts +0 -124
  138. package/src/cli/cmds/start_pxe.ts +0 -49
  139. package/src/sandbox/index.ts +0 -4
  140. package/src/sandbox/sandbox.ts +0 -253
  141. package/src/sandbox/sponsored_fpc.ts +0 -27
  142. package/src/testing/aztec_cheat_codes.ts +0 -77
@@ -1,12 +1,11 @@
1
- import { type InitialAccountData, getInitialTestAccounts } from '@aztec/accounts/testing';
2
- import type { Wallet } from '@aztec/aztec.js';
3
- import { Fr } from '@aztec/foundation/fields';
1
+ import { type InitialAccountData, getInitialTestAccountsData } from '@aztec/accounts/testing';
2
+ import type { Wallet } from '@aztec/aztec.js/wallet';
3
+ import { Fr } from '@aztec/foundation/curves/bn254';
4
4
  import type { LogFn } from '@aztec/foundation/log';
5
5
  import { FPCContract } from '@aztec/noir-contracts.js/FPC';
6
6
  import { TokenContract } from '@aztec/noir-contracts.js/Token';
7
7
  import { AztecAddress } from '@aztec/stdlib/aztec-address';
8
8
  import { type ContractInstanceWithAddress, getContractInstanceFromInstantiationParams } from '@aztec/stdlib/contract';
9
- import type { PXE } from '@aztec/stdlib/interfaces/client';
10
9
 
11
10
  const BANANA_COIN_SALT = new Fr(0);
12
11
  const bananaCoinArgs = {
@@ -46,38 +45,38 @@ export async function getBananaFPCAddress(initialAccounts: InitialAccountData[])
46
45
  return (await getBananaFPCInstance(initialAccounts)).address;
47
46
  }
48
47
 
49
- export async function setupBananaFPC(initialAccounts: InitialAccountData[], deployer: Wallet, log: LogFn) {
48
+ export async function setupBananaFPC(initialAccounts: InitialAccountData[], wallet: Wallet, log: LogFn) {
50
49
  const bananaCoinAddress = await getBananaCoinAddress(initialAccounts);
51
50
  const admin = getBananaAdmin(initialAccounts);
52
51
  const [bananaCoin, fpc] = await Promise.all([
53
- TokenContract.deploy(deployer, admin, bananaCoinArgs.name, bananaCoinArgs.symbol, bananaCoinArgs.decimal)
54
- .send({ from: admin, contractAddressSalt: BANANA_COIN_SALT, universalDeploy: true })
55
- .deployed(),
56
- FPCContract.deploy(deployer, bananaCoinAddress, admin)
57
- .send({ from: admin, contractAddressSalt: BANANA_FPC_SALT, universalDeploy: true })
58
- .deployed(),
52
+ TokenContract.deploy(wallet, admin, bananaCoinArgs.name, bananaCoinArgs.symbol, bananaCoinArgs.decimal).send({
53
+ from: admin,
54
+ contractAddressSalt: BANANA_COIN_SALT,
55
+ universalDeploy: true,
56
+ }),
57
+ FPCContract.deploy(wallet, bananaCoinAddress, admin).send({
58
+ from: admin,
59
+ contractAddressSalt: BANANA_FPC_SALT,
60
+ universalDeploy: true,
61
+ }),
59
62
  ]);
60
63
 
61
64
  log(`BananaCoin: ${bananaCoin.address}`);
62
65
  log(`FPC: ${fpc.address}`);
63
66
  }
64
67
 
65
- export async function getDeployedBananaCoinAddress(pxe: PXE) {
66
- const initialAccounts = await getInitialTestAccounts();
67
- const bananaCoin = await getBananaCoinAddress(initialAccounts);
68
- const contracts = await pxe.getContracts();
69
- if (!contracts.find(c => c.equals(bananaCoin))) {
70
- throw new Error('BananaCoin not deployed.');
71
- }
72
- return bananaCoin;
68
+ export async function registerDeployedBananaCoinInWalletAndGetAddress(wallet: Wallet) {
69
+ const initialAccounts = await getInitialTestAccountsData();
70
+ const bananaCoin = await getBananaCoinInstance(initialAccounts);
71
+ // The following is no-op if the contract is already registered
72
+ await wallet.registerContract(bananaCoin, TokenContract.artifact);
73
+ return bananaCoin.address;
73
74
  }
74
75
 
75
- export async function getDeployedBananaFPCAddress(pxe: PXE) {
76
- const initialAccounts = await getInitialTestAccounts();
76
+ export async function registerDeployedBananaFPCInWalletAndGetAddress(wallet: Wallet) {
77
+ const initialAccounts = await getInitialTestAccountsData();
77
78
  const fpc = await getBananaFPCInstance(initialAccounts);
78
- const contracts = await pxe.getContracts();
79
- if (!contracts.find(c => c.equals(fpc.address))) {
80
- throw new Error('BananaFPC not deployed.');
81
- }
79
+ // The following is no-op if the contract is already registered
80
+ await wallet.registerContract(fpc, FPCContract.artifact);
82
81
  return fpc.address;
83
82
  }
@@ -0,0 +1,7 @@
1
+ export * from './local-network.js';
2
+
3
+ export {
4
+ registerDeployedBananaCoinInWalletAndGetAddress,
5
+ registerDeployedBananaFPCInWalletAndGetAddress,
6
+ } from './banana_fpc.js';
7
+ export { registerDeployedSponsoredFPCInWalletAndGetAddress } from './sponsored_fpc.js';
@@ -0,0 +1,267 @@
1
+ #!/usr/bin/env -S node --no-warnings
2
+ import { getInitialTestAccountsData } from '@aztec/accounts/testing';
3
+ import { AztecNodeService } from '@aztec/aztec-node';
4
+ import { type AztecNodeConfig, getConfigEnvVars } from '@aztec/aztec-node/config';
5
+ import { Fr } from '@aztec/aztec.js/fields';
6
+ import { createLogger } from '@aztec/aztec.js/log';
7
+ import { type BlobClientInterface, createBlobClient } from '@aztec/blob-client/client';
8
+ import { GENESIS_ARCHIVE_ROOT } from '@aztec/constants';
9
+ import { createEthereumChain } from '@aztec/ethereum/chain';
10
+ import { waitForPublicClient } from '@aztec/ethereum/client';
11
+ import { getL1ContractsConfigEnvVars } from '@aztec/ethereum/config';
12
+ import { NULL_KEY } from '@aztec/ethereum/constants';
13
+ import { deployAztecL1Contracts } from '@aztec/ethereum/deploy-aztec-l1-contracts';
14
+ import { EthCheatCodes } from '@aztec/ethereum/test';
15
+ import { SecretValue } from '@aztec/foundation/config';
16
+ import { EthAddress } from '@aztec/foundation/eth-address';
17
+ import type { LogFn } from '@aztec/foundation/log';
18
+ import { DateProvider, TestDateProvider } from '@aztec/foundation/timer';
19
+ import { getVKTreeRoot } from '@aztec/noir-protocol-circuits-types/vk-tree';
20
+ import { protocolContractsHash } from '@aztec/protocol-contracts';
21
+ import { SequencerState } from '@aztec/sequencer-client';
22
+ import type { ProvingJobBroker } from '@aztec/stdlib/interfaces/server';
23
+ import type { PublicDataTreeLeaf } from '@aztec/stdlib/trees';
24
+ import {
25
+ type TelemetryClient,
26
+ getConfigEnvVars as getTelemetryClientConfig,
27
+ initTelemetryClient,
28
+ } from '@aztec/telemetry-client';
29
+ import { EmbeddedWallet } from '@aztec/wallets/embedded';
30
+ import { deployFundedSchnorrAccounts } from '@aztec/wallets/testing';
31
+ import { getGenesisValues } from '@aztec/world-state/testing';
32
+
33
+ import { type Hex, createPublicClient, fallback, http as httpViemTransport } from 'viem';
34
+ import { mnemonicToAccount, privateKeyToAddress } from 'viem/accounts';
35
+ import { foundry } from 'viem/chains';
36
+
37
+ import { createAccountLogs } from '../cli/util.js';
38
+ import { DefaultMnemonic } from '../mnemonic.js';
39
+ import { AnvilTestWatcher } from '../testing/anvil_test_watcher.js';
40
+ import { EpochTestSettler } from '../testing/epoch_test_settler.js';
41
+ import { getBananaFPCAddress, setupBananaFPC } from './banana_fpc.js';
42
+ import { getSponsoredFPCAddress } from './sponsored_fpc.js';
43
+
44
+ const logger = createLogger('local-network');
45
+
46
+ const localAnvil = foundry;
47
+
48
+ /**
49
+ * Function to deploy our L1 contracts to the local network L1
50
+ * @param aztecNodeConfig - The Aztec Node Config
51
+ * @param hdAccount - Account for publishing L1 contracts
52
+ */
53
+ export async function deployContractsToL1(
54
+ aztecNodeConfig: AztecNodeConfig,
55
+ privateKey: Hex,
56
+ opts: {
57
+ genesisArchiveRoot?: Fr;
58
+ feeJuicePortalInitialBalance?: bigint;
59
+ } = {},
60
+ ) {
61
+ await waitForPublicClient(aztecNodeConfig);
62
+
63
+ const l1Contracts = await deployAztecL1Contracts(aztecNodeConfig.l1RpcUrls[0], privateKey, foundry.id, {
64
+ ...getL1ContractsConfigEnvVars(), // TODO: We should not need to be loading config from env again, caller should handle this
65
+ ...aztecNodeConfig,
66
+ vkTreeRoot: getVKTreeRoot(),
67
+ protocolContractsHash,
68
+ genesisArchiveRoot: opts.genesisArchiveRoot ?? new Fr(GENESIS_ARCHIVE_ROOT),
69
+ feeJuicePortalInitialBalance: opts.feeJuicePortalInitialBalance,
70
+ aztecTargetCommitteeSize: 0, // no committee in local network
71
+ slasherFlavor: 'none', // no slashing in local network
72
+ realVerifier: false,
73
+ });
74
+
75
+ aztecNodeConfig.l1Contracts = l1Contracts.l1ContractAddresses;
76
+ aztecNodeConfig.rollupVersion = l1Contracts.rollupVersion;
77
+
78
+ return aztecNodeConfig.l1Contracts;
79
+ }
80
+
81
+ /** Local network settings. */
82
+ export type LocalNetworkConfig = AztecNodeConfig & {
83
+ /** Mnemonic used to derive the L1 deployer private key.*/
84
+ l1Mnemonic: string;
85
+ /** Whether to deploy test accounts on local network start.*/
86
+ testAccounts: boolean;
87
+ };
88
+
89
+ /**
90
+ * Create and start a new Aztec Node and PXE. Deploys L1 contracts.
91
+ * Does not start any HTTP services nor populate any initial accounts.
92
+ * @param config - Optional local network settings.
93
+ */
94
+ export async function createLocalNetwork(config: Partial<LocalNetworkConfig> = {}, userLog: LogFn) {
95
+ // local network is meant for test envs. We should only need one l1RpcUrl
96
+ const l1RpcUrl = config.l1RpcUrls?.[0];
97
+ if (!l1RpcUrl) {
98
+ throw new Error('An L1 RPC URL is required');
99
+ }
100
+ if ((config.l1RpcUrls?.length || 0) > 1) {
101
+ logger.warn(`Multiple L1 RPC URLs provided. Local networks will only use the first one: ${l1RpcUrl}`);
102
+ }
103
+
104
+ const aztecNodeConfig: AztecNodeConfig = {
105
+ ...getConfigEnvVars(),
106
+ ...config,
107
+ };
108
+ const hdAccount = mnemonicToAccount(config.l1Mnemonic || DefaultMnemonic);
109
+ if (
110
+ aztecNodeConfig.sequencerPublisherPrivateKeys == undefined ||
111
+ !aztecNodeConfig.sequencerPublisherPrivateKeys.length ||
112
+ aztecNodeConfig.sequencerPublisherPrivateKeys[0].getValue() === NULL_KEY
113
+ ) {
114
+ const privKey = hdAccount.getHdKey().privateKey;
115
+ aztecNodeConfig.sequencerPublisherPrivateKeys = [
116
+ new SecretValue(`0x${Buffer.from(privKey!).toString('hex')}` as const),
117
+ ];
118
+ }
119
+ if (!aztecNodeConfig.validatorPrivateKeys?.getValue().length) {
120
+ const privKey = hdAccount.getHdKey().privateKey;
121
+ aztecNodeConfig.validatorPrivateKeys = new SecretValue([`0x${Buffer.from(privKey!).toString('hex')}`]);
122
+ }
123
+ aztecNodeConfig.coinbase = EthAddress.fromString(
124
+ privateKeyToAddress(aztecNodeConfig.validatorPrivateKeys.getValue()[0]),
125
+ );
126
+
127
+ const initialAccounts = await (async () => {
128
+ if (config.testAccounts === true || config.testAccounts === undefined) {
129
+ if (aztecNodeConfig.p2pEnabled) {
130
+ userLog(`Not setting up test accounts as we are connecting to a network`);
131
+ } else {
132
+ userLog(`Setting up test accounts`);
133
+ return await getInitialTestAccountsData();
134
+ }
135
+ }
136
+ return [];
137
+ })();
138
+
139
+ const bananaFPC = await getBananaFPCAddress(initialAccounts);
140
+ const sponsoredFPC = await getSponsoredFPCAddress();
141
+ const fundedAddresses = initialAccounts.length
142
+ ? [...initialAccounts.map(a => a.address), bananaFPC, sponsoredFPC]
143
+ : [];
144
+ const { genesisArchiveRoot, prefilledPublicData, fundingNeeded } = await getGenesisValues(fundedAddresses);
145
+
146
+ const dateProvider = new TestDateProvider();
147
+
148
+ let cheatcodes: EthCheatCodes | undefined;
149
+ let rollupAddress: EthAddress | undefined;
150
+ let watcher: AnvilTestWatcher | undefined;
151
+ if (!aztecNodeConfig.p2pEnabled) {
152
+ ({ rollupAddress } = await deployContractsToL1(
153
+ aztecNodeConfig,
154
+ aztecNodeConfig.validatorPrivateKeys.getValue()[0],
155
+ {
156
+ genesisArchiveRoot,
157
+ feeJuicePortalInitialBalance: fundingNeeded,
158
+ },
159
+ ));
160
+
161
+ const chain =
162
+ aztecNodeConfig.l1RpcUrls.length > 0
163
+ ? createEthereumChain([l1RpcUrl], aztecNodeConfig.l1ChainId)
164
+ : { chainInfo: localAnvil };
165
+
166
+ const publicClient = createPublicClient({
167
+ chain: chain.chainInfo,
168
+ transport: fallback([httpViemTransport(l1RpcUrl)]) as any,
169
+ });
170
+
171
+ cheatcodes = new EthCheatCodes([l1RpcUrl], dateProvider);
172
+
173
+ watcher = new AnvilTestWatcher(cheatcodes, rollupAddress, publicClient, dateProvider);
174
+ watcher.setisLocalNetwork(true);
175
+ watcher.setIsMarkingAsProven(false); // Do not mark as proven in the watcher. It's marked in the epochTestSettler after the out hash is set.
176
+
177
+ await watcher.start();
178
+ }
179
+
180
+ const telemetry = await initTelemetryClient(getTelemetryClientConfig());
181
+ // Create a local blob client client inside the local network, no http connectivity
182
+ const blobClient = createBlobClient();
183
+ const node = await createAztecNode(aztecNodeConfig, { telemetry, blobClient, dateProvider }, { prefilledPublicData });
184
+
185
+ // Now that the node is up, let the watcher check for pending txs so it can skip unfilled slots faster when
186
+ // transactions are waiting in the mempool. Also let it check if the sequencer is actively building, to avoid
187
+ // warping time out from under an in-progress block.
188
+ watcher?.setGetPendingTxCount(() => node.getPendingTxCount());
189
+ const sequencer = node.getSequencer()?.getSequencer();
190
+ if (sequencer) {
191
+ const idleStates: Set<string> = new Set([
192
+ SequencerState.STOPPED,
193
+ SequencerState.STOPPING,
194
+ SequencerState.IDLE,
195
+ SequencerState.SYNCHRONIZING,
196
+ ]);
197
+ watcher?.setIsSequencerBuilding(() => !idleStates.has(sequencer.getState()));
198
+ }
199
+
200
+ let epochTestSettler: EpochTestSettler | undefined;
201
+ if (!aztecNodeConfig.p2pEnabled) {
202
+ epochTestSettler = new EpochTestSettler(
203
+ cheatcodes!,
204
+ rollupAddress!,
205
+ node.getBlockSource(),
206
+ logger.createChild('epoch-settler'),
207
+ { pollingIntervalMs: 200 },
208
+ );
209
+ await epochTestSettler.start();
210
+ }
211
+
212
+ if (initialAccounts.length) {
213
+ const wallet = await EmbeddedWallet.create(node, {
214
+ pxeConfig: { proverEnabled: aztecNodeConfig.realProofs },
215
+ ephemeral: true,
216
+ });
217
+
218
+ userLog('Setting up funded test accounts...');
219
+ const accountManagers = await deployFundedSchnorrAccounts(wallet, initialAccounts);
220
+ const accLogs = await createAccountLogs(accountManagers, wallet);
221
+ userLog(accLogs.join(''));
222
+
223
+ await setupBananaFPC(initialAccounts, wallet, userLog);
224
+
225
+ userLog(`SponsoredFPC: ${await getSponsoredFPCAddress()}`);
226
+
227
+ // We no longer need the wallet once we've setup the accounts so we stop the underlying PXE job queue
228
+ await wallet.stop();
229
+ }
230
+
231
+ const stop = async () => {
232
+ await node.stop();
233
+ await watcher?.stop();
234
+ await epochTestSettler?.stop();
235
+ };
236
+
237
+ return { node, stop };
238
+ }
239
+
240
+ /**
241
+ * Create and start a new Aztec RPC HTTP Server
242
+ * @param config - Optional Aztec node settings.
243
+ */
244
+ export async function createAztecNode(
245
+ config: Partial<AztecNodeConfig> = {},
246
+ deps: {
247
+ telemetry?: TelemetryClient;
248
+ blobClient?: BlobClientInterface;
249
+ dateProvider?: DateProvider;
250
+ proverBroker?: ProvingJobBroker;
251
+ } = {},
252
+ options: { prefilledPublicData?: PublicDataTreeLeaf[] } = {},
253
+ ) {
254
+ // TODO(#12272): will clean this up. This is criminal.
255
+ const { l1Contracts, ...rest } = getConfigEnvVars();
256
+ const aztecNodeConfig: AztecNodeConfig = {
257
+ ...rest,
258
+ ...config,
259
+ l1Contracts: { ...l1Contracts, ...config.l1Contracts },
260
+ };
261
+ const node = await AztecNodeService.createAndSync(
262
+ aztecNodeConfig,
263
+ { ...deps, proverNodeDeps: { broker: deps.proverBroker } },
264
+ options,
265
+ );
266
+ return node;
267
+ }
@@ -0,0 +1,26 @@
1
+ import type { AztecAddress } from '@aztec/aztec.js/addresses';
2
+ import {
3
+ type ContractInstanceWithAddress,
4
+ getContractInstanceFromInstantiationParams,
5
+ } from '@aztec/aztec.js/contracts';
6
+ import { Fr } from '@aztec/aztec.js/fields';
7
+ import type { Wallet } from '@aztec/aztec.js/wallet';
8
+ import { SPONSORED_FPC_SALT } from '@aztec/constants';
9
+ import { SponsoredFPCContract } from '@aztec/noir-contracts.js/SponsoredFPC';
10
+
11
+ async function getSponsoredFPCInstance(): Promise<ContractInstanceWithAddress> {
12
+ return await getContractInstanceFromInstantiationParams(SponsoredFPCContract.artifact, {
13
+ salt: new Fr(SPONSORED_FPC_SALT),
14
+ });
15
+ }
16
+
17
+ export async function getSponsoredFPCAddress(): Promise<AztecAddress> {
18
+ return (await getSponsoredFPCInstance()).address;
19
+ }
20
+
21
+ export async function registerDeployedSponsoredFPCInWalletAndGetAddress(wallet: Wallet): Promise<AztecAddress> {
22
+ const fpc = await getSponsoredFPCInstance();
23
+ // The following is no-op if the contract is already registered
24
+ await wallet.registerContract(fpc, SponsoredFPCContract.artifact);
25
+ return fpc.address;
26
+ }
@@ -1,5 +1,6 @@
1
- import type { ViemClient } from '@aztec/ethereum';
2
1
  import { EthCheatCodes, RollupCheatCodes } from '@aztec/ethereum/test';
2
+ import type { ViemClient } from '@aztec/ethereum/types';
3
+ import { SlotNumber } from '@aztec/foundation/branded-types';
3
4
  import type { EthAddress } from '@aztec/foundation/eth-address';
4
5
  import { type Logger, createLogger } from '@aztec/foundation/log';
5
6
  import { RunningPromise } from '@aztec/foundation/running-promise';
@@ -16,11 +17,11 @@ import { type GetContractReturnType, getAddress, getContract } from 'viem';
16
17
  * block within the slot. And if so, it will time travel into the next slot.
17
18
  */
18
19
  export class AnvilTestWatcher {
19
- private isSandbox: boolean = false;
20
+ private isLocalNetwork: boolean = false;
20
21
 
21
22
  private rollup: GetContractReturnType<typeof RollupAbi, ViemClient>;
22
23
  private rollupCheatCodes: RollupCheatCodes;
23
- private l2SlotDuration!: bigint;
24
+ private l2SlotDuration!: number;
24
25
 
25
26
  private filledRunningPromise?: RunningPromise;
26
27
  private syncDateProviderPromise?: RunningPromise;
@@ -30,6 +31,15 @@ export class AnvilTestWatcher {
30
31
 
31
32
  private isMarkingAsProven = true;
32
33
 
34
+ // Optional callback to check if there are pending txs in the mempool.
35
+ private getPendingTxCount?: () => Promise<number>;
36
+
37
+ // Optional callback to check if the sequencer is actively building a block.
38
+ private isSequencerBuilding?: () => boolean;
39
+
40
+ // Tracks when we first observed the current unfilled slot with pending txs (real wall time).
41
+ private unfilledSlotFirstSeen?: { slot: number; realTime: number };
42
+
33
43
  constructor(
34
44
  private cheatcodes: EthCheatCodes,
35
45
  rollupAddress: EthAddress,
@@ -50,11 +60,22 @@ export class AnvilTestWatcher {
50
60
  }
51
61
 
52
62
  setIsMarkingAsProven(isMarkingAsProven: boolean) {
63
+ this.logger.warn(`Watcher is now ${isMarkingAsProven ? 'marking' : 'not marking'} blocks as proven`);
53
64
  this.isMarkingAsProven = isMarkingAsProven;
54
65
  }
55
66
 
56
- setIsSandbox(isSandbox: boolean) {
57
- this.isSandbox = isSandbox;
67
+ setisLocalNetwork(isLocalNetwork: boolean) {
68
+ this.isLocalNetwork = isLocalNetwork;
69
+ }
70
+
71
+ /** Sets a callback to check for pending txs, used to skip unfilled slots faster when txs are waiting. */
72
+ setGetPendingTxCount(fn: () => Promise<number>) {
73
+ this.getPendingTxCount = fn;
74
+ }
75
+
76
+ /** Sets a callback to check if the sequencer is actively building, to avoid warping while it works. */
77
+ setIsSequencerBuilding(fn: () => boolean) {
78
+ this.isSequencerBuilding = fn;
58
79
  }
59
80
 
60
81
  async start() {
@@ -67,7 +88,7 @@ export class AnvilTestWatcher {
67
88
 
68
89
  // If auto mining is not supported (e.g., we are on a real network), then we
69
90
  // will simple do nothing. But if on an anvil or the like, this make sure that
70
- // the sandbox and tests don't break because time is frozen and we never get to
91
+ // the local network and tests don't break because time is frozen and we never get to
71
92
  // the next slot.
72
93
  const isAutoMining = await this.cheatcodes.isAutoMining();
73
94
 
@@ -104,7 +125,7 @@ export class AnvilTestWatcher {
104
125
  }
105
126
 
106
127
  async syncDateProviderToL1IfBehind() {
107
- // this doesn't apply to the sandbox, because we don't have a date provider in the sandbox
128
+ // this doesn't apply to the local network, because we don't have a date provider in the local network
108
129
  if (!this.dateProvider) {
109
130
  return;
110
131
  }
@@ -122,46 +143,68 @@ export class AnvilTestWatcher {
122
143
 
123
144
  async warpTimeIfNeeded() {
124
145
  try {
125
- const currentSlot = await this.rollup.read.getCurrentSlot();
126
- const pendingBlockNumber = BigInt(await this.rollup.read.getPendingBlockNumber());
127
- const blockLog = await this.rollup.read.getBlock([pendingBlockNumber]);
128
- const nextSlotTimestamp = Number(await this.rollup.read.getTimestampForSlot([currentSlot + 1n]));
129
-
130
- if (currentSlot === blockLog.slotNumber) {
131
- // We should jump to the next slot
132
- try {
133
- await this.cheatcodes.warp(nextSlotTimestamp, {
134
- resetBlockInterval: true,
135
- updateDateProvider: this.dateProvider,
136
- });
137
- } catch (e) {
138
- this.logger.error(`Failed to warp to timestamp ${nextSlotTimestamp}: ${e}`);
139
- }
140
-
146
+ const currentSlot = SlotNumber.fromBigInt(await this.rollup.read.getCurrentSlot());
147
+ const pendingCheckpointNumber = await this.rollup.read.getPendingCheckpointNumber();
148
+ const checkpointLog = await this.rollup.read.getCheckpoint([pendingCheckpointNumber]);
149
+ const nextSlot = SlotNumber(currentSlot + 1);
150
+ const nextSlotTimestamp = Number(await this.rollup.read.getTimestampForSlot([BigInt(nextSlot)]));
151
+
152
+ if (BigInt(currentSlot) === checkpointLog.slotNumber) {
153
+ // The current slot has been filled, we should jump to the next slot.
154
+ await this.warpToTimestamp(nextSlotTimestamp);
141
155
  this.logger.info(`Slot ${currentSlot} was filled, jumped to next slot`);
142
156
  return;
143
157
  }
144
158
 
145
- // If we are not in sandbox, we don't need to warp time
146
- if (!this.isSandbox) {
159
+ // If we are not in local network, we don't need to warp time
160
+ if (!this.isLocalNetwork) {
147
161
  return;
148
162
  }
149
163
 
150
- const currentTimestamp = this.dateProvider?.now() ?? Date.now();
151
- if (currentTimestamp > nextSlotTimestamp * 1000) {
152
- try {
153
- await this.cheatcodes.warp(nextSlotTimestamp, {
154
- resetBlockInterval: true,
155
- updateDateProvider: this.dateProvider,
156
- });
157
- } catch (e) {
158
- this.logger.error(`Failed to warp to timestamp ${nextSlotTimestamp}: ${e}`);
164
+ // If there are pending txs and the sequencer missed them, warp quickly (after a 2s real-time debounce) so the
165
+ // sequencer can retry in the next slot. Without this, we'd have to wait a full real-time slot duration (~36s) for
166
+ // the dateProvider to catch up to the next slot timestamp. We skip the warp if the sequencer is actively building
167
+ // to avoid invalidating its in-progress work.
168
+ if (this.getPendingTxCount) {
169
+ const pendingTxs = await this.getPendingTxCount();
170
+ if (pendingTxs > 0) {
171
+ if (this.isSequencerBuilding?.()) {
172
+ this.unfilledSlotFirstSeen = undefined;
173
+ return;
174
+ }
175
+
176
+ const realNow = Date.now();
177
+ if (!this.unfilledSlotFirstSeen || this.unfilledSlotFirstSeen.slot !== currentSlot) {
178
+ this.unfilledSlotFirstSeen = { slot: currentSlot, realTime: realNow };
179
+ return;
180
+ }
181
+
182
+ if (realNow - this.unfilledSlotFirstSeen.realTime > 2000) {
183
+ await this.warpToTimestamp(nextSlotTimestamp);
184
+ this.unfilledSlotFirstSeen = undefined;
185
+ this.logger.info(`Slot ${currentSlot} was missed with pending txs, jumped to next slot`);
186
+ }
187
+
188
+ return;
159
189
  }
190
+ }
160
191
 
192
+ // Fallback: warp when the dateProvider time has passed the next slot timestamp.
193
+ const currentTimestamp = this.dateProvider?.now() ?? Date.now();
194
+ if (currentTimestamp > nextSlotTimestamp * 1000) {
195
+ await this.warpToTimestamp(nextSlotTimestamp);
161
196
  this.logger.info(`Slot ${currentSlot} was missed, jumped to next slot`);
162
197
  }
163
198
  } catch {
164
199
  this.logger.error('mineIfSlotFilled failed');
165
200
  }
166
201
  }
202
+
203
+ private async warpToTimestamp(timestamp: number) {
204
+ try {
205
+ await this.cheatcodes.warp(timestamp, { resetBlockInterval: true });
206
+ } catch (e) {
207
+ this.logger.error(`Failed to warp to timestamp ${timestamp}: ${e}`);
208
+ }
209
+ }
167
210
  }
@@ -1,31 +1,31 @@
1
- import { retryUntil } from '@aztec/aztec.js';
2
1
  import { EthCheatCodes, RollupCheatCodes } from '@aztec/ethereum/test';
2
+ import { BlockNumber } from '@aztec/foundation/branded-types';
3
+ import { retryUntil } from '@aztec/foundation/retry';
4
+ import type { DateProvider } from '@aztec/foundation/timer';
3
5
  import type { SequencerClient } from '@aztec/sequencer-client';
4
- import type { AztecNode, PXE } from '@aztec/stdlib/interfaces/client';
5
-
6
- import { AztecCheatCodes } from './aztec_cheat_codes.js';
6
+ import type { AztecNode } from '@aztec/stdlib/interfaces/client';
7
7
 
8
8
  /**
9
9
  * A class that provides utility functions for interacting with the chain.
10
+ * @deprecated There used to be 3 kinds of cheat codes: eth, rollup and aztec. We have nuked the Aztec ones because
11
+ * they became unused (we now have better testing tools). If you are introducing a new functionality to the cheat
12
+ * codes, please consider whether it makes sense to just introduce new utils in your tests instead.
10
13
  */
11
14
  export class CheatCodes {
12
15
  constructor(
13
16
  /** Cheat codes for L1.*/
14
17
  public eth: EthCheatCodes,
15
- /** Cheat codes for Aztec L2. */
16
- public aztec: AztecCheatCodes,
17
18
  /** Cheat codes for the Aztec Rollup contract on L1. */
18
19
  public rollup: RollupCheatCodes,
19
20
  ) {}
20
21
 
21
- static async create(rpcUrls: string[], pxe: PXE): Promise<CheatCodes> {
22
- const ethCheatCodes = new EthCheatCodes(rpcUrls);
23
- const aztecCheatCodes = new AztecCheatCodes(pxe);
22
+ static async create(rpcUrls: string[], node: AztecNode, dateProvider: DateProvider): Promise<CheatCodes> {
23
+ const ethCheatCodes = new EthCheatCodes(rpcUrls, dateProvider);
24
24
  const rollupCheatCodes = new RollupCheatCodes(
25
25
  ethCheatCodes,
26
- await pxe.getNodeInfo().then(n => n.l1ContractAddresses),
26
+ await node.getNodeInfo().then(n => n.l1ContractAddresses),
27
27
  );
28
- return new CheatCodes(ethCheatCodes, aztecCheatCodes, rollupCheatCodes);
28
+ return new CheatCodes(ethCheatCodes, rollupCheatCodes);
29
29
  }
30
30
 
31
31
  /**
@@ -38,7 +38,7 @@ export class CheatCodes {
38
38
  * @param targetTimestamp - The target timestamp to warp to (in seconds)
39
39
  */
40
40
  async warpL2TimeAtLeastTo(sequencerClient: SequencerClient, node: AztecNode, targetTimestamp: bigint | number) {
41
- const currentL2BlockNumber = await node.getBlockNumber();
41
+ const currentL2BlockNumber: BlockNumber = await node.getBlockNumber();
42
42
 
43
43
  // We warp the L1 timestamp
44
44
  await this.eth.warp(targetTimestamp, { resetBlockInterval: true });
@@ -50,7 +50,7 @@ export class CheatCodes {
50
50
 
51
51
  await retryUntil(
52
52
  async () => {
53
- const newL2BlockNumber = await node.getBlockNumber();
53
+ const newL2BlockNumber: BlockNumber = await node.getBlockNumber();
54
54
  return newL2BlockNumber > currentL2BlockNumber;
55
55
  },
56
56
  'new block after warping L2 time',