Skip to content
This repository was archived by the owner on Mar 26, 2025. It is now read-only.

Commit bcd01a1

Browse files
authoredMay 9, 2023
fix: fixed minting compressed NFTs (#511)

File tree

5 files changed

+96
-24
lines changed

5 files changed

+96
-24
lines changed
 

‎.changeset/four-pugs-march.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'@metaplex-foundation/js': patch
3+
---
4+
5+
Fix minting compressed NFTs

‎packages/js/src/plugins/nftModule/NftClient.ts

+10-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,8 @@ import {
1111
approveNftDelegateOperation,
1212
ApproveNftUseAuthorityInput,
1313
approveNftUseAuthorityOperation,
14+
CreateCompressedNftInput,
15+
createCompressedNftOperation,
1416
CreateNftInput,
1517
createNftOperation,
1618
CreateSftInput,
@@ -229,7 +231,7 @@ export class NftClient {
229231
model: T,
230232
input?: Omit<
231233
FindNftByMintInput,
232-
'mintAddress' | 'tokenAddres' | 'tokenOwner'
234+
'mintAddress' | 'tokenAddress' | 'tokenOwner'
233235
>,
234236
options?: OperationOptions
235237
): Promise<T extends Metadata | PublicKey ? Nft | Sft : T> {
@@ -249,6 +251,13 @@ export class NftClient {
249251

250252
/** {@inheritDoc createNftOperation} */
251253
create(input: CreateNftInput, options?: OperationOptions) {
254+
if (input?.tree)
255+
return this.metaplex
256+
.operations()
257+
.execute(
258+
createCompressedNftOperation(input as CreateCompressedNftInput),
259+
options
260+
);
252261
return this.metaplex
253262
.operations()
254263
.execute(createNftOperation(input), options);

‎packages/js/src/plugins/nftModule/operations/createCompressedNft.ts

+78-21
Original file line numberDiff line numberDiff line change
@@ -12,10 +12,11 @@ import {
1212
import {
1313
SPL_ACCOUNT_COMPRESSION_PROGRAM_ID,
1414
SPL_NOOP_PROGRAM_ID,
15-
changeLogEventV1Beet,
15+
deserializeChangeLogEventV1,
1616
} from '@solana/spl-account-compression';
1717
import { PublicKey } from '@solana/web3.js';
1818
import { BN } from 'bn.js';
19+
import base58 from 'bs58';
1920
import { SendAndConfirmTransactionResponse } from '../../rpcModule';
2021
import { assertNft, Nft } from '../models';
2122
import { Option, TransactionBuilder, TransactionBuilderOptions } from '@/utils';
@@ -194,11 +195,20 @@ export type CreateCompressedNftOutput = {
194195
/** The blockchain response from sending and confirming the transaction. */
195196
response: SendAndConfirmTransactionResponse;
196197

197-
/** The newly created SFT and, potentially, its associated token. */
198+
/** The newly created NFT and, potentially, its associated token. */
198199
nft: Nft;
199200

200-
/** The asset id of the leaf. */
201+
/** The mint address is the compressed NFT's assetId. */
201202
mintAddress: PublicKey;
203+
204+
/** The metadata address is the compressed NFT's assetId. */
205+
metadataAddress: PublicKey;
206+
207+
/** The master edition address is the compressed NFT's assetId. */
208+
masterEditionAddress: PublicKey;
209+
210+
/** The token address is the compressed NFT's assetId. */
211+
tokenAddress: PublicKey;
202212
};
203213

204214
/**
@@ -226,29 +236,64 @@ export const createCompressedNftOperationHandler: OperationHandler<CreateCompres
226236
const output = await builder.sendAndConfirm(metaplex, confirmOptions);
227237
scope.throwIfCanceled();
228238

229-
const {
230-
response: { signature },
231-
} = output;
232-
const txInfo = await metaplex.connection.getTransaction(signature, {
233-
maxSupportedTransactionVersion: 0,
239+
const txInfo = await metaplex.connection.getTransaction(
240+
output.response.signature,
241+
{
242+
maxSupportedTransactionVersion: 0,
243+
}
244+
);
245+
scope.throwIfCanceled();
246+
247+
// find the index of the bubblegum instruction
248+
const relevantIndex =
249+
txInfo!.transaction.message.compiledInstructions.findIndex(
250+
(instruction) => {
251+
return (
252+
txInfo?.transaction.message.staticAccountKeys[
253+
instruction.programIdIndex
254+
].toBase58() === 'BGUMAp9Gq7iTEuizy4pqaxsTyUCBK68MDfK752saRPUY'
255+
);
256+
}
257+
);
258+
259+
// locate the no-op inner instructions called via cpi from bubblegum
260+
const relevantInnerIxs = txInfo!.meta?.innerInstructions?.[
261+
relevantIndex
262+
].instructions.filter((instruction) => {
263+
return (
264+
txInfo?.transaction.message.staticAccountKeys[
265+
instruction.programIdIndex
266+
].toBase58() === 'noopb9bkMVfRPU8AsbpTUg8AQkHtKwMYZiFUjNRtMmV'
267+
);
234268
});
235-
const relevantIx = txInfo!.transaction.message.compiledInstructions.find(
236-
(instruction) => {
237-
return (
238-
txInfo!.transaction.message.staticAccountKeys[
239-
instruction.programIdIndex
240-
].toBase58() === 'noopb9bkMVfRPU8AsbpTUg8AQkHtKwMYZiFUjNRtMmV'
269+
270+
// when no valid noop instructions are found, throw an error
271+
if (!relevantInnerIxs || relevantInnerIxs.length == 0)
272+
throw Error('Unable to locate valid noop instructions');
273+
274+
// locate the asset index by attempting to locate and parse the correct `relevantInnerIx`
275+
let assetIndex: number | undefined = undefined;
276+
// note: the `assetIndex` is expected to be at position `1`, and normally expect only 2 `relevantInnerIx`
277+
for (let i = relevantInnerIxs.length - 1; i > 0; i--) {
278+
try {
279+
const changeLogEvent = deserializeChangeLogEventV1(
280+
Buffer.from(base58.decode(relevantInnerIxs[i]?.data!))
241281
);
282+
283+
// extract a successful changelog index
284+
assetIndex = changeLogEvent?.index;
285+
} catch (__) {
286+
// do nothing, invalid data is handled just after the for loop
242287
}
243-
);
288+
}
244289

245-
const [changeLog] = changeLogEventV1Beet.deserialize(
246-
Buffer.from(relevantIx!.data)
247-
);
290+
// when no `assetIndex` was found, throw an error
291+
if (typeof assetIndex == 'undefined')
292+
throw Error('Unable to locate the newly minted assetId ');
248293

249294
const assetId = await getLeafAssetId(
250295
operation.input.tree,
251-
new BN(changeLog.index)
296+
new BN(assetIndex)
252297
);
253298

254299
const nft = await metaplex.nfts().findByAssetId(
@@ -260,7 +305,19 @@ export const createCompressedNftOperationHandler: OperationHandler<CreateCompres
260305
scope.throwIfCanceled();
261306

262307
assertNft(nft);
263-
return { ...output, nft };
308+
309+
return {
310+
...output,
311+
nft,
312+
/**
313+
* the assetId is impossible to know before the compressed nft is minted
314+
* all these addresses are derived from, or are, the `assetId`
315+
*/
316+
mintAddress: assetId,
317+
tokenAddress: assetId,
318+
metadataAddress: nft.metadataAddress,
319+
masterEditionAddress: nft.edition.address,
320+
};
264321
},
265322
};
266323

@@ -316,7 +373,7 @@ export type CreateCompressedNftBuilderContext = Omit<
316373
>;
317374

318375
/**
319-
* Creates a new SFT.
376+
* Creates a new compressed NFT.
320377
*
321378
* ```ts
322379
* const transactionBuilder = await metaplex

‎packages/js/src/plugins/nftModule/operations/createNft.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -133,7 +133,7 @@ export type CreateNftInput = {
133133
* Describes the asset class of the token.
134134
* It can be one of the following:
135135
* - `TokenStandard.NonFungible`: A traditional NFT (master edition).
136-
* - `TokenStandard.FungibleAsset`: A fungible token with metadata that can also have attrributes.
136+
* - `TokenStandard.FungibleAsset`: A fungible token with metadata that can also have attributes.
137137
* - `TokenStandard.Fungible`: A fungible token with simple metadata.
138138
* - `TokenStandard.NonFungibleEdition`: A limited edition NFT "printed" from a master edition.
139139
* - `TokenStandard.ProgrammableNonFungible`: A master edition NFT with programmable configuration.

‎packages/js/test/helpers/setup.ts

+2-1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import {
1010
CreateNftInput,
1111
KeypairSigner,
1212
CreateSftInput,
13+
NftWithToken,
1314
} from '@/index';
1415

1516
export type MetaplexTestOptions = {
@@ -57,7 +58,7 @@ export const createNft = async (
5758
...input,
5859
});
5960

60-
return nft;
61+
return nft as NftWithToken;
6162
};
6263

6364
export const createCollectionNft = (

0 commit comments

Comments
 (0)
This repository has been archived.