UNPKG

@solana/spl-token

Version:
171 lines (158 loc) 5.05 kB
import { struct, u8 } from '@solana/buffer-layout'; import { u64 } from '@solana/buffer-layout-utils'; import type { AccountMeta, PublicKey, Signer } from '@solana/web3.js'; import { TransactionInstruction } from '@solana/web3.js'; import { TOKEN_PROGRAM_ID } from '../constants.js'; import { TokenInvalidInstructionDataError, TokenInvalidInstructionKeysError, TokenInvalidInstructionProgramError, TokenInvalidInstructionTypeError, } from '../errors.js'; import { addSigners } from './internal.js'; import { TokenInstruction } from './types.js'; /** TODO: docs */ export interface TransferCheckedInstructionData { instruction: TokenInstruction.TransferChecked; amount: bigint; decimals: number; } /** TODO: docs */ export const transferCheckedInstructionData = struct<TransferCheckedInstructionData>([ u8('instruction'), u64('amount'), u8('decimals'), ]); /** * Construct a TransferChecked instruction * * @param source Source account * @param mint Mint account * @param destination Destination account * @param owner Owner of the source account * @param amount Number of tokens to transfer * @param decimals Number of decimals in transfer amount * @param multiSigners Signing accounts if `owner` is a multisig * @param programId SPL Token program account * * @return Instruction to add to a transaction */ export function createTransferCheckedInstruction( source: PublicKey, mint: PublicKey, destination: PublicKey, owner: PublicKey, amount: number | bigint, decimals: number, multiSigners: (Signer | PublicKey)[] = [], programId = TOKEN_PROGRAM_ID, ): TransactionInstruction { const keys = addSigners( [ { pubkey: source, isSigner: false, isWritable: true }, { pubkey: mint, isSigner: false, isWritable: false }, { pubkey: destination, isSigner: false, isWritable: true }, ], owner, multiSigners, ); const data = Buffer.alloc(transferCheckedInstructionData.span); transferCheckedInstructionData.encode( { instruction: TokenInstruction.TransferChecked, amount: BigInt(amount), decimals, }, data, ); return new TransactionInstruction({ keys, programId, data }); } /** A decoded, valid TransferChecked instruction */ export interface DecodedTransferCheckedInstruction { programId: PublicKey; keys: { source: AccountMeta; mint: AccountMeta; destination: AccountMeta; owner: AccountMeta; multiSigners: AccountMeta[]; }; data: { instruction: TokenInstruction.TransferChecked; amount: bigint; decimals: number; }; } /** * Decode a TransferChecked instruction and validate it * * @param instruction Transaction instruction to decode * @param programId SPL Token program account * * @return Decoded, valid instruction */ export function decodeTransferCheckedInstruction( instruction: TransactionInstruction, programId = TOKEN_PROGRAM_ID, ): DecodedTransferCheckedInstruction { if (!instruction.programId.equals(programId)) throw new TokenInvalidInstructionProgramError(); if (instruction.data.length !== transferCheckedInstructionData.span) throw new TokenInvalidInstructionDataError(); const { keys: { source, mint, destination, owner, multiSigners }, data, } = decodeTransferCheckedInstructionUnchecked(instruction); if (data.instruction !== TokenInstruction.TransferChecked) throw new TokenInvalidInstructionTypeError(); if (!source || !mint || !destination || !owner) throw new TokenInvalidInstructionKeysError(); // TODO: key checks? return { programId, keys: { source, mint, destination, owner, multiSigners, }, data, }; } /** A decoded, non-validated TransferChecked instruction */ export interface DecodedTransferCheckedInstructionUnchecked { programId: PublicKey; keys: { source: AccountMeta | undefined; mint: AccountMeta | undefined; destination: AccountMeta | undefined; owner: AccountMeta | undefined; multiSigners: AccountMeta[]; }; data: { instruction: number; amount: bigint; decimals: number; }; } /** * Decode a TransferChecked instruction without validating it * * @param instruction Transaction instruction to decode * * @return Decoded, non-validated instruction */ export function decodeTransferCheckedInstructionUnchecked({ programId, keys: [source, mint, destination, owner, ...multiSigners], data, }: TransactionInstruction): DecodedTransferCheckedInstructionUnchecked { return { programId, keys: { source, mint, destination, owner, multiSigners, }, data: transferCheckedInstructionData.decode(data), }; }