Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

SVM Branch to main #116

Draft
wants to merge 4 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion clients/js/package.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"name": "@metaplex-foundation/mpl-bubblegum",
"version": "4.2.1",
"version": "4.3.1-beta.0",
"description": "Create and interact with compressed Metaplex NFTs",
"main": "dist/src/index.js",
"types": "dist/src/index.d.ts",
Expand Down
17 changes: 13 additions & 4 deletions clients/js/src/createTree.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,18 @@ export const createTree = async (
getMerkleTreeSize(input.maxDepth, input.maxBufferSize, input.canopyDepth);
const lamports = await context.rpc.getRent(space);

let programId;
if (input.compressionProgram) {
programId = Array.isArray(input.compressionProgram)
? input.compressionProgram[0]
: input.compressionProgram;
} else {
programId = context.programs.getPublicKey(
'splAccountCompression',
SPL_ACCOUNT_COMPRESSION_PROGRAM_ID
);
}

return (
transactionBuilder()
// Create the empty Merkle tree account.
Expand All @@ -36,10 +48,7 @@ export const createTree = async (
newAccount: input.merkleTree,
lamports,
space,
programId: context.programs.getPublicKey(
'splAccountCompression',
SPL_ACCOUNT_COMPRESSION_PROGRAM_ID
),
programId,
})
)
// Create the tree config.
Expand Down
26 changes: 26 additions & 0 deletions clients/js/src/generated/errors/mplBubblegum.ts
Original file line number Diff line number Diff line change
Expand Up @@ -604,6 +604,32 @@ export class InvalidCanopySizeError extends ProgramError {
codeToErrorMap.set(0x1799, InvalidCanopySizeError);
nameToErrorMap.set('InvalidCanopySize', InvalidCanopySizeError);

/** InvalidLogWrapper: Invalid log wrapper program */
export class InvalidLogWrapperError extends ProgramError {
readonly name: string = 'InvalidLogWrapper';

readonly code: number = 0x179a; // 6042

constructor(program: Program, cause?: Error) {
super('Invalid log wrapper program', program, cause);
}
}
codeToErrorMap.set(0x179a, InvalidLogWrapperError);
nameToErrorMap.set('InvalidLogWrapper', InvalidLogWrapperError);

/** InvalidCompressionProgram: Invalid compression program */
export class InvalidCompressionProgramError extends ProgramError {
readonly name: string = 'InvalidCompressionProgram';

readonly code: number = 0x179b; // 6043

constructor(program: Program, cause?: Error) {
super('Invalid compression program', program, cause);
}
}
codeToErrorMap.set(0x179b, InvalidCompressionProgramError);
nameToErrorMap.set('InvalidCompressionProgram', InvalidCompressionProgramError);

/**
* Attempts to resolve a custom program error from the provided error code.
* @category Errors
Expand Down
49 changes: 49 additions & 0 deletions clients/js/src/getCompressionPrograms.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
import { Context, PublicKey } from '@metaplex-foundation/umi';
import {
SPL_ACCOUNT_COMPRESSION_PROGRAM_ID,
SPL_NOOP_PROGRAM_ID,
} from './generated';

export const MPL_ACCOUNT_COMPRESSION_PROGRAM_ID =
'mcmt6YrQEMKw8Mw43FmpRLmf7BqRnFMKmAcbxE3xkAW' as PublicKey<'mcmt6YrQEMKw8Mw43FmpRLmf7BqRnFMKmAcbxE3xkAW'>;

export const MPL_NOOP_PROGRAM_ID =
'mnoopTCrg4p8ry25e4bcWA9XZjbNjMTfgYVGGEdRsf3' as PublicKey<'mnoopTCrg4p8ry25e4bcWA9XZjbNjMTfgYVGGEdRsf3'>;

// Constants for known genesis blockhashes on Solana.
const SOLANA_MAINNET_GENESIS_HASH =
'5eykt4UsFv8P8NJdTREpY1vzqKqZKvdpKuc147dw2N9d';
const SOLANA_DEVNET_GENESIS_HASH =
'EtWTRABZaYq6iMfeYKouRu166VU2xqa1wcaWoxPkrZBG';
const SOLANA_TESTNET_GENESIS_HASH =
'4uhcVJyU9pJkvQyS88uRDiswHXSCkY3zQawwpjk2NsNY';

export type CompressionPrograms = {
logWrapper: PublicKey;
compressionProgram: PublicKey;
};

export async function getCompressionPrograms(
context: Pick<Context, 'programs' | 'eddsa' | 'rpc'>
): Promise<CompressionPrograms> {
const genesisHash = await context.rpc.call<string>('getGenesisHash');

// Determine if the genesis hash matches known clusters.
const isSolanaCluster = [
SOLANA_MAINNET_GENESIS_HASH,
SOLANA_DEVNET_GENESIS_HASH,
SOLANA_TESTNET_GENESIS_HASH,
].includes(genesisHash);

// Return appropriate program IDs based on the cluster.
if (isSolanaCluster) {
return {
logWrapper: SPL_NOOP_PROGRAM_ID,
compressionProgram: SPL_ACCOUNT_COMPRESSION_PROGRAM_ID,
};
}
return {
logWrapper: MPL_NOOP_PROGRAM_ID,
compressionProgram: MPL_ACCOUNT_COMPRESSION_PROGRAM_ID,
};
}
1 change: 1 addition & 0 deletions clients/js/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,3 +7,4 @@ export * from './hooked';
export * from './leafAssetId';
export * from './merkle';
export * from './plugin';
export * from './getCompressionPrograms';
256 changes: 254 additions & 2 deletions clients/js/test/createTree.test.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,78 @@
import { generateSigner, publicKey } from '@metaplex-foundation/umi';
import { createAccount } from '@metaplex-foundation/mpl-toolbox';
import {
generateSigner,
publicKey,
Context,
Signer,
TransactionBuilder,
transactionBuilder,
PublicKey,
} from '@metaplex-foundation/umi';
import test from 'ava';
import { TreeConfig, createTree, fetchTreeConfigFromSeeds } from '../src';
import {
TreeConfig,
createTree,
createTreeConfig,
fetchTreeConfigFromSeeds,
safeFetchTreeConfigFromSeeds,
getMerkleTreeSize,
SPL_ACCOUNT_COMPRESSION_PROGRAM_ID,
getCompressionPrograms,
MPL_NOOP_PROGRAM_ID,
MPL_ACCOUNT_COMPRESSION_PROGRAM_ID,
} from '../src';
import { createUmi } from './_setup';

const createTreeWithSpecificMerkleOwner = async (
context: Parameters<typeof createAccount>[0] &
Parameters<typeof createTreeConfig>[0] &
Pick<Context, 'rpc'>,
input: Omit<Parameters<typeof createTreeConfig>[1], 'merkleTree'> & {
merkleTree: Signer;
merkleTreeSize?: number;
canopyDepth?: number;
merkleTreeOwner?: PublicKey;
}
): Promise<TransactionBuilder> => {
const space =
input.merkleTreeSize ??
getMerkleTreeSize(input.maxDepth, input.maxBufferSize, input.canopyDepth);
const lamports = await context.rpc.getRent(space);

let programId;
if (input.compressionProgram) {
programId = Array.isArray(input.compressionProgram)
? input.compressionProgram[0]
: input.compressionProgram;
} else {
programId = context.programs.getPublicKey(
'splAccountCompression',
SPL_ACCOUNT_COMPRESSION_PROGRAM_ID
);
}

return (
transactionBuilder()
// Create the empty Merkle tree account.
.add(
createAccount(context, {
payer: input.payer ?? context.payer,
newAccount: input.merkleTree,
lamports,
space,
programId: input.merkleTreeOwner ? input.merkleTreeOwner : programId,
})
)
// Create the tree config.
.add(
createTreeConfig(context, {
...input,
merkleTree: input.merkleTree.publicKey,
})
)
);
};

test('it can create a Bubblegum tree', async (t) => {
// Given a brand new merkle tree signer.
const umi = await createUmi();
Expand Down Expand Up @@ -60,3 +130,185 @@ test('it can create a Bubblegum tree using a newer size', async (t) => {
isPublic: false,
});
});

test('it can create a Bubblegum tree using mpl-account-compression and mpl-noop', async (t) => {
// Given a brand new merkle tree signer.
const umi = await createUmi();
const merkleTree = generateSigner(umi);

// For these tests, make sure `getCompressionPrograms` doesn't return spl programs.
const { logWrapper, compressionProgram } = await getCompressionPrograms(umi);
t.is(logWrapper, MPL_NOOP_PROGRAM_ID);
t.is(compressionProgram, MPL_ACCOUNT_COMPRESSION_PROGRAM_ID);

// When we create a tree at this address.
const builder = await createTree(umi, {
merkleTree,
maxDepth: 14,
maxBufferSize: 64,
...(await getCompressionPrograms(umi)),
});
await builder.sendAndConfirm(umi);

// Then an account exists at the merkle tree address.
t.true(await umi.rpc.accountExists(merkleTree.publicKey));

// And a tree config was created with the correct data.
const treeConfig = await fetchTreeConfigFromSeeds(umi, {
merkleTree: merkleTree.publicKey,
});
t.like(treeConfig, <TreeConfig>{
treeCreator: publicKey(umi.identity),
treeDelegate: publicKey(umi.identity),
totalMintCapacity: 2n ** 14n,
numMinted: 0n,
isPublic: false,
});
});

test('it cannot create a Bubblegum tree using invalid logWrapper with spl-account-compression', async (t) => {
// Given a brand new merkle tree signer.
const umi = await createUmi();
const merkleTree = generateSigner(umi);

// When we create a tree at this address.
const builder = await createTree(umi, {
merkleTree,
maxDepth: 14,
maxBufferSize: 64,
logWrapper: generateSigner(umi).publicKey,
});

const promise = builder.sendAndConfirm(umi);

// Then we expect a program error.
await t.throwsAsync(promise, { name: 'InvalidLogWrapper' });

// And an account does not exist at the merkle tree address.
t.false(await umi.rpc.accountExists(merkleTree.publicKey));

// And a tree config was not created with the correct data.
const treeConfig = await safeFetchTreeConfigFromSeeds(umi, {
merkleTree: merkleTree.publicKey,
});
t.is(treeConfig, null);
});

test('it cannot create a Bubblegum tree using invalid logWrapper with mpl-account-compression', async (t) => {
// Given a brand new merkle tree signer.
const umi = await createUmi();
const merkleTree = generateSigner(umi);

// When we create a tree at this address.
const builder = await createTree(umi, {
merkleTree,
maxDepth: 14,
maxBufferSize: 64,
logWrapper: generateSigner(umi).publicKey,
compressionProgram: MPL_ACCOUNT_COMPRESSION_PROGRAM_ID,
});

const promise = builder.sendAndConfirm(umi);

// Then we expect a program error.
await t.throwsAsync(promise, { name: 'InvalidLogWrapper' });

// And an account does not exist at the merkle tree address.
t.false(await umi.rpc.accountExists(merkleTree.publicKey));

// And a tree config was not created with the correct data.
const treeConfig = await safeFetchTreeConfigFromSeeds(umi, {
merkleTree: merkleTree.publicKey,
});
t.is(treeConfig, null);
});

test('it cannot create a Bubblegum tree when compression program does not match tree owned by spl-account-compression', async (t) => {
// Given a brand new merkle tree signer.
const umi = await createUmi();
const merkleTree = generateSigner(umi);

// When we create a tree at this address.
const builder = await createTreeWithSpecificMerkleOwner(umi, {
merkleTree,
maxDepth: 14,
maxBufferSize: 64,
compressionProgram: generateSigner(umi).publicKey,
merkleTreeOwner: umi.programs.getPublicKey(
'splAccountCompression',
SPL_ACCOUNT_COMPRESSION_PROGRAM_ID
),
});

const promise = builder.sendAndConfirm(umi);

// Then we expect a program error.
await t.throwsAsync(promise, { name: 'InvalidCompressionProgram' });

// And an account does not exist at the merkle tree address.
t.false(await umi.rpc.accountExists(merkleTree.publicKey));

// And a tree config was not created with the correct data.
const treeConfig = await safeFetchTreeConfigFromSeeds(umi, {
merkleTree: merkleTree.publicKey,
});
t.is(treeConfig, null);
});

test('it cannot create a Bubblegum tree when compression program does not match tree owned by mpl-account-compression', async (t) => {
// Given a brand new merkle tree signer.
const umi = await createUmi();
const merkleTree = generateSigner(umi);

// When we create a tree at this address.
const builder = await createTreeWithSpecificMerkleOwner(umi, {
merkleTree,
maxDepth: 14,
maxBufferSize: 64,
logWrapper: MPL_NOOP_PROGRAM_ID,
compressionProgram: generateSigner(umi).publicKey,
merkleTreeOwner: MPL_ACCOUNT_COMPRESSION_PROGRAM_ID,
});

const promise = builder.sendAndConfirm(umi);

// Then we expect a program error.
await t.throwsAsync(promise, { name: 'InvalidCompressionProgram' });

// And an account does not exist at the merkle tree address.
t.false(await umi.rpc.accountExists(merkleTree.publicKey));

// And a tree config was not created with the correct data.
const treeConfig = await safeFetchTreeConfigFromSeeds(umi, {
merkleTree: merkleTree.publicKey,
});
t.is(treeConfig, null);
});

test('it cannot create a Bubblegum tree with incorrect Merkle tree owner', async (t) => {
// Given a brand new merkle tree signer.
const umi = await createUmi();
const merkleTree = generateSigner(umi);

// When we create a tree at this address.
const builder = await createTreeWithSpecificMerkleOwner(umi, {
merkleTree,
maxDepth: 14,
maxBufferSize: 64,
merkleTreeOwner: generateSigner(umi).publicKey,
});

const promise = builder.sendAndConfirm(umi);

// Then we expect a program error.
await t.throwsAsync(promise, { name: 'IncorrectOwner' });

// And an account does not exist at the merkle tree address.
t.false(await umi.rpc.accountExists(merkleTree.publicKey));

// And a tree config was not created with the correct data.
const treeConfig = await safeFetchTreeConfigFromSeeds(umi, {
merkleTree: merkleTree.publicKey,
});
t.is(treeConfig, null);
});
Loading
Loading