Freeze Token Payment
Overview
The Freeze Token Payment guard allows minting frozen NFTs by charging the payer a specific amount of tokens from a certain mint account. Frozen NFTs cannot be transferred or listed on any marketplaces until thawed.
Frozen NFTs can be thawed by anyone as long as one of the following conditions is met:
- The Candy Machine has minted out.
- The Candy Machine was deleted.
- The configured Freeze Period — which can be a maximum of 30 days — has passed.
The tokens are transferred to a "Freeze Escrow" account which must be initialized by the Candy Guard authority before minting can start. Once all Frozen NFTs have been thawed, the funds can be unlocked and transferred to the configured destination account by the Candy Guard authority.
You may initialize the Freeze Escrow account, thaw NFTs and unlock funds via the route instruction of this guard.
Guard Settings
The Freeze Token Payment guard contains the following settings:
- Amount: The number of tokens to charge the payer.
- Mint: The address of the mint account defining the SPL Token we want to pay with.
- Destination Associated Token Address (ATA): The address of the associated token account to eventually send the tokens to. We can get this address by finding the Associated Token Address PDA using the Mint attribute and the address of any wallet that should receive these tokens.
JavaScript — Umi library (recommended)
Here’s how we can create a Candy Machine using the Freeze Token Payment guard. Note that, in this example, we’re using the current identity as the destination wallet.
import { findAssociatedTokenPda } from "@metaplex-foundation/mpl-toolbox";
create(umi, {
// ...
guards: {
freezeTokenPayment: some({
amount: 300,
mint: tokenMint.publicKey,
destinationAta: findAssociatedTokenPda(umi, {
mint: tokenMint.publicKey,
owner: umi.identity.publicKey,
}),
}),
},
});
API References: create, FreezeTokenPayment
JavaScript — SDK
Here’s how we can create a Candy Machine using the Freeze Token Payment guard via the JS SDK. Note that, in this example, we’re using the current identity as the destination wallet.
import { token } from "@metaplex-foundation/js";
const { candyMachine } = await metaplex.candyMachines().create({
// ...
guards: {
freezeTokenPayment: {
amount: token(300),
mint: tokenMint.address,
destinationAta: metaplex.tokens().pdas().associatedTokenAccount({
mint: tokenMint.address,
owner: metaplex.identity().publicKey,
}),
},
},
});
API References: Operation, Input, Output, Transaction Builder, Guard Settings.
Mint Settings
The Freeze Token Payment guard contains the following Mint Settings:
- Mint: The address of the mint account defining the SPL Token we want to pay with.
- Destination Associated Token Address (ATA): The address of the associated token account to eventually send the tokens to.
- NFT Rule Set (optional): The Rule Set of the minted NFT, if we are minting a Programmable NFT with a Rule Set.
Note that, if you’re planning on constructing instructions without the help of our SDKs, you will need to provide these Mint Settings and more as a combination of instruction arguments and remaining accounts. See the Candy Guard’s program documentation for more details.
JavaScript — Umi library (recommended)
You may pass the Mint Settings of the Freeze Token Payment guard using the mintArgs
argument like so.
mintV2(umi, {
// ...
mintArgs: {
freezeTokenPayment: some({
mint: tokenMint.publicKey,
destinationAta,
}),
},
});
API References: mintV2, FreezeTokenPaymentMintArgs
JavaScript — SDK
The JS SDK does not require any Mint Settings for the Freeze Token Payment guard since it can infer them from the provided Candy Machine model.
Route Instruction
The Freeze Token Payment route instruction supports the following features.
Initialize the Freeze Escrow
Path: initialize
When using the Freeze Token Payment guard, we must initialize the Freeze Escrow account before minting can start. This will create a PDA account derived from the Destination ATA attribute of the guard's settings.
The Freeze Escrow PDA account will keep track of several parameters such as:
- How many Frozen NFTs were minted through this guard.
- When was the first Frozen NFT minted via this guard as the Freeze Period starts counting after that.
When initializing this Freeze Escrow account, we must provide the following arguments to the route instruction of the guard:
- Path =
initialize
: Selects the path to execute in the route instruction. - Mint: The address of the mint account defining the SPL Token we want to pay with.
- Destination Associated Token Address (ATA): The address of the associated token account to eventually send the tokens to.
- Period: The amount of time in seconds that the Freeze Period should last. This can be a maximum of 30 days (2,592,000 seconds) and it will start from the very first Frozen NFT minted via this guard. The Freeze Period provides a safety mechanism to ensure Frozen NFTs can eventually be thawed even if the Candy Machine never mints out.
- Candy Guard Authority: The authority of the Candy Guard account as a Signer.
Last but not least, the Freeze Escrow PDA account will receive the tokens of all Frozen NFTs minted through this guard.
JavaScript — Umi library (recommended)
In the example below, we initialize the Freeze Escrow account with a maximum Freeze Period of 15 days and we use the current identity as the Candy Guard authority.
route(umi, {
// ...
guard: "freezeTokenPayment",
routeArgs: {
path: "initialize",
mint: tokenMint.publicKey,
destinationAta,
period: 15 * 24 * 60 * 60, // 15 days.
candyGuardAuthority: umi.identity,
},
});
API References: route, FreezeTokenPaymentRouteArgsInitialize
JavaScript — SDK
In the example below, we initialize the Freeze Escrow account with a maximum Freeze Period of 15 days and we use the current identity as the Candy Guard authority.
Note that the JS SDK does not require the Mint and Destination ATA to be passed in since it can get it from the provided Candy Machine model.
await metaplex.candyMachines().callGuardRoute({
candyMachine,
guard: "freezeTokenPayment",
settings: {
path: "initialize",
period: 15 * 24 * 60 * 60, // 15 days.
candyGuardAuthority: metaplex.identity(),
},
});
API References: Operation, Input, Output, Transaction Builder, Route Settings.
Thaw a Frozen NFT
Path: thaw
Frozen NFTs can be thawed by anyone as long as one of the following conditions is met:
- The Candy Machine has minted out.
- The Candy Machine was deleted.
- The configured Freeze Period — which can be a maximum of 30 days — has passed.
Note that since the tokens in the Freeze Escrow are not transferrable until all NFTs are thawed, this creates an incentive for the treasury to thaw all NFTs as soon as possible.
To thaw a Frozen NFT, we must provide the following arguments to the route instruction of the guard:
- Path =
thaw
: Selects the path to execute in the route instruction. - Mint: The address of the mint account defining the SPL Token we want to pay with.
- Destination Associated Token Address (ATA): The address of the associated token account to eventually send the tokens to.
- NFT Mint: The mint address of the Frozen NFT to thaw.
- NFT Owner: The address of the owner of the Frozen NFT to thaw.
- NFT Token Standard: The token standard of the Frozen NFT to thaw.
- NFT Rule Set (optional): The Rule Set of the Frozen NFT to thaw, if we are thawing a Programmable NFT with a Rule Set.
JavaScript — Umi library (recommended)
In the example below, we thaw a Frozen NFT that belongs to the current identity.
route(umi, {
// ...
guard: "freezeTokenPayment",
routeArgs: {
path: "thaw",
mint: tokenMint.publicKey,
destinationAta,
nftMint: nftMint.publicKey,
nftOwner: umi.identity.publicKey,
nftTokenStandard: candyMachine.tokenStandard,
},
});
API References: route, FreezeTokenPaymentRouteArgsThaw
JavaScript — SDK
In the example below, we thaw a Frozen NFT that belongs to the current identity.
Note that the JS SDK does not require the Mint and Destination ATA to be passed in since it can get it from the provided Candy Machine model. It also does not require the NFT Token Standard or the NFT Rule Set as it does not support minting Programmable NFTs.
import { toPublicKey } from "@metaplex-foundation/js";
await metaplex.candyMachines().callGuardRoute({
candyMachine,
guard: "freezeTokenPayment",
settings: {
path: "thaw",
nftMint: toPublicKey("GhFM53E6NEW7Ud8Gqh34WLBztkpe74PRtbHEU4b6cwWo"),
nftOwner: metaplex.identity().publicKey,
},
});
API References: Operation, Input, Output, Transaction Builder, Route Settings.
Unlock Funds
Path: unlockFunds
Once all Frozen NFTs have been thawed, the treasury can unlock the funds from the Freeze Escrow account. This will transfer the tokens to the configured Destination ATA address.
To unlock the funds, we must provide the following arguments to the route instruction of the guard:
- Path =
unlockFunds
: Selects the path to execute in the route instruction. - Mint: The address of the mint account defining the SPL Token we want to pay with.
- Destination Associated Token Address (ATA): The address of the associated token account to eventually send the tokens to.
- Candy Guard Authority: The authority of the Candy Guard account as a Signer.
JavaScript — Umi library (recommended)
In the example below, we unlock the funds from the Freeze Escrow account using the current identity as the Candy Guard authority.
route(umi, {
// ...
guard: "freezeTokenPayment",
routeArgs: {
path: "unlockFunds",
mint: tokenMint.publicKey,
destinationAta,
candyGuardAuthority: umi.identity,
},
});
API References: route, FreezeTokenPaymentRouteArgsUnlockFunds
JavaScript — SDK
In the example below, we unlock the funds from the Freeze Escrow account using the current identity as the Candy Guard authority.
await metaplex.candyMachines().callGuardRoute({
candyMachine,
guard: "freezeTokenPayment",
settings: {
path: "unlockFunds",
candyGuardAuthority: metaplex.identity(),
},
});
API References: Operation, Input, Output, Transaction Builder, Route Settings.
Stop Freezing NFTs
It is possible to stop the freezing of NFTs within a Freeze Token Payment guard. In other words, new-minted NFTs will no longer be frozen but existing Frozen NFTs will remain frozen.
There are several ways of achieving this, which can be separated into two categories:
- ☀️ Can Thaw: Existing Frozen NFTs can be thawed by anyone using the
thaw
path of the route instruction. - ❄️ Cannot Thaw: Existing Frozen NFTs cannot be thawed yet and we have to wait for one "Can Thaw" condition to be met.
With that in mind, here is the exhaustive list of ways to stop freezing NFTs and whether or not each of them allows thawing existing Frozen NFTs:
- The Candy Machine has minted out → ☀️ Can Thaw.
- The configured Freeze Period — which can be a maximum of 30 days — has passed → ☀️ Can Thaw.
- The Candy Machine account was deleted → ☀️ Can Thaw.
- The Candy Guard account was deleted → ❄️ Cannot Thaw.
- The Freeze Token Payment guard was removed from the settings → ❄️ Cannot Thaw.
Freeze Escrows and Guard Groups
When using multiple Freeze Token Payment guards within various Guard Groups, it is important to understand the relationship between a Freeze Token Payment guard and a Freeze Escrow account.
The Freeze Escrow account is a PDA derived from a Destination ATA address. This means that if multiple Freeze Token Payment guards are configured to use the same Destination ATA address, they will all share the same Freeze Escrow account.
Therefore, they will also share the same Freeze Period and all tokens will be collected by the same escrow account. This also means, we only need to call the initialize
route instruction once per configured Destination ATA address.
It is also possible to use multiple Freeze Token Payment guards with different Destination ATA addresses. In this case, each Freeze Token Payment guard will have its own Freeze Escrow account and its own Freeze Period.
The example below illustrates a Candy Machine with three Freeze Token Payment guards in three groups such that:
- Groups 1 and 2 share the same Destination ATA address and therefore the same Freeze Escrow account.
- Group 3 has its own Destination ATA address and therefore its own Freeze Escrow account.