UNPKG

@solana/spl-token

Version:
152 lines (139 loc) 4.38 kB
import { struct, u8 } from '@solana/buffer-layout'; import type { AccountMeta, Signer } from '@solana/web3.js'; import { PublicKey, SYSVAR_RENT_PUBKEY, 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 InitializeMultisigInstructionData { instruction: TokenInstruction.InitializeMultisig; m: number; } /** TODO: docs */ export const initializeMultisigInstructionData = struct<InitializeMultisigInstructionData>([ u8('instruction'), u8('m'), ]); /** * Construct an InitializeMultisig instruction * * @param account Multisig account * @param signers Full set of signers * @param m Number of required signatures * @param programId SPL Token program account * * @return Instruction to add to a transaction */ export function createInitializeMultisigInstruction( account: PublicKey, signers: (Signer | PublicKey)[], m: number, programId = TOKEN_PROGRAM_ID, ): TransactionInstruction { const keys = [ { pubkey: account, isSigner: false, isWritable: true }, { pubkey: SYSVAR_RENT_PUBKEY, isSigner: false, isWritable: false }, ]; for (const signer of signers) { keys.push({ pubkey: signer instanceof PublicKey ? signer : signer.publicKey, isSigner: false, isWritable: false, }); } const data = Buffer.alloc(initializeMultisigInstructionData.span); initializeMultisigInstructionData.encode( { instruction: TokenInstruction.InitializeMultisig, m, }, data, ); return new TransactionInstruction({ keys, programId, data }); } /** A decoded, valid InitializeMultisig instruction */ export interface DecodedInitializeMultisigInstruction { programId: PublicKey; keys: { account: AccountMeta; rent: AccountMeta; signers: AccountMeta[]; }; data: { instruction: TokenInstruction.InitializeMultisig; m: number; }; } /** * Decode an InitializeMultisig instruction and validate it * * @param instruction Transaction instruction to decode * @param programId SPL Token program account * * @return Decoded, valid instruction */ export function decodeInitializeMultisigInstruction( instruction: TransactionInstruction, programId = TOKEN_PROGRAM_ID, ): DecodedInitializeMultisigInstruction { if (!instruction.programId.equals(programId)) throw new TokenInvalidInstructionProgramError(); if (instruction.data.length !== initializeMultisigInstructionData.span) throw new TokenInvalidInstructionDataError(); const { keys: { account, rent, signers }, data, } = decodeInitializeMultisigInstructionUnchecked(instruction); if (data.instruction !== TokenInstruction.InitializeMultisig) throw new TokenInvalidInstructionTypeError(); if (!account || !rent || !signers.length) throw new TokenInvalidInstructionKeysError(); // TODO: key checks? return { programId, keys: { account, rent, signers, }, data, }; } /** A decoded, non-validated InitializeMultisig instruction */ export interface DecodedInitializeMultisigInstructionUnchecked { programId: PublicKey; keys: { account: AccountMeta | undefined; rent: AccountMeta | undefined; signers: AccountMeta[]; }; data: { instruction: number; m: number; }; } /** * Decode an InitializeMultisig instruction without validating it * * @param instruction Transaction instruction to decode * * @return Decoded, non-validated instruction */ export function decodeInitializeMultisigInstructionUnchecked({ programId, keys: [account, rent, ...signers], data, }: TransactionInstruction): DecodedInitializeMultisigInstructionUnchecked { return { programId, keys: { account, rent, signers, }, data: initializeMultisigInstructionData.decode(data), }; }