use hpsvm::{HPSVM, types::FailedTransactionMetadata};
use solana_address::Address;
use solana_keypair::Keypair;
use solana_signer::Signer;
use solana_transaction::Transaction;
use spl_associated_token_account_interface::instruction::create_associated_token_account_idempotent;
use super::TOKEN_ID;
#[derive(Debug)]
pub struct CreateAssociatedTokenAccountIdempotent<'a> {
svm: &'a mut HPSVM,
payer: &'a Keypair,
mint: &'a Address,
token_program_id: Option<&'a Address>,
owner: Option<Address>,
}
impl<'a> CreateAssociatedTokenAccountIdempotent<'a> {
pub fn new(svm: &'a mut HPSVM, payer: &'a Keypair, mint: &'a Address) -> Self {
CreateAssociatedTokenAccountIdempotent {
svm,
payer,
owner: None,
token_program_id: None,
mint,
}
}
pub fn owner(mut self, owner: &'a Address) -> Self {
self.owner = Some(*owner);
self
}
pub fn token_program_id(mut self, program_id: &'a Address) -> Self {
self.token_program_id = Some(program_id);
self
}
pub fn send(self) -> Result<Address, FailedTransactionMetadata> {
let token_program_id = self.token_program_id.unwrap_or(&TOKEN_ID);
let payer_pk = self.payer.pubkey();
let authority = self.owner.unwrap_or(payer_pk);
let ix = create_associated_token_account_idempotent(
&payer_pk,
&authority,
self.mint,
token_program_id,
);
let block_hash = self.svm.latest_blockhash();
let tx =
Transaction::new_signed_with_payer(&[ix], Some(&payer_pk), &[self.payer], block_hash);
self.svm.send_transaction(tx)?;
let ata = spl_associated_token_account_interface::address::get_associated_token_address_with_program_id(
&authority,
self.mint,
token_program_id,
);
Ok(ata)
}
}