Skip to content

Latest commit

 

History

History
232 lines (188 loc) · 8.82 KB

README.md

File metadata and controls

232 lines (188 loc) · 8.82 KB

Batch mint SDK

This library allows to easily create a batch mint (a compressed NFT tree that is initialized off-chain).

The merkle tree is fully compatible with Metaplex Bubblegum.

Motivation

In case if you are minting a lot of NFTs, Metaplex Bubblegum allows to significantly reduce the cost on storage. Yet you still have to make a separate transaction for each minted NFT, that's why the initial creation of a big package of NFTs (e.g. for a game you are launching) can be not cheap.

The batch mint solves this problem by moving the creation of the whole initial set of NFTs to off-chain.

  1. you create the whole batch mint, which is an off-chain representation of the merkle tree populate it with all the assets you want to be included in your initial set
  2. you persist the batch mint into an immutable storage, to make it available for validators
  3. you push the whole tree of NFTs to Solana in a single operation (can consist of a couple of transactions)

And as the result you have a whole tree of NFTs, with expense of one account and a couple of solana transaction.

Usage

This section demonstrates the complete flow of batch mint creation.

⚠️ To be able to create a batch mint, you need to have a stake in MPLX tokens. TODO: add link to staking page.

Example of batch minting:

use bubblegum_batch_sdk::batch_mint_client::BatchMintClient;
use solana_client::nonblocking::rpc_client::RpcClient;
use solana_sdk::signer::keypair::Keypair;
use mpl_bubblegum::types::{Collection, Creator, MetadataArgs, TokenProgramVersion, TokenStandard};
use std::sync::Arc;
use std::time::Duration;
use solana_sdk::pubkey::Pubkey;
use solana_sdk::signature::Signer;

// the one who pays for the batch mint
// also it's tree creator in this context
let payer: Keypair = Keypair::new();
// can be same as payer
let staker: Keypair = Keypair::new();

let url = "https://api.devnet.solana.com".to_string(); // Solana RPC node address
let timeout = Duration::from_secs(1);
let solana_client = Arc::new(RpcClient::new_with_timeout(url, timeout));

let batch_mint_client = BatchMintClient::new(solana_client);

// Account for a merkle tree data we are going to create
let tree_data_account = Keypair::new();

// Creating Solana account for storing tree and initializing it.
// Will prepare merkle tree with depth 10 (not counting root),
// 32 cells changelog buffer (max 32 concurrent changes),
// and canopy tree with depth 4 (not counting root).
let sign = batch_mint_client.prepare_tree(
    &payer,
    &payer,
    &tree_data_account,
    20, // tree depth
    256,// maximum concurrent changes
    4   // canopy tree depth
).await?;

let batch_mint_builder = batch_mint_client.create_batch_mint_builder(&tree_data_account.pubkey())
    .await?;

// Adding NTF asset
let assets_to_add: &[(MetadataArgs, Pubkey, Pubkey)] = &[
    (
        MetadataArgs {
            name: "asset name".to_string(),
            symbol: "symbol".to_string(),
            uri: "https://asset.uri".to_string(),
            // Royalty basis points that goes to creators in secondary sales (0-10000)
            seller_fee_basis_points: 10,
            primary_sale_happened: false,
            is_mutable: true,
            // nonce for easy calculation of editions, if present
            edition_nonce: Some(1),
            token_standard: Some(TokenStandard::NonFungibleEdition),
            collection: Some(Collection {
                verified: false,
                key: Pubkey::new_unique(),
            }),
            uses: None,
            token_program_version: TokenProgramVersion::Original,
            creators: vec![Creator {
                address: Pubkey::new_unique(),
                verified: false,
                // The percentage share.
                //
                // The value is a percentage, not basis points.
                share: 5,
            }],
        },
        Pubkey::new_unique(),
        Pubkey::new_unique(),
    )
];
for (asset, asset_owner, asset_delegate) in assets_to_add {
    batch_mint_builder.add_asset(&asset_owner, &asset_delegate, &asset);
}

// Creating batch mint object to be persisted in Arweave/IPFS/etc.
let batch_mint = batch_mint_builder.build_batch_mint()?;

// Persisting batch mint to Arweave, where it will
// be picked up from by a DAS operator node.
let mut batch_mint_json_bytes = Vec::<u8>::new();
batch_mint.write_as_json(&mut batch_mint_json_bytes)?;
// save batch mint JSON file to decentralized storage
let metadata_url: String = "https://decentralize.storage/batch_minted_assets.json".to_string();
// hash(xxhash) of persisted batch mint
// hash can be count with xxhsum CLI tool
// example: xxhsum batch_mint_assets.json
let metadata_hash: String = "e1b2effd80ad2ada".to_string();

// Finalize batch mint in solana:
// "move" offchain merkle tree along with the canopy tree to the account.
let sign = batch_mint_client.finalize_tree(
    &payer,
    &metadata_url,
    &metadata_hash,
    &batch_mint_builder,
    &payer,
    &staker
).await?;

First we need to have an account with a stack in MPLX (TODO add stack details). Tree creator can be same account as staker, or can be different.

Then we need two accounts for storing merkle tree data itself, and additional configs required by Metaplex Bubblegum.

After that we create a batch mint builder object. It is a convenient wrapper that helps to easily:

  • add assets to the tree
  • generate batch mint object that should be persisted to an immutable storage
  • finalize batch mint on chain (by sending tree root with proofs)
  • write canopy (top level part of the tree that is saved on chain)

Using batch mint builder object we add an asset to the batch mint we build.

When all the assets are added (only one in our example), we generate a batch mint metadata JSON docuument and save it to Arweave. As the result of this saving we need to get a URL of the persisted metadata, and the metadata hash.

At this point we are ready to finalize the batch mint by calling finalize_tree. This operation "pushes" the merkle tree that had been created off-chain into solana. If the canopy has been specified, it is also pushed in scope finalize_tree.

When DAO operator nodes detect finalize_tree transaction, they download batch mint JSON from the immutable storage it had been saved to, and become ready to validate changes to the tree.

Compressed NFTs (merkle trees) that are created using the batch mint flow, are fully compatible with all Metaplex Bubblegum instructions.

Also if you want some asset have collection verified, tou can add collection config

let mut batch_mint_builder = batch_mint_client.create_batch_mint_builder(&tree_data_account.pubkey())
    .await?;

// Setup collection config if you want to add assets with verified collection
let collection_authority = Keypair::new();
batch_mint_builder.setup_collection_config(CollectionConfig {
    collection_authority,
    collection_authority_record_pda: None,
    collection_mint: Pubkey::new_unique(),
    collection_metadata: Pubkey::new_unique(),
    edition_account: Pubkey::new_unique(),
});

All other steps are the same as regular batch mint

Splitting the batch mint creation in time

You may want to not fill all the assets and create the merkle tree at once, but prepare a part of batch mint, and then later (after hours, days, etc.) fill the rest of asserts, and push the tree to Solana.

For that you need to create a BatchMintBuilder, populate it with a batch of assets, generate the BatchMint object and save it as JSON somewhere (immutable storage, object store, local file, etc.)

let tree_data_account = Keypair::new();
let batch_mint_client: BatchMintClient = ...;

batch_mint_client.prepare_tree(&payer, &tree_creator, &tree_data_account.pubkey(), 10, 32, 4)
    .awailt()?;

let batch_mint_builder = batch_mint_client.create_batch_mint_builder(&tree_data_account.pubkey())
    .await()?;

let assets_to_add: &[(MetadataArgs, Pubkey, Pubkey)] = ...;
for (asset, asset_owner, asset_delegate) in assets_to_add {
    batch_mint_builder.add_asset(&asset_owner, &asset_delegate, &asset);
}

{
    let mut file = std::fs::File::create("batch_mint.json")?;
    let batch_mint = batch_mint_builder.build_batch_mint();
    batch_mint.write_as_json(&mut file)?;
}

Later you can recover the BatchMintBuilder from this persisted BatchMint JSON representation and continue the flow.

let mut file = std::fs::File::create("batch_mint.json")?;
let batch_mint = BatchMint::read_as_json(&file).unwrap();
let batch_mint_builder = batch_mint_client.restore_batch_mint_builder(&batch_mint).await?;

Running tests

See Integration tests