use crate::{write_bytes, UNINIT_BYTE};
use core::slice::from_raw_parts;
use hayabusa_cpi::{CpiCtx, CheckProgramId};
use hayabusa_errors::Result;
use pinocchio::{account_info::AccountInfo, cpi::{invoke, invoke_signed}, instruction::{AccountMeta, Instruction}, pubkey::Pubkey};
pub struct MintTo<'a> {
pub mint: &'a AccountInfo,
pub destination: &'a AccountInfo,
pub authority: &'a AccountInfo,
}
impl CheckProgramId for MintTo<'_> {
const ID: Pubkey = crate::ID;
}
const DISCRIMINATOR: [u8; 1] = [7];
#[inline(always)]
pub fn mint_to<'a>(
cpi_ctx: CpiCtx<'a, '_, '_, '_, MintTo<'a>>,
amount: u64,
) -> Result<()> {
let infos = [cpi_ctx.mint, cpi_ctx.destination, cpi_ctx.authority];
let metas = [
AccountMeta::writable(cpi_ctx.mint.key()),
AccountMeta::writable(cpi_ctx.destination.key()),
AccountMeta::readonly_signer(cpi_ctx.authority.key()),
];
let mut ix_data = [UNINIT_BYTE; 9];
write_bytes(&mut ix_data, &DISCRIMINATOR);
write_bytes(&mut ix_data[1..9], &amount.to_le_bytes());
let ix = Instruction {
program_id: &crate::ID,
accounts: &metas,
data: unsafe { from_raw_parts(ix_data.as_ptr() as _, 9) }
};
if let Some(signers) = cpi_ctx.signers {
invoke_signed(&ix, &infos, signers)
} else {
invoke(&ix, &infos)
}
}