@aztec/bot 0.0.1-commit.4ad48494d → 0.0.1-commit.4d3c002

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 (46) hide show
  1. package/dest/amm_bot.d.ts +1 -1
  2. package/dest/amm_bot.d.ts.map +1 -1
  3. package/dest/amm_bot.js +24 -17
  4. package/dest/base_bot.d.ts +3 -3
  5. package/dest/base_bot.d.ts.map +1 -1
  6. package/dest/base_bot.js +12 -22
  7. package/dest/bot.d.ts +1 -1
  8. package/dest/bot.d.ts.map +1 -1
  9. package/dest/bot.js +3 -6
  10. package/dest/config.d.ts +28 -12
  11. package/dest/config.d.ts.map +1 -1
  12. package/dest/config.js +31 -9
  13. package/dest/cross_chain_bot.d.ts +54 -0
  14. package/dest/cross_chain_bot.d.ts.map +1 -0
  15. package/dest/cross_chain_bot.js +134 -0
  16. package/dest/factory.d.ts +20 -1
  17. package/dest/factory.d.ts.map +1 -1
  18. package/dest/factory.js +279 -58
  19. package/dest/index.d.ts +2 -1
  20. package/dest/index.d.ts.map +1 -1
  21. package/dest/index.js +1 -0
  22. package/dest/l1_to_l2_seeding.d.ts +8 -0
  23. package/dest/l1_to_l2_seeding.d.ts.map +1 -0
  24. package/dest/l1_to_l2_seeding.js +63 -0
  25. package/dest/runner.d.ts +1 -1
  26. package/dest/runner.d.ts.map +1 -1
  27. package/dest/runner.js +17 -1
  28. package/dest/store/bot_store.d.ts +30 -5
  29. package/dest/store/bot_store.d.ts.map +1 -1
  30. package/dest/store/bot_store.js +37 -6
  31. package/dest/store/index.d.ts +2 -2
  32. package/dest/store/index.d.ts.map +1 -1
  33. package/dest/utils.js +3 -3
  34. package/package.json +16 -13
  35. package/src/amm_bot.ts +21 -16
  36. package/src/base_bot.ts +8 -16
  37. package/src/bot.ts +3 -5
  38. package/src/config.ts +37 -13
  39. package/src/cross_chain_bot.ts +203 -0
  40. package/src/factory.ts +326 -55
  41. package/src/index.ts +1 -0
  42. package/src/l1_to_l2_seeding.ts +79 -0
  43. package/src/runner.ts +16 -3
  44. package/src/store/bot_store.ts +60 -5
  45. package/src/store/index.ts +1 -1
  46. package/src/utils.ts +3 -3
package/src/factory.ts CHANGED
@@ -1,4 +1,5 @@
1
1
  import { getInitialTestAccountsData } from '@aztec/accounts/testing';
2
+ import { NO_FROM } from '@aztec/aztec.js/account';
2
3
  import { AztecAddress } from '@aztec/aztec.js/addresses';
3
4
  import {
4
5
  BatchCall,
@@ -8,32 +9,41 @@ import {
8
9
  type DeployOptions,
9
10
  NO_WAIT,
10
11
  } from '@aztec/aztec.js/contracts';
11
- import { L1FeeJuicePortalManager } from '@aztec/aztec.js/ethereum';
12
12
  import type { L2AmountClaim } from '@aztec/aztec.js/ethereum';
13
+ import { L1FeeJuicePortalManager } from '@aztec/aztec.js/ethereum';
13
14
  import { FeeJuicePaymentMethodWithClaim } from '@aztec/aztec.js/fee';
14
15
  import { deriveKeys } from '@aztec/aztec.js/keys';
15
16
  import { createLogger } from '@aztec/aztec.js/log';
16
17
  import { waitForL1ToL2MessageReady } from '@aztec/aztec.js/messaging';
17
18
  import { waitForTx } from '@aztec/aztec.js/node';
19
+ import { getFeeJuiceBalance } from '@aztec/aztec.js/utils';
20
+ import { ContractInitializationStatus } from '@aztec/aztec.js/wallet';
18
21
  import { createEthereumChain } from '@aztec/ethereum/chain';
19
22
  import { createExtendedL1Client } from '@aztec/ethereum/client';
23
+ import { RollupContract } from '@aztec/ethereum/contracts';
24
+ import type { ExtendedViemWalletClient } from '@aztec/ethereum/types';
20
25
  import { Fr } from '@aztec/foundation/curves/bn254';
26
+ import { EthAddress } from '@aztec/foundation/eth-address';
21
27
  import { Timer } from '@aztec/foundation/timer';
22
28
  import { AMMContract } from '@aztec/noir-contracts.js/AMM';
23
29
  import { PrivateTokenContract } from '@aztec/noir-contracts.js/PrivateToken';
24
30
  import { TokenContract } from '@aztec/noir-contracts.js/Token';
31
+ import { TestContract } from '@aztec/noir-test-contracts.js/Test';
25
32
  import type { ContractInstanceWithAddress } from '@aztec/stdlib/contract';
26
- import { GasSettings } from '@aztec/stdlib/gas';
33
+ import { GasFees, GasSettings } from '@aztec/stdlib/gas';
27
34
  import type { AztecNode, AztecNodeAdmin } from '@aztec/stdlib/interfaces/client';
28
35
  import { deriveSigningKey } from '@aztec/stdlib/keys';
29
36
  import { EmbeddedWallet } from '@aztec/wallets/embedded';
30
37
 
31
38
  import { type BotConfig, SupportedTokenContracts } from './config.js';
39
+ import { seedL1ToL2Message } from './l1_to_l2_seeding.js';
32
40
  import type { BotStore } from './store/index.js';
33
41
  import { getBalances, getPrivateBalance, isStandardTokenContract } from './utils.js';
34
42
 
35
43
  const MINT_BALANCE = 1e12;
36
44
  const MIN_BALANCE = 1e3;
45
+ const FEE_JUICE_TOP_UP_THRESHOLD = 100n * 10n ** 18n;
46
+ const FEE_JUICE_TOP_UP_TARGET = 10_000n * 10n ** 18n;
37
47
 
38
48
  export class BotFactory {
39
49
  private log = createLogger('bot');
@@ -44,7 +54,11 @@ export class BotFactory {
44
54
  private readonly store: BotStore,
45
55
  private readonly aztecNode: AztecNode,
46
56
  private readonly aztecNodeAdmin?: AztecNodeAdmin,
47
- ) {}
57
+ ) {
58
+ // Set fee padding on the wallet so that all transactions during setup
59
+ // (token deploy, minting, etc.) use the configured padding, not the default.
60
+ this.wallet.setMinFeePadding(config.minFeePadding);
61
+ }
48
62
 
49
63
  /**
50
64
  * Initializes a new bot by setting up the sender account, registering the recipient,
@@ -59,7 +73,8 @@ export class BotFactory {
59
73
  }> {
60
74
  const defaultAccountAddress = await this.setupAccount();
61
75
  const recipient = (await this.wallet.createSchnorrAccount(Fr.random(), Fr.random())).address;
62
- const token = await this.setupToken(defaultAccountAddress);
76
+ const token = await this.setupTokenWithOptionalEarlyRefuel(defaultAccountAddress);
77
+ await this.ensureFeeJuiceBalance(defaultAccountAddress, token);
63
78
  await this.mintTokens(token, defaultAccountAddress);
64
79
  return { wallet: this.wallet, defaultAccountAddress, token, node: this.aztecNode, recipient };
65
80
  }
@@ -73,7 +88,13 @@ export class BotFactory {
73
88
  node: AztecNode;
74
89
  }> {
75
90
  const defaultAccountAddress = await this.setupAccount();
76
- const token0 = await this.setupTokenContract(defaultAccountAddress, this.config.tokenSalt, 'BotToken0', 'BOT0');
91
+ const token0 = await this.setupTokenContractWithOptionalEarlyRefuel(
92
+ defaultAccountAddress,
93
+ this.config.tokenSalt,
94
+ 'BotToken0',
95
+ 'BOT0',
96
+ );
97
+ await this.ensureFeeJuiceBalance(defaultAccountAddress, token0);
77
98
  const token1 = await this.setupTokenContract(defaultAccountAddress, this.config.tokenSalt, 'BotToken1', 'BOT1');
78
99
  const liquidityToken = await this.setupTokenContract(
79
100
  defaultAccountAddress,
@@ -95,6 +116,89 @@ export class BotFactory {
95
116
  return { wallet: this.wallet, defaultAccountAddress, amm, token0, token1, node: this.aztecNode };
96
117
  }
97
118
 
119
+ /**
120
+ * Initializes the cross-chain bot by deploying TestContract, creating an L1 client,
121
+ * seeding initial L1→L2 messages, and waiting for the first to be ready.
122
+ */
123
+ public async setupCrossChain(): Promise<{
124
+ wallet: EmbeddedWallet;
125
+ defaultAccountAddress: AztecAddress;
126
+ contract: TestContract;
127
+ node: AztecNode;
128
+ l1Client: ExtendedViemWalletClient;
129
+ rollupVersion: bigint;
130
+ }> {
131
+ const defaultAccountAddress = await this.setupAccount();
132
+
133
+ // Create L1 client (same pattern as bridgeL1FeeJuice)
134
+ const l1RpcUrls = this.config.l1RpcUrls;
135
+ if (!l1RpcUrls?.length) {
136
+ throw new Error('L1 RPC URLs required for cross-chain bot');
137
+ }
138
+ const mnemonicOrPrivateKey = this.config.l1PrivateKey?.getValue() ?? this.config.l1Mnemonic?.getValue();
139
+ if (!mnemonicOrPrivateKey) {
140
+ throw new Error('L1 mnemonic or private key required for cross-chain bot');
141
+ }
142
+ const { l1ChainId, l1ContractAddresses } = await this.aztecNode.getNodeInfo();
143
+ const chain = createEthereumChain(l1RpcUrls, l1ChainId);
144
+ const l1Client = createExtendedL1Client(chain.rpcUrls, mnemonicOrPrivateKey, chain.chainInfo);
145
+
146
+ // Fetch Rollup version (needed for Inbox L2Actor struct)
147
+ const rollupContract = new RollupContract(l1Client, l1ContractAddresses.rollupAddress.toString());
148
+ const rollupVersion = await rollupContract.getVersion();
149
+
150
+ // Deploy TestContract
151
+ const contract = await this.setupTestContract(defaultAccountAddress);
152
+
153
+ // Recover any pending messages from store (clean up stale ones first)
154
+ await this.store.cleanupOldPendingMessages();
155
+ const pendingMessages = await this.store.getUnconsumedL1ToL2Messages();
156
+
157
+ // Seed initial L1→L2 messages if pipeline is empty
158
+ const seedCount = Math.max(0, this.config.l1ToL2SeedCount - pendingMessages.length);
159
+ for (let i = 0; i < seedCount; i++) {
160
+ await seedL1ToL2Message(
161
+ l1Client,
162
+ EthAddress.fromString(l1ContractAddresses.inboxAddress.toString()),
163
+ contract.address,
164
+ rollupVersion,
165
+ this.store,
166
+ this.log,
167
+ );
168
+ }
169
+
170
+ // Block until at least one message is ready
171
+ const allMessages = await this.store.getUnconsumedL1ToL2Messages();
172
+ if (allMessages.length > 0) {
173
+ this.log.info(`Waiting for first L1→L2 message to be ready...`);
174
+ const firstMsg = allMessages[0];
175
+ await waitForL1ToL2MessageReady(this.aztecNode, Fr.fromHexString(firstMsg.msgHash), {
176
+ timeoutSeconds: this.config.l1ToL2MessageTimeoutSeconds,
177
+ });
178
+ this.log.info(`First L1→L2 message is ready`);
179
+ }
180
+
181
+ return {
182
+ wallet: this.wallet,
183
+ defaultAccountAddress,
184
+ contract,
185
+ node: this.aztecNode,
186
+ l1Client,
187
+ rollupVersion,
188
+ };
189
+ }
190
+
191
+ private async setupTestContract(deployer: AztecAddress): Promise<TestContract> {
192
+ const deployOpts: DeployOptions = {
193
+ from: deployer,
194
+ contractAddressSalt: this.config.tokenSalt,
195
+ universalDeploy: true,
196
+ };
197
+ const deploy = TestContract.deploy(this.wallet);
198
+ const instance = await this.registerOrDeployContract('TestContract', deploy, deployOpts);
199
+ return TestContract.at(instance.address, this.wallet);
200
+ }
201
+
98
202
  /**
99
203
  * Checks if the sender account contract is initialized, and initializes it if necessary.
100
204
  * @returns The sender wallet.
@@ -115,7 +219,7 @@ export class BotFactory {
115
219
  const signingKey = deriveSigningKey(secret);
116
220
  const accountManager = await this.wallet.createSchnorrAccount(secret, salt, signingKey);
117
221
  const metadata = await this.wallet.getContractMetadata(accountManager.address);
118
- if (metadata.isContractInitialized) {
222
+ if (metadata.initializationStatus === ContractInitializationStatus.INITIALIZED) {
119
223
  this.log.info(`Account at ${accountManager.address.toString()} already initialized`);
120
224
  const timer = new Timer();
121
225
  const address = accountManager.address;
@@ -130,13 +234,11 @@ export class BotFactory {
130
234
 
131
235
  const paymentMethod = new FeeJuicePaymentMethodWithClaim(accountManager.address, claim);
132
236
  const deployMethod = await accountManager.getDeployMethod();
133
- const maxFeesPerGas = (await this.aztecNode.getCurrentMinFees()).mul(1 + this.config.minFeePadding);
134
- const gasSettings = GasSettings.default({ maxFeesPerGas });
135
237
 
136
238
  await this.withNoMinTxsPerBlock(async () => {
137
- const txHash = await deployMethod.send({
138
- from: AztecAddress.ZERO,
139
- fee: { gasSettings, paymentMethod },
239
+ const { txHash } = await deployMethod.send({
240
+ from: NO_FROM,
241
+ fee: { paymentMethod },
140
242
  wait: NO_WAIT,
141
243
  });
142
244
  this.log.info(`Sent tx for account deployment with hash ${txHash.toString()}`);
@@ -161,14 +263,79 @@ export class BotFactory {
161
263
  return accountManager.address;
162
264
  }
163
265
 
266
+ /**
267
+ * Setup token and refuel first: if the token already exists (restart scenario),
268
+ * run ensureFeeJuiceBalance before any step that might need fee juice. When deploying,
269
+ * use a bridge claim if balance is below threshold.
270
+ */
271
+ private async setupTokenWithOptionalEarlyRefuel(sender: AztecAddress): Promise<TokenContract | PrivateTokenContract> {
272
+ const token = await this.getTokenInstance(sender);
273
+ const address = token.address;
274
+ const metadata = await this.wallet.getContractMetadata(address);
275
+ if (metadata.isContractPublished) {
276
+ this.log.info(`Token at ${address.toString()} already deployed, refueling before setup`);
277
+ await this.ensureFeeJuiceBalance(sender, token);
278
+ }
279
+ return this.setupToken(sender);
280
+ }
281
+
282
+ /**
283
+ * Setup token0 for AMM with refuel-first behaviour when token already exists.
284
+ */
285
+ private async setupTokenContractWithOptionalEarlyRefuel(
286
+ deployer: AztecAddress,
287
+ contractAddressSalt: Fr,
288
+ name: string,
289
+ ticker: string,
290
+ decimals = 18,
291
+ ): Promise<TokenContract> {
292
+ const deployOpts: DeployOptions = { from: deployer, contractAddressSalt, universalDeploy: true };
293
+ const deploy = TokenContract.deploy(this.wallet, deployer, name, ticker, decimals);
294
+ const instance = await deploy.getInstance(deployOpts);
295
+ const metadata = await this.wallet.getContractMetadata(instance.address);
296
+ if (metadata.isContractPublished) {
297
+ this.log.info(`Token ${name} at ${instance.address.toString()} already deployed, refueling before setup`);
298
+ const token = TokenContract.at(instance.address, this.wallet);
299
+ await this.ensureFeeJuiceBalance(deployer, token);
300
+ }
301
+ return this.setupTokenContract(deployer, contractAddressSalt, name, ticker, decimals);
302
+ }
303
+
304
+ private async getTokenInstance(sender: AztecAddress): Promise<TokenContract | PrivateTokenContract> {
305
+ const deployOpts: DeployOptions = {
306
+ from: sender,
307
+ contractAddressSalt: this.config.tokenSalt,
308
+ universalDeploy: true,
309
+ };
310
+ if (this.config.contract === SupportedTokenContracts.TokenContract) {
311
+ const deploy = TokenContract.deploy(this.wallet, sender, 'BotToken', 'BOT', 18);
312
+ const instance = await deploy.getInstance(deployOpts);
313
+ return TokenContract.at(instance.address, this.wallet);
314
+ }
315
+ if (this.config.contract === SupportedTokenContracts.PrivateTokenContract) {
316
+ const tokenSecretKey = Fr.random();
317
+ const tokenPublicKeys = (await deriveKeys(tokenSecretKey)).publicKeys;
318
+ const deploy = PrivateTokenContract.deployWithPublicKeys(tokenPublicKeys, this.wallet, MINT_BALANCE, sender);
319
+ const instance = await deploy.getInstance({
320
+ ...deployOpts,
321
+ skipInstancePublication: true,
322
+ skipClassPublication: true,
323
+ skipInitialization: false,
324
+ });
325
+ return PrivateTokenContract.at(instance.address, this.wallet);
326
+ }
327
+ throw new Error(`Unsupported token contract type: ${this.config.contract}`);
328
+ }
329
+
164
330
  /**
165
331
  * Checks if the token contract is deployed and deploys it if necessary.
166
- * @param wallet - Wallet to deploy the token contract from.
167
- * @returns The TokenContract instance.
332
+ * Uses a bridge claim for deploy when balance is below threshold to avoid failing before refuel.
333
+ * @param sender - Aztec address to deploy the token contract from.
334
+ * @param existingToken - Optional token instance when called from setupTokenWithOptionalEarlyRefuel.
335
+ * @returns The TokenContract or PrivateTokenContract instance.
168
336
  */
169
337
  private async setupToken(sender: AztecAddress): Promise<TokenContract | PrivateTokenContract> {
170
338
  let deploy: DeployMethod<TokenContract | PrivateTokenContract>;
171
- let tokenInstance: ContractInstanceWithAddress | undefined;
172
339
  const deployOpts: DeployOptions = {
173
340
  from: sender,
174
341
  contractAddressSalt: this.config.tokenSalt,
@@ -177,8 +344,8 @@ export class BotFactory {
177
344
  let token: TokenContract | PrivateTokenContract;
178
345
  if (this.config.contract === SupportedTokenContracts.TokenContract) {
179
346
  deploy = TokenContract.deploy(this.wallet, sender, 'BotToken', 'BOT', 18);
180
- tokenInstance = await deploy.getInstance(deployOpts);
181
- token = TokenContract.at(tokenInstance.address, this.wallet);
347
+ const instance = await deploy.getInstance(deployOpts);
348
+ token = TokenContract.at(instance.address, this.wallet);
182
349
  } else if (this.config.contract === SupportedTokenContracts.PrivateTokenContract) {
183
350
  // Generate keys for the contract since PrivateToken uses SinglePrivateMutable which requires keys
184
351
  const tokenSecretKey = Fr.random();
@@ -189,27 +356,16 @@ export class BotFactory {
189
356
  deployOpts.skipInitialization = false;
190
357
 
191
358
  // Register the contract with the secret key before deployment
192
- tokenInstance = await deploy.getInstance(deployOpts);
359
+ const tokenInstance = await deploy.getInstance(deployOpts);
193
360
  token = PrivateTokenContract.at(tokenInstance.address, this.wallet);
194
361
  await this.wallet.registerContract(tokenInstance, PrivateTokenContract.artifact, tokenSecretKey);
362
+ // The contract constructor initializes private storage vars that need the contract's own nullifier key.
363
+ deployOpts.additionalScopes = [tokenInstance.address];
195
364
  } else {
196
365
  throw new Error(`Unsupported token contract type: ${this.config.contract}`);
197
366
  }
198
367
 
199
- const address = tokenInstance?.address ?? (await deploy.getInstance(deployOpts)).address;
200
- const metadata = await this.wallet.getContractMetadata(address);
201
- if (metadata.isContractPublished) {
202
- this.log.info(`Token at ${address.toString()} already deployed`);
203
- await deploy.register();
204
- } else {
205
- this.log.info(`Deploying token contract at ${address.toString()}`);
206
- const txHash = await deploy.send({ ...deployOpts, wait: NO_WAIT });
207
- this.log.info(`Sent tx for token setup with hash ${txHash.toString()}`);
208
- await this.withNoMinTxsPerBlock(async () => {
209
- await waitForTx(this.aztecNode, txHash, { timeout: this.config.txMinedWaitSeconds });
210
- return token;
211
- });
212
- }
368
+ await this.registerOrDeployContract('token', deploy, deployOpts);
213
369
  return token;
214
370
  }
215
371
 
@@ -244,9 +400,11 @@ export class BotFactory {
244
400
  const amm = AMMContract.at(instance.address, this.wallet);
245
401
 
246
402
  this.log.info(`AMM deployed at ${amm.address}`);
247
- const minterReceipt = await lpToken.methods
248
- .set_minter(amm.address, true)
249
- .send({ from: deployer, wait: { timeout: this.config.txMinedWaitSeconds } });
403
+ const setMinterInteraction = lpToken.methods.set_minter(amm.address, true);
404
+ const { receipt: minterReceipt } = await setMinterInteraction.send({
405
+ from: deployer,
406
+ wait: { timeout: this.config.txMinedWaitSeconds },
407
+ });
250
408
  this.log.info(`Set LP token minter to AMM txHash=${minterReceipt.txHash.toString()}`);
251
409
  this.log.info(`Liquidity token initialized`);
252
410
 
@@ -263,9 +421,18 @@ export class BotFactory {
263
421
  ): Promise<void> {
264
422
  const getPrivateBalances = () =>
265
423
  Promise.all([
266
- token0.methods.balance_of_private(liquidityProvider).simulate({ from: liquidityProvider }),
267
- token1.methods.balance_of_private(liquidityProvider).simulate({ from: liquidityProvider }),
268
- lpToken.methods.balance_of_private(liquidityProvider).simulate({ from: liquidityProvider }),
424
+ token0.methods
425
+ .balance_of_private(liquidityProvider)
426
+ .simulate({ from: liquidityProvider })
427
+ .then(r => r.result),
428
+ token1.methods
429
+ .balance_of_private(liquidityProvider)
430
+ .simulate({ from: liquidityProvider })
431
+ .then(r => r.result),
432
+ lpToken.methods
433
+ .balance_of_private(liquidityProvider)
434
+ .simulate({ from: liquidityProvider })
435
+ .then(r => r.result),
269
436
  ]);
270
437
 
271
438
  const authwitNonce = Fr.random();
@@ -306,20 +473,29 @@ export class BotFactory {
306
473
  .getFunctionCall(),
307
474
  });
308
475
 
309
- const mintReceipt = await new BatchCall(this.wallet, [
476
+ const mintBatch = new BatchCall(this.wallet, [
310
477
  token0.methods.mint_to_private(liquidityProvider, MINT_BALANCE),
311
478
  token1.methods.mint_to_private(liquidityProvider, MINT_BALANCE),
312
- ]).send({ from: liquidityProvider, wait: { timeout: this.config.txMinedWaitSeconds } });
479
+ ]);
480
+ const { receipt: mintReceipt } = await mintBatch.send({
481
+ from: liquidityProvider,
482
+ wait: { timeout: this.config.txMinedWaitSeconds },
483
+ });
313
484
 
314
485
  this.log.info(`Sent mint tx: ${mintReceipt.txHash.toString()}`);
315
486
 
316
- const addLiquidityReceipt = await amm.methods
317
- .add_liquidity(amount0Max, amount1Max, amount0Min, amount1Min, authwitNonce)
318
- .send({
319
- from: liquidityProvider,
320
- authWitnesses: [token0Authwit, token1Authwit],
321
- wait: { timeout: this.config.txMinedWaitSeconds },
322
- });
487
+ const addLiquidityInteraction = amm.methods.add_liquidity(
488
+ amount0Max,
489
+ amount1Max,
490
+ amount0Min,
491
+ amount1Min,
492
+ authwitNonce,
493
+ );
494
+ const { receipt: addLiquidityReceipt } = await addLiquidityInteraction.send({
495
+ from: liquidityProvider,
496
+ authWitnesses: [token0Authwit, token1Authwit],
497
+ wait: { timeout: this.config.txMinedWaitSeconds },
498
+ });
323
499
 
324
500
  this.log.info(`Sent tx to add liquidity to the AMM: ${addLiquidityReceipt.txHash.toString()}`);
325
501
  this.log.info(`Liquidity added`);
@@ -342,12 +518,42 @@ export class BotFactory {
342
518
  this.log.info(`Contract ${name} at ${address.toString()} already deployed`);
343
519
  await deploy.register();
344
520
  } else {
345
- this.log.info(`Deploying contract ${name} at ${address.toString()}`);
346
- await this.withNoMinTxsPerBlock(async () => {
347
- const txHash = await deploy.send({ ...deployOpts, wait: NO_WAIT });
348
- this.log.info(`Sent contract ${name} setup tx with hash ${txHash.toString()}`);
349
- return waitForTx(this.aztecNode, txHash, { timeout: this.config.txMinedWaitSeconds });
350
- });
521
+ const sender = deployOpts.from === NO_FROM ? undefined : deployOpts.from;
522
+ const balance = sender ? await getFeeJuiceBalance(sender, this.aztecNode) : 0n;
523
+ const useClaim =
524
+ sender &&
525
+ balance < FEE_JUICE_TOP_UP_THRESHOLD &&
526
+ this.config.feePaymentMethod === 'fee_juice' &&
527
+ !!this.config.l1RpcUrls?.length;
528
+ const mnemonicOrPrivateKey = this.config.l1PrivateKey?.getValue() ?? this.config.l1Mnemonic?.getValue();
529
+
530
+ if (useClaim && mnemonicOrPrivateKey) {
531
+ const claim = await this.getOrCreateBridgeClaim(sender!);
532
+ const paymentMethod = new FeeJuicePaymentMethodWithClaim(sender!, claim);
533
+ const { estimatedGas } = await deploy.simulate({ ...deployOpts, fee: { estimateGas: true, paymentMethod } });
534
+ const maxFeesPerGas = (await this.aztecNode.getCurrentMinFees()).mul(1 + this.config.minFeePadding);
535
+ const gasSettings = GasSettings.from({
536
+ ...estimatedGas!,
537
+ maxFeesPerGas,
538
+ maxPriorityFeesPerGas: GasFees.empty(),
539
+ });
540
+ await this.withNoMinTxsPerBlock(async () => {
541
+ const { txHash } = await deploy.send({ ...deployOpts, fee: { gasSettings, paymentMethod }, wait: NO_WAIT });
542
+ this.log.info(
543
+ `Sent contract ${name} deploy tx ${txHash.toString()} (using bridge claim, balance was ${balance})`,
544
+ );
545
+ return waitForTx(this.aztecNode, txHash, { timeout: this.config.txMinedWaitSeconds });
546
+ });
547
+ await this.store.deleteBridgeClaim(sender!);
548
+ } else {
549
+ const { estimatedGas } = await deploy.simulate({ ...deployOpts, fee: { estimateGas: true } });
550
+ this.log.info(`Deploying contract ${name} at ${address.toString()}`, { estimatedGas });
551
+ await this.withNoMinTxsPerBlock(async () => {
552
+ const { txHash } = await deploy.send({ ...deployOpts, fee: { gasSettings: estimatedGas }, wait: NO_WAIT });
553
+ this.log.info(`Sent contract ${name} setup tx with hash ${txHash.toString()}`);
554
+ return waitForTx(this.aztecNode, txHash, { timeout: this.config.txMinedWaitSeconds });
555
+ });
556
+ }
351
557
  }
352
558
  return instance;
353
559
  }
@@ -356,6 +562,66 @@ export class BotFactory {
356
562
  * Mints private and public tokens for the sender if their balance is below the minimum.
357
563
  * @param token - Token contract.
358
564
  */
565
+ /**
566
+ * Ensures the account has sufficient fee juice by bridging from L1 if balance is below threshold.
567
+ * Bridges repeatedly until balance reaches the target (10k FJ).
568
+ * Used on startup/restart to top up when the account has run out after previous runs.
569
+ */
570
+ private async ensureFeeJuiceBalance(
571
+ account: AztecAddress,
572
+ token: TokenContract | PrivateTokenContract,
573
+ ): Promise<void> {
574
+ const { feePaymentMethod, l1RpcUrls } = this.config;
575
+ if (feePaymentMethod !== 'fee_juice' || !l1RpcUrls?.length) {
576
+ return;
577
+ }
578
+ const mnemonicOrPrivateKey = this.config.l1PrivateKey?.getValue() ?? this.config.l1Mnemonic?.getValue();
579
+ if (!mnemonicOrPrivateKey) {
580
+ return;
581
+ }
582
+
583
+ let balance = await getFeeJuiceBalance(account, this.aztecNode);
584
+ if (balance >= FEE_JUICE_TOP_UP_THRESHOLD) {
585
+ this.log.info(`Fee juice balance ${balance} above threshold ${FEE_JUICE_TOP_UP_THRESHOLD}, skipping top-up`);
586
+ return;
587
+ }
588
+
589
+ this.log.info(
590
+ `Fee juice balance ${balance} below threshold ${FEE_JUICE_TOP_UP_THRESHOLD}, bridging from L1 until ${FEE_JUICE_TOP_UP_TARGET}`,
591
+ );
592
+ const maxFeesPerGas = (await this.aztecNode.getCurrentMinFees()).mul(1 + this.config.minFeePadding);
593
+ const minimalInteraction = isStandardTokenContract(token)
594
+ ? token.methods.transfer_in_public(account, account, 0n, 0)
595
+ : token.methods.transfer(0n, account, account);
596
+
597
+ while (balance < FEE_JUICE_TOP_UP_TARGET) {
598
+ const claim = await this.bridgeL1FeeJuice(account);
599
+ const paymentMethod = new FeeJuicePaymentMethodWithClaim(account, claim);
600
+ const { estimatedGas } = await minimalInteraction.simulate({
601
+ from: account,
602
+ fee: { estimateGas: true, paymentMethod },
603
+ });
604
+ const gasSettings = GasSettings.from({
605
+ ...estimatedGas!,
606
+ maxFeesPerGas,
607
+ maxPriorityFeesPerGas: GasFees.empty(),
608
+ });
609
+
610
+ await this.withNoMinTxsPerBlock(async () => {
611
+ const { txHash } = await minimalInteraction.send({
612
+ from: account,
613
+ fee: { gasSettings, paymentMethod },
614
+ wait: NO_WAIT,
615
+ });
616
+ this.log.info(`Sent fee juice top-up tx ${txHash.toString()}`);
617
+ return waitForTx(this.aztecNode, txHash, { timeout: this.config.txMinedWaitSeconds });
618
+ });
619
+ balance = await getFeeJuiceBalance(account, this.aztecNode);
620
+ this.log.info(`Fee juice balance after top-up: ${balance}`);
621
+ }
622
+ this.log.info(`Fee juice top-up complete for ${account.toString()}`);
623
+ }
624
+
359
625
  private async mintTokens(token: TokenContract | PrivateTokenContract, minter: AztecAddress) {
360
626
  const isStandardToken = isStandardTokenContract(token);
361
627
  let privateBalance = 0n;
@@ -386,8 +652,15 @@ export class BotFactory {
386
652
  return;
387
653
  }
388
654
 
655
+ // PrivateToken's mint accesses contract-level private storage vars (admin, total_supply).
656
+ const additionalScopes = isStandardToken ? undefined : [token.address];
657
+ const mintBatch = new BatchCall(token.wallet, calls);
389
658
  await this.withNoMinTxsPerBlock(async () => {
390
- const txHash = await new BatchCall(token.wallet, calls).send({ from: minter, wait: NO_WAIT });
659
+ const { txHash } = await mintBatch.send({
660
+ from: minter,
661
+ additionalScopes,
662
+ wait: NO_WAIT,
663
+ });
391
664
  this.log.info(`Sent token mint tx with hash ${txHash.toString()}`);
392
665
  return waitForTx(this.aztecNode, txHash, { timeout: this.config.txMinedWaitSeconds });
393
666
  });
@@ -410,7 +683,6 @@ export class BotFactory {
410
683
  await this.withNoMinTxsPerBlock(() =>
411
684
  waitForL1ToL2MessageReady(this.aztecNode, messageHash, {
412
685
  timeoutSeconds: this.config.l1ToL2MessageTimeoutSeconds,
413
- forPublicConsumption: false,
414
686
  }),
415
687
  );
416
688
  return existingClaim.claim;
@@ -449,7 +721,6 @@ export class BotFactory {
449
721
  await this.withNoMinTxsPerBlock(() =>
450
722
  waitForL1ToL2MessageReady(this.aztecNode, Fr.fromHexString(claim.messageHash), {
451
723
  timeoutSeconds: this.config.l1ToL2MessageTimeoutSeconds,
452
- forPublicConsumption: false,
453
724
  }),
454
725
  );
455
726
 
package/src/index.ts CHANGED
@@ -1,5 +1,6 @@
1
1
  export { Bot } from './bot.js';
2
2
  export { AmmBot } from './amm_bot.js';
3
+ export { CrossChainBot } from './cross_chain_bot.js';
3
4
  export { BotRunner } from './runner.js';
4
5
  export { BotStore } from './store/bot_store.js';
5
6
  export {
@@ -0,0 +1,79 @@
1
+ import { generateClaimSecret } from '@aztec/aztec.js/ethereum';
2
+ import type { ExtendedViemWalletClient } from '@aztec/ethereum/types';
3
+ import { compactArray } from '@aztec/foundation/collection';
4
+ import { Fr } from '@aztec/foundation/curves/bn254';
5
+ import { EthAddress } from '@aztec/foundation/eth-address';
6
+ import type { Logger } from '@aztec/foundation/log';
7
+ import { InboxAbi } from '@aztec/l1-artifacts';
8
+ import type { AztecAddress } from '@aztec/stdlib/aztec-address';
9
+
10
+ import { decodeEventLog, getContract } from 'viem';
11
+
12
+ import type { BotStore, PendingL1ToL2Message } from './store/index.js';
13
+
14
+ /** Sends an L1→L2 message via the Inbox contract and stores it. */
15
+ export async function seedL1ToL2Message(
16
+ l1Client: ExtendedViemWalletClient,
17
+ inboxAddress: EthAddress,
18
+ l2Recipient: AztecAddress,
19
+ rollupVersion: bigint,
20
+ store: BotStore,
21
+ log: Logger,
22
+ ): Promise<PendingL1ToL2Message> {
23
+ log.info('Seeding L1→L2 message');
24
+ const [secret, secretHash] = await generateClaimSecret(log);
25
+ const content = Fr.random();
26
+
27
+ const inbox = getContract({
28
+ address: inboxAddress.toString(),
29
+ abi: InboxAbi,
30
+ client: l1Client,
31
+ });
32
+
33
+ const txHash = await inbox.write.sendL2Message(
34
+ [{ actor: l2Recipient.toString(), version: rollupVersion }, content.toString(), secretHash.toString()],
35
+ { gas: 1_000_000n },
36
+ );
37
+ log.info(`L1→L2 message sent in tx ${txHash}`);
38
+
39
+ const txReceipt = await l1Client.waitForTransactionReceipt({ hash: txHash });
40
+ if (txReceipt.status !== 'success') {
41
+ throw new Error(`L1→L2 message tx failed: ${txHash}`);
42
+ }
43
+
44
+ // Extract MessageSent event
45
+ const messageSentLogs = compactArray(
46
+ txReceipt.logs
47
+ .filter(l => l.address.toLowerCase() === inboxAddress.toString().toLowerCase())
48
+ .map(l => {
49
+ try {
50
+ return decodeEventLog({ abi: InboxAbi, eventName: 'MessageSent', data: l.data, topics: l.topics });
51
+ } catch {
52
+ return undefined;
53
+ }
54
+ }),
55
+ );
56
+
57
+ if (messageSentLogs.length !== 1) {
58
+ throw new Error(`Expected 1 MessageSent event, got ${messageSentLogs.length}`);
59
+ }
60
+
61
+ const event = messageSentLogs[0];
62
+
63
+ const msgHash = event.args.hash;
64
+ const globalLeafIndex = event.args.index;
65
+
66
+ const msg: PendingL1ToL2Message = {
67
+ content: content.toString(),
68
+ secret: secret.toString(),
69
+ secretHash: secretHash.toString(),
70
+ msgHash,
71
+ sender: l1Client.account!.address,
72
+ globalLeafIndex: globalLeafIndex.toString(),
73
+ timestamp: Date.now(),
74
+ };
75
+
76
+ await store.savePendingL1ToL2Message(msg);
77
+ log.info(`Seeded L1→L2 message msgHash=${msg.msgHash}`);
78
+ return msg;
79
+ }
package/src/runner.ts CHANGED
@@ -10,6 +10,7 @@ import { AmmBot } from './amm_bot.js';
10
10
  import type { BaseBot } from './base_bot.js';
11
11
  import { Bot } from './bot.js';
12
12
  import type { BotConfig } from './config.js';
13
+ import { CrossChainBot } from './cross_chain_bot.js';
13
14
  import type { BotInfo, BotRunnerApi } from './interface.js';
14
15
  import { BotStore } from './store/index.js';
15
16
 
@@ -146,9 +147,21 @@ export class BotRunner implements BotRunnerApi, Traceable {
146
147
 
147
148
  async #createBot() {
148
149
  try {
149
- this.bot = this.config.ammTxs
150
- ? AmmBot.create(this.config, this.wallet, this.aztecNode, this.aztecNodeAdmin, this.store)
151
- : Bot.create(this.config, this.wallet, this.aztecNode, this.aztecNodeAdmin, this.store);
150
+ switch (this.config.botMode) {
151
+ case 'crosschain':
152
+ this.bot = CrossChainBot.create(this.config, this.wallet, this.aztecNode, this.aztecNodeAdmin, this.store);
153
+ break;
154
+ case 'amm':
155
+ this.bot = AmmBot.create(this.config, this.wallet, this.aztecNode, this.aztecNodeAdmin, this.store);
156
+ break;
157
+ case 'transfer':
158
+ this.bot = Bot.create(this.config, this.wallet, this.aztecNode, this.aztecNodeAdmin, this.store);
159
+ break;
160
+ default: {
161
+ const _exhaustive: never = this.config.botMode;
162
+ throw new Error(`Unsupported bot mode: [${_exhaustive}]`);
163
+ }
164
+ }
152
165
  await this.bot;
153
166
  } catch (err) {
154
167
  this.log.error(`Error setting up bot: ${err}`);