UNPKG

@solana/spl-token

Version:
383 lines (345 loc) 13.1 kB
import type { ConfirmOptions, Connection, PublicKey, Signer, TransactionSignature } from '@solana/web3.js'; import { sendAndConfirmTransaction, SystemProgram, Transaction } from '@solana/web3.js'; import type { Field, TokenMetadata } from '@solana/spl-token-metadata'; import { createInitializeInstruction, createRemoveKeyInstruction, createUpdateAuthorityInstruction, createUpdateFieldInstruction, pack, unpack, } from '@solana/spl-token-metadata'; import { TOKEN_2022_PROGRAM_ID } from '../../constants.js'; import { getSigners } from '../../actions/internal.js'; import { ExtensionType, getExtensionData, getNewAccountLenForExtensionLen } from '../extensionType.js'; import { updateTokenMetadata } from './state.js'; import { TokenAccountNotFoundError } from '../../errors.js'; import { unpackMint } from '../../state/index.js'; async function getAdditionalRentForNewMetadata( connection: Connection, address: PublicKey, tokenMetadata: TokenMetadata, programId = TOKEN_2022_PROGRAM_ID, ): Promise<number> { const info = await connection.getAccountInfo(address); if (!info) { throw new TokenAccountNotFoundError(); } const extensionLen = pack(tokenMetadata).length; const newAccountLen = getNewAccountLenForExtensionLen( info, address, ExtensionType.TokenMetadata, extensionLen, programId, ); if (newAccountLen <= info.data.length) { return 0; } const newRentExemptMinimum = await connection.getMinimumBalanceForRentExemption(newAccountLen); return newRentExemptMinimum - info.lamports; } async function getAdditionalRentForUpdatedMetadata( connection: Connection, address: PublicKey, field: string | Field, value: string, programId = TOKEN_2022_PROGRAM_ID, ): Promise<number> { const info = await connection.getAccountInfo(address); if (!info) { throw new TokenAccountNotFoundError(); } const mint = unpackMint(address, info, programId); const extensionData = getExtensionData(ExtensionType.TokenMetadata, mint.tlvData); if (extensionData === null) { throw new Error('TokenMetadata extension not initialized'); } const updatedTokenMetadata = updateTokenMetadata(unpack(extensionData), field, value); const extensionLen = pack(updatedTokenMetadata).length; const newAccountLen = getNewAccountLenForExtensionLen( info, address, ExtensionType.TokenMetadata, extensionLen, programId, ); if (newAccountLen <= info.data.length) { return 0; } const newRentExemptMinimum = await connection.getMinimumBalanceForRentExemption(newAccountLen); return newRentExemptMinimum - info.lamports; } /** * Initializes a TLV entry with the basic token-metadata fields. * * @param connection Connection to use * @param payer Payer of the transaction fees * @param mint Mint Account * @param updateAuthority Update Authority * @param mintAuthority Mint Authority * @param name Longer name of token * @param symbol Shortened symbol of token * @param uri URI pointing to more metadata (image, video, etc) * @param multiSigners Signing accounts if `authority` is a multisig * @param confirmOptions Options for confirming the transaction * @param programId SPL Token program account * * @return Signature of the confirmed transaction */ export async function tokenMetadataInitialize( connection: Connection, payer: Signer, mint: PublicKey, updateAuthority: PublicKey, mintAuthority: PublicKey | Signer, name: string, symbol: string, uri: string, multiSigners: Signer[] = [], confirmOptions?: ConfirmOptions, programId = TOKEN_2022_PROGRAM_ID, ): Promise<TransactionSignature> { const [mintAuthorityPublicKey, signers] = getSigners(mintAuthority, multiSigners); const transaction = new Transaction().add( createInitializeInstruction({ programId, metadata: mint, updateAuthority, mint, mintAuthority: mintAuthorityPublicKey, name, symbol, uri, }), ); return await sendAndConfirmTransaction(connection, transaction, [payer, ...signers], confirmOptions); } /** * Initializes a TLV entry with the basic token-metadata fields, * Includes a transfer for any additional rent-exempt SOL if required. * * @param connection Connection to use * @param payer Payer of the transaction fees * @param mint Mint Account * @param updateAuthority Update Authority * @param mintAuthority Mint Authority * @param name Longer name of token * @param symbol Shortened symbol of token * @param uri URI pointing to more metadata (image, video, etc) * @param multiSigners Signing accounts if `authority` is a multisig * @param confirmOptions Options for confirming the transaction * @param programId SPL Token program account * * @return Signature of the confirmed transaction */ export async function tokenMetadataInitializeWithRentTransfer( connection: Connection, payer: Signer, mint: PublicKey, updateAuthority: PublicKey, mintAuthority: PublicKey | Signer, name: string, symbol: string, uri: string, multiSigners: Signer[] = [], confirmOptions?: ConfirmOptions, programId = TOKEN_2022_PROGRAM_ID, ): Promise<TransactionSignature> { const [mintAuthorityPublicKey, signers] = getSigners(mintAuthority, multiSigners); const transaction = new Transaction(); const lamports = await getAdditionalRentForNewMetadata( connection, mint, { updateAuthority, mint, name, symbol, uri, additionalMetadata: [], }, programId, ); if (lamports > 0) { transaction.add(SystemProgram.transfer({ fromPubkey: payer.publicKey, toPubkey: mint, lamports: lamports })); } transaction.add( createInitializeInstruction({ programId, metadata: mint, updateAuthority, mint, mintAuthority: mintAuthorityPublicKey, name, symbol, uri, }), ); return await sendAndConfirmTransaction(connection, transaction, [payer, ...signers], confirmOptions); } /** * Updates a field in a token-metadata account. * If the field does not exist on the account, it will be created. * If the field does exist, it will be overwritten. * * The field can be one of the required fields (name, symbol, URI), or a * totally new field denoted by a "key" string. * @param connection Connection to use * @param payer Payer of the transaction fees * @param mint Mint Account * @param updateAuthority Update Authority * @param field Field to update in the metadata * @param value Value to write for the field * @param multiSigners Signing accounts if `authority` is a multisig * @param confirmOptions Options for confirming the transaction * @param programId SPL Token program account * * @return Signature of the confirmed transaction */ export async function tokenMetadataUpdateField( connection: Connection, payer: Signer, mint: PublicKey, updateAuthority: PublicKey | Signer, field: string | Field, value: string, multiSigners: Signer[] = [], confirmOptions?: ConfirmOptions, programId = TOKEN_2022_PROGRAM_ID, ): Promise<TransactionSignature> { const [updateAuthorityPublicKey, signers] = getSigners(updateAuthority, multiSigners); const transaction = new Transaction().add( createUpdateFieldInstruction({ programId, metadata: mint, updateAuthority: updateAuthorityPublicKey, field, value, }), ); return await sendAndConfirmTransaction(connection, transaction, [payer, ...signers], confirmOptions); } /** * Updates a field in a token-metadata account. * If the field does not exist on the account, it will be created. * If the field does exist, it will be overwritten. * Includes a transfer for any additional rent-exempt SOL if required. * * The field can be one of the required fields (name, symbol, URI), or a * totally new field denoted by a "key" string. * @param connection Connection to use * @param payer Payer of the transaction fees * @param mint Mint Account * @param updateAuthority Update Authority * @param field Field to update in the metadata * @param value Value to write for the field * @param multiSigners Signing accounts if `authority` is a multisig * @param confirmOptions Options for confirming the transaction * @param programId SPL Token program account * * @return Signature of the confirmed transaction */ export async function tokenMetadataUpdateFieldWithRentTransfer( connection: Connection, payer: Signer, mint: PublicKey, updateAuthority: PublicKey | Signer, field: string | Field, value: string, multiSigners: Signer[] = [], confirmOptions?: ConfirmOptions, programId = TOKEN_2022_PROGRAM_ID, ): Promise<TransactionSignature> { const [updateAuthorityPublicKey, signers] = getSigners(updateAuthority, multiSigners); const transaction = new Transaction(); const lamports = await getAdditionalRentForUpdatedMetadata(connection, mint, field, value, programId); if (lamports > 0) { transaction.add(SystemProgram.transfer({ fromPubkey: payer.publicKey, toPubkey: mint, lamports: lamports })); } transaction.add( createUpdateFieldInstruction({ programId, metadata: mint, updateAuthority: updateAuthorityPublicKey, field, value, }), ); return await sendAndConfirmTransaction(connection, transaction, [payer, ...signers], confirmOptions); } /** * Remove a field in a token-metadata account. * * The field can be one of the required fields (name, symbol, URI), or a * totally new field denoted by a "key" string. * @param connection Connection to use * @param payer Payer of the transaction fees * @param mint Mint Account * @param updateAuthority Update Authority * @param key Key to remove in the additional metadata portion * @param idempotent When true, instruction will not error if the key does not exist * @param multiSigners Signing accounts if `authority` is a multisig * @param confirmOptions Options for confirming the transaction * @param programId SPL Token program account * * @return Signature of the confirmed transaction */ export async function tokenMetadataRemoveKey( connection: Connection, payer: Signer, mint: PublicKey, updateAuthority: PublicKey | Signer, key: string, idempotent: boolean, multiSigners: Signer[] = [], confirmOptions?: ConfirmOptions, programId = TOKEN_2022_PROGRAM_ID, ): Promise<TransactionSignature> { const [updateAuthorityPublicKey, signers] = getSigners(updateAuthority, multiSigners); const transaction = new Transaction().add( createRemoveKeyInstruction({ programId, metadata: mint, updateAuthority: updateAuthorityPublicKey, key, idempotent, }), ); return await sendAndConfirmTransaction(connection, transaction, [payer, ...signers], confirmOptions); } /** * Update authority * * @param connection Connection to use * @param payer Payer of the transaction fees * @param mint Mint Account * @param updateAuthority Update Authority * @param newAuthority New authority for the token metadata, or unset * @param multiSigners Signing accounts if `authority` is a multisig * @param confirmOptions Options for confirming the transaction * @param programId SPL Token program account * * @return Signature of the confirmed transaction */ export async function tokenMetadataUpdateAuthority( connection: Connection, payer: Signer, mint: PublicKey, updateAuthority: PublicKey | Signer, newAuthority: PublicKey | null, multiSigners: Signer[] = [], confirmOptions?: ConfirmOptions, programId = TOKEN_2022_PROGRAM_ID, ): Promise<TransactionSignature> { const [updateAuthorityPublicKey, signers] = getSigners(updateAuthority, multiSigners); const transaction = new Transaction().add( createUpdateAuthorityInstruction({ programId, metadata: mint, oldAuthority: updateAuthorityPublicKey, newAuthority, }), ); return await sendAndConfirmTransaction(connection, transaction, [payer, ...signers], confirmOptions); }