@solana/spl-token
Version:
SPL Token Program JS API
149 lines (127 loc) • 5.65 kB
text/typescript
import { blob, greedy, seq, struct, u32, u8 } from '@solana/buffer-layout';
import type { Mint } from '../../state/mint.js';
import { ExtensionType, getExtensionData } from '../extensionType.js';
import type { AccountInfo, AccountMeta, Connection } from '@solana/web3.js';
import { PublicKey } from '@solana/web3.js';
import { bool, publicKey, u64 } from '@solana/buffer-layout-utils';
import type { Account } from '../../state/account.js';
import { TokenTransferHookAccountNotFound } from '../../errors.js';
import { unpackSeeds } from './seeds.js';
import { unpackPubkeyData } from './pubkeyData.js';
/** TransferHook as stored by the program */
export interface TransferHook {
/** The transfer hook update authority */
authority: PublicKey;
/** The transfer hook program account */
programId: PublicKey;
}
/** Buffer layout for de/serializing a transfer hook extension */
export const TransferHookLayout = struct<TransferHook>([publicKey('authority'), publicKey('programId')]);
export const TRANSFER_HOOK_SIZE = TransferHookLayout.span;
export function getTransferHook(mint: Mint): TransferHook | null {
const extensionData = getExtensionData(ExtensionType.TransferHook, mint.tlvData);
if (extensionData !== null) {
return TransferHookLayout.decode(extensionData);
} else {
return null;
}
}
/** TransferHookAccount as stored by the program */
export interface TransferHookAccount {
/**
* Whether or not this account is currently transferring tokens
* True during the transfer hook cpi, otherwise false
*/
transferring: boolean;
}
/** Buffer layout for de/serializing a transfer hook account extension */
export const TransferHookAccountLayout = struct<TransferHookAccount>([bool('transferring')]);
export const TRANSFER_HOOK_ACCOUNT_SIZE = TransferHookAccountLayout.span;
export function getTransferHookAccount(account: Account): TransferHookAccount | null {
const extensionData = getExtensionData(ExtensionType.TransferHookAccount, account.tlvData);
if (extensionData !== null) {
return TransferHookAccountLayout.decode(extensionData);
} else {
return null;
}
}
export function getExtraAccountMetaAddress(mint: PublicKey, programId: PublicKey): PublicKey {
const seeds = [Buffer.from('extra-account-metas'), mint.toBuffer()];
return PublicKey.findProgramAddressSync(seeds, programId)[0];
}
/** ExtraAccountMeta as stored by the transfer hook program */
export interface ExtraAccountMeta {
discriminator: number;
addressConfig: Uint8Array;
isSigner: boolean;
isWritable: boolean;
}
/** Buffer layout for de/serializing an ExtraAccountMeta */
export const ExtraAccountMetaLayout = struct<ExtraAccountMeta>([
u8('discriminator'),
blob(32, 'addressConfig'),
bool('isSigner'),
bool('isWritable'),
]);
export interface ExtraAccountMetaList {
count: number;
extraAccounts: ExtraAccountMeta[];
}
/** Buffer layout for de/serializing a list of ExtraAccountMeta prefixed by a u32 length */
export const ExtraAccountMetaListLayout = struct<ExtraAccountMetaList>([
u32('count'),
seq<ExtraAccountMeta>(ExtraAccountMetaLayout, greedy(ExtraAccountMetaLayout.span), 'extraAccounts'),
]);
/** Buffer layout for de/serializing a list of ExtraAccountMetaAccountData prefixed by a u32 length */
export interface ExtraAccountMetaAccountData {
instructionDiscriminator: bigint;
length: number;
extraAccountsList: ExtraAccountMetaList;
}
/** Buffer layout for de/serializing an ExtraAccountMetaAccountData */
export const ExtraAccountMetaAccountDataLayout = struct<ExtraAccountMetaAccountData>([
u64('instructionDiscriminator'),
u32('length'),
ExtraAccountMetaListLayout.replicate('extraAccountsList'),
]);
/** Unpack an extra account metas account and parse the data into a list of ExtraAccountMetas */
export function getExtraAccountMetas(account: AccountInfo<Buffer>): ExtraAccountMeta[] {
const extraAccountsList = ExtraAccountMetaAccountDataLayout.decode(account.data).extraAccountsList;
return extraAccountsList.extraAccounts.slice(0, extraAccountsList.count);
}
/** Take an ExtraAccountMeta and construct that into an actual AccountMeta */
export async function resolveExtraAccountMeta(
connection: Connection,
extraMeta: ExtraAccountMeta,
previousMetas: AccountMeta[],
instructionData: Buffer,
transferHookProgramId: PublicKey,
): Promise<AccountMeta> {
if (extraMeta.discriminator === 0) {
return {
pubkey: new PublicKey(extraMeta.addressConfig),
isSigner: extraMeta.isSigner,
isWritable: extraMeta.isWritable,
};
} else if (extraMeta.discriminator === 2) {
const pubkey = await unpackPubkeyData(extraMeta.addressConfig, previousMetas, instructionData, connection);
return {
pubkey,
isSigner: extraMeta.isSigner,
isWritable: extraMeta.isWritable,
};
}
let programId = PublicKey.default;
if (extraMeta.discriminator === 1) {
programId = transferHookProgramId;
} else {
const accountIndex = extraMeta.discriminator - (1 << 7);
if (previousMetas.length <= accountIndex) {
throw new TokenTransferHookAccountNotFound();
}
programId = previousMetas[accountIndex].pubkey;
}
const seeds = await unpackSeeds(extraMeta.addressConfig, previousMetas, instructionData, connection);
const pubkey = PublicKey.findProgramAddressSync(seeds, programId)[0];
return { pubkey, isSigner: extraMeta.isSigner, isWritable: extraMeta.isWritable };
}