Skip to main content
Version: v0.83.1-alpha-testnet.0

How to Pay Fees

There a several ways to pay for transaction fees, thanks to fee abstraction implemented in the Aztec protocol.

This guide shows the various ways this can be done. For more of a background see Fees on Aztec.

Methods to pay a tx fee

  1. Natively via an existing balance of fee juice

Either your own account or someone else's address has a balance of fee juice, and the transaction is paid for from this balance.

  1. Natively via a balance of fee juice bridged from L1 to be claimed on L2

Processing this transaction first claims bridged fee juice then is paid for from the balance, all in the same public/private tx.

  1. Privately or publicly via a Fee Paying Contract

A fee paying contract (FPC) is created and nominates a token that it accepts to then pay for transactions in fee juice. So a user doesn't need to hold fee juice, they only need the token that the FPC accepts.

The FPC can also decide to refund users in cases of overpayment. For private FPCs, this incurrs a substantial gate count (proving time) and DA costs so in some cases it might be cheaper to not get the refund. The sample FPCs we show here refund the users but a FPC deployer or user should think carefully if asking for refunds is a smart option.

The most straightforward way to pay for a transaction is via the sponsored fee payment method, bootstrapping some transactions by skipping the need to bridge fee juice to the account. This method uses a type of fee paying contract configured to pay for a number of transactions without requiring payment, but also requires that there is a sponsor to pay for the transactions. This only exists on the sandbox as a convenience to get started quickly, though you could deploy it on testnet too for your own use case

The most straightforward way to pay for a transaction is via the sponsored fee payment method, bootstrapping some transactions by skipping the need to bridge fee juice to the account. This method uses a type of fee paying contract configured to pay for a number of transactions without requiring payment, but also requires that there is a sponsor to pay for the transactions.

Bridging Fee Juice

Sandbox

To first get fee juice into an account it needs to be bridged from L1. You can skip this section if you want to first make a transaction via the SponsoredFPC.

note

See here to Bridge Fee Juice via the CLI wallet.

One way of bridging of tokens is described fully here. Below summarises specifically bridging fee juice on the sandbox.

First get the node info and create a public client pointing to the sandbox's anvil L1 node (from foundry):

get_node_info_pub_client
const info = await pxe.getNodeInfo();
const publicClient = getPublicClient({
l1RpcUrls: ['http://localhost:8545'],
l1ChainId: foundry.id,
});
Source code: yarn-project/end-to-end/src/spartan/smoke.test.ts#L52-L58

After importing:

import { L1FeeJuicePortalManager } from "@aztec/aztec.js";

Create a new fee juice portal manager and bridge fee juice publicly to Aztec:

bridge_fee_juice
const portal = await L1FeeJuicePortalManager.new(pxe, publicClient, walletClient, log);
const claim = await portal.bridgeTokensPublic(recipient, amount, true /* mint */);
Source code: yarn-project/end-to-end/src/spartan/setup_test_wallets.ts#L109-L112

For the mechanisms to complete bridging between L1 and Aztec, we have to wait for 2 Aztec blocks to progress. This can be triggered manually by sending 2 transactions in the sandbox, or by waiting for 2 blocks on a public network. After this, an account should have its fee juice ready to use in transactions.

Alternatively, the resulting claim object can be used to create a payment method to claim and pay for a transaction in one, where the transaction is the contract's deployment.

Testnet

The L1FeeJuicePortalManager will not be able to mint assets for you on testnet. You will need to call the fee asset's mint function on L1 to mint fee juice, e.g.:

cast call $FEE_ASSET_HANDLER_CONTRACT "mint(address)" $MY_L1_ADDRESS --rpc-url <RPC_URL>

Then bridge it to L2, using the the L1FeeJuicePortalManager as described above.

Examples

Pay with FeeJuice

To send a transaction from an already deployed account already holding fee juice. A user's fee juice balance is public, so paying for a transaction with fee juice will reveal information about which account is paying for the transaction and how much they paid.

Note: this example is a public token transfer call, but can equally be a private function call

import { FeeJuicePaymentMethod } from "@aztec/aztec.js";
pay_fee_juice_send
const paymentMethod = new FeeJuicePaymentMethod(aliceAddress);
const { transactionFee } = await bananaCoin.methods
.transfer_in_public(aliceAddress, bobAddress, 1n, 0n)
.send({ fee: { gasSettings, paymentMethod } })
.wait();
Source code: yarn-project/end-to-end/src/e2e_fees/fee_juice_payments.test.ts#L86-L92

The equivalent to specify fees via CLI...

The CLI tool aztec-wallet takes the fee payment method via the param: --payment method=fee_juice. See help for sending txs, eg aztec-wallet help deploy

Claim and deploy

Here we will use the claim object previously from the bridging section, and the corresponding wallet, to create the fee payment method. The payment method is then used to claim fee juice and pay for account contract deployment. Note the function used to bridge fee juice (private/public) should correspond to how the fee juice is claimed.

import { FeeJuicePaymentMethodWithClaim } from "@aztec/aztec.js";
claim_and_deploy
const wallet = await account.getWallet();
const paymentMethod = new FeeJuicePaymentMethodWithClaim(wallet, claim);
const sentTx = account.deploy({ fee: { paymentMethod } });
const txHash = await sentTx.getTxHash();
Source code: yarn-project/bot/src/factory.ts#L121-L126

The equivalent to specify fees via CLI...

The CLI tool aztec-wallet takes the fee payment method via the param: --payment method=fee_juice,claim

Claim and Pay

Claiming bridged fee juice and using it to pay for transaction fees results in fees being paid publicly.

Calling a function, in this case checking the balance of the fee juice contract:

claim_and_pay
const paymentMethod = new FeeJuicePaymentMethodWithClaim(bobWallet, claim);
const receipt = await feeJuiceContract
.withWallet(bobWallet)
.methods.check_balance(0n)
.send({ fee: { gasSettings, paymentMethod } })
.wait();
Source code: yarn-project/end-to-end/src/e2e_fees/fee_juice_payments.test.ts#L67-L74

Fee Paying Contract

Users can pay fees privately or publicly via FPCs. A FPC must be deployed and funded with fee juice before it can be used to pay for transactions. Users must have the token that the FPC accepts to pay for transactions. They must have a private balance of the token if they want to pay for fees privately, and a public balance if they want to pay for fees publicly.

Import the appropriate method from aztec.js.

import {
PrivateFeePaymentMethod,
PublicFeePaymentMethod,
} from "@aztec/aztec.js";

The FPC contract must be registered in a users PXE before it can be used.

import { FPCContract } from "@aztec/noir-contracts.js/FPC";

// ... (set up the wallet and PXE)

// register the already deployed FPC contract in users PXE
const fpcContract = FPCContract.at(fpcAddress, userWallet);
await pxe.registerContract(fpcContract);

The fee payment method is created and used as follows, with similar syntax for private or public fee payments:

fpc
const tx = await bananaCoin.methods
.transfer_in_public(aliceAddress, bobAddress, bananasToSendToBob, 0)
.send({
fee: {
gasSettings,
paymentMethod: new PublicFeePaymentMethod(bananaFPC.address, aliceWallet),
},
})
.wait();
Source code: yarn-project/end-to-end/src/e2e_fees/public_payments.test.ts#L59-L69

In this example, thanks to the FPC's accepted_asset being banana tokens, Alice only needs to hold this token and not fee juice. The asset that a FPC accepts for paying fees is determined when the FPC is deployed. The function being called happens to also be a transfer of banana tokens to Bob.

More on FPCs here

See this section for paying fees via an FPC using the CLI wallet.

This method of fee payment will only work for environments where a sponsored fee paying contract is deployed. The sandbox comes with a sponsored fee paying contract deployed, so this can be used to pay for transactions without needing to bridge fee juice. To use sponsored FPCs in other environments, they will need to be deployed and funded with fee juice. Using a SponsoredFPC payment method is as simple as importing it, registering it and passing it the PXE:

import { SponsoredFeePaymentMethod } from "@aztec/aztec.js/fee/testing";
const paymentMethod = new SponsoredFeePaymentMethod(deployedSponsoredFPC);

Register the SponsoredFPC in the PXE:

import { SponsoredFPCContract } from "@aztec/noir-contracts.js/SponsoredFPC";

// ... (set up the wallet and PXE)

// register the already deployed SponsoredFPC contract in users PXE
const fpcContract = SponsoredFPCContract.at(sponsoredFpcAddress, userWallet);
await pxe.registerContract(fpcContract);

Then a transaction can specify this as the paymentMethod in the fee object. You can see an example of how to get a deployed instance of the sponsored FPC in the sandbox here. For example, a contract can be deployed with an fpc as follows:

const paymentMethod = new SponsoredFeePaymentMethod(deployedSponsoredFPC);
myAccountManager.deploy({ fee: { paymentMethod } });

You can find the corresponding CLI command info here

Fee Options

Functions pertaining to sending a transaction, such as deploy and send, each include a fee variable defined with the following (optional) parameters:

user_fee_options
/** Fee options as set by a user. */
export type UserFeeOptions = {
/** The fee payment method to use */
paymentMethod?: FeePaymentMethod;
/** The gas settings */
gasSettings?: Partial<FieldsOf<GasSettings>>;
/** Percentage to pad the base fee by, if empty, defaults to 0.5 */
baseFeePadding?: number;
/** Whether to run an initial simulation of the tx with high gas limit to figure out actual gas settings. */
estimateGas?: boolean;
/** Percentage to pad the estimated gas limits by, if empty, defaults to 0.1. Only relevant if estimateGas is set. */
estimatedGasPadding?: number;
};
Source code: yarn-project/entrypoints/src/interfaces.ts#L79-L93

Fee Payment Method

The paymentMethod is an object for the type of payment. Each of the implementations can be found here. For example:

fee_juice_method
/**
* Pay fee directly in the Fee Juice.
*/
export class FeeJuicePaymentMethod implements FeePaymentMethod {
Source code: yarn-project/aztec.js/src/fee/fee_juice_payment_method.ts#L6-L11

Gas Settings

gas_settings_vars
/** Gas usage and fees limits set by the transaction sender for different dimensions and phases. */
export class GasSettings {
constructor(
public readonly gasLimits: Gas,
public readonly teardownGasLimits: Gas,
public readonly maxFeesPerGas: GasFees,
public readonly maxPriorityFeesPerGas: GasFees,
) {}
Source code: yarn-project/stdlib/src/gas/gas_settings.ts#L11-L20

Gas Settings used in transactions specify gas limits and maximum fee rates (fees-per-gas)

The Gas and GasFees types each specify Data availability and L2 cost components, so the settings are:

  • gasLimits: DA and L2 gas limits
  • teardownGasLimits: DA and L2 gas limits for a txs optional teardown operation
  • maxFeesPerGas: maximum DA and L2 fees-per-gas
  • maxPriorityFeesPerGas: maximum priority DA and L2 fees-per-gas

Other params

Fee and gas padding params can be left to their default values, and the estimateGas boolean can be used when simulating a tx to calculate gas.