2026-03-17 18:08:53 +01:00
|
|
|
use nssa_core::{
|
|
|
|
|
account::AccountWithMetadata,
|
|
|
|
|
program::{AccountPostState, ChainedCall, ProgramId},
|
|
|
|
|
};
|
|
|
|
|
use token_core::TokenHolding;
|
|
|
|
|
|
|
|
|
|
pub fn burn_from_associated_token_account(
|
|
|
|
|
owner: AccountWithMetadata,
|
|
|
|
|
holder_ata: AccountWithMetadata,
|
|
|
|
|
token_definition: AccountWithMetadata,
|
|
|
|
|
ata_program_id: ProgramId,
|
2026-05-13 17:24:11 -03:00
|
|
|
token_program_id: ProgramId,
|
2026-03-17 18:08:53 +01:00
|
|
|
amount: u128,
|
|
|
|
|
) -> (Vec<AccountPostState>, Vec<ChainedCall>) {
|
|
|
|
|
assert!(owner.is_authorized, "Owner authorization is missing");
|
2026-05-13 17:24:11 -03:00
|
|
|
assert_eq!(
|
|
|
|
|
holder_ata.account.program_owner, token_program_id,
|
|
|
|
|
"Holder ATA must be owned by expected token program"
|
|
|
|
|
);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
token_definition.account.program_owner, token_program_id,
|
|
|
|
|
"Token definition must be owned by expected token program"
|
|
|
|
|
);
|
2026-03-17 18:08:53 +01:00
|
|
|
let definition_id = TokenHolding::try_from(&holder_ata.account.data)
|
|
|
|
|
.expect("Holder ATA must hold a valid token")
|
|
|
|
|
.definition_id();
|
2026-05-13 17:24:11 -03:00
|
|
|
assert_eq!(
|
|
|
|
|
definition_id, token_definition.account_id,
|
|
|
|
|
"Holder ATA token definition does not match"
|
|
|
|
|
);
|
|
|
|
|
let seed = ata_core::verify_ata_and_get_seed(
|
|
|
|
|
&holder_ata,
|
|
|
|
|
&owner,
|
|
|
|
|
token_program_id,
|
|
|
|
|
definition_id,
|
|
|
|
|
ata_program_id,
|
|
|
|
|
);
|
2026-03-17 18:08:53 +01:00
|
|
|
|
|
|
|
|
let post_states = vec![
|
|
|
|
|
AccountPostState::new(owner.account.clone()),
|
|
|
|
|
AccountPostState::new(holder_ata.account.clone()),
|
|
|
|
|
AccountPostState::new(token_definition.account.clone()),
|
|
|
|
|
];
|
|
|
|
|
let mut holder_ata_auth = holder_ata.clone();
|
|
|
|
|
holder_ata_auth.is_authorized = true;
|
|
|
|
|
|
|
|
|
|
let chained_call = ChainedCall::new(
|
|
|
|
|
token_program_id,
|
|
|
|
|
vec![token_definition.clone(), holder_ata_auth],
|
|
|
|
|
&token_core::Instruction::Burn {
|
|
|
|
|
amount_to_burn: amount,
|
|
|
|
|
},
|
|
|
|
|
)
|
|
|
|
|
.with_pda_seeds(vec![seed]);
|
|
|
|
|
(post_states, vec![chained_call])
|
|
|
|
|
}
|