lssa/nssa/program_methods/guest/src/bin/authenticated_transfer.rs

91 lines
2.8 KiB
Rust
Raw Normal View History

2025-10-10 17:47:23 -03:00
use nssa_core::{
account::{Account, AccountWithMetadata},
2025-12-03 16:39:33 -03:00
program::{
AccountPostState, DEFAULT_PROGRAM_ID, ProgramInput, read_nssa_inputs, write_nssa_outputs,
},
2025-10-10 17:47:23 -03:00
};
2025-08-10 18:59:29 -03:00
2025-10-13 18:06:02 -03:00
/// Initializes a default account under the ownership of this program.
fn initialize_account(pre_state: AccountWithMetadata) -> AccountPostState {
2025-12-03 16:39:33 -03:00
let account_to_claim = AccountPostState::new_claimed(pre_state.account.clone());
2025-10-10 17:47:23 -03:00
let is_authorized = pre_state.is_authorized;
// Continue only if the account to claim has default values
2025-12-04 16:26:40 -03:00
if account_to_claim.account() != &Account::default() {
panic!("Account must be uninitialized");
2025-10-10 17:47:23 -03:00
}
// Continue only if the owner authorized this operation
if !is_authorized {
2025-11-18 01:38:47 -03:00
panic!("Invalid input");
2025-10-10 17:47:23 -03:00
}
account_to_claim
2025-10-10 17:47:23 -03:00
}
2025-08-06 20:05:04 -03:00
2025-10-13 18:06:02 -03:00
/// Transfers `balance_to_move` native balance from `sender` to `recipient`.
2025-11-18 01:38:47 -03:00
fn transfer(
sender: AccountWithMetadata,
recipient: AccountWithMetadata,
balance_to_move: u128,
) -> Vec<AccountPostState> {
// Continue only if the sender has authorized this operation
if !sender.is_authorized {
2025-11-18 01:38:47 -03:00
panic!("Invalid input");
}
2025-08-06 20:05:04 -03:00
// Continue only if the sender has enough balance
if sender.account.balance < balance_to_move {
2025-11-18 01:38:47 -03:00
panic!("Invalid input");
}
2025-08-06 20:05:04 -03:00
// Create accounts post states, with updated balances
2025-12-04 16:46:43 -03:00
let sender_post = {
2025-12-03 16:39:33 -03:00
// Modify sender's balance
let mut sender_post_account = sender.account.clone();
sender_post_account.balance -= balance_to_move;
AccountPostState::new(sender_post_account)
};
2025-08-06 20:05:04 -03:00
2025-12-03 16:39:33 -03:00
let recipient_post = {
// Modify recipient's balance
let mut recipient_post_account = recipient.account.clone();
recipient_post_account.balance += balance_to_move;
// Claim recipient account if it has default program owner
if recipient_post_account.program_owner == DEFAULT_PROGRAM_ID {
AccountPostState::new_claimed(recipient_post_account)
} else {
AccountPostState::new(recipient_post_account)
}
};
vec![sender_post, recipient_post]
2025-08-06 20:05:04 -03:00
}
2025-09-04 12:44:22 -03:00
2025-10-10 17:47:23 -03:00
/// A transfer of balance program.
/// To be used both in public and private contexts.
fn main() {
// Read input accounts.
2025-11-18 01:38:47 -03:00
let (
ProgramInput {
pre_states,
instruction: balance_to_move,
},
instruction_words,
) = read_nssa_inputs();
2025-10-10 17:47:23 -03:00
let post_states = match (pre_states.as_slice(), balance_to_move) {
2025-11-18 01:38:47 -03:00
([account_to_claim], 0) => {
let post = initialize_account(account_to_claim.clone());
vec![post]
2025-11-18 01:38:47 -03:00
}
2025-10-13 17:45:03 -03:00
([sender, recipient], balance_to_move) => {
2025-11-26 17:41:49 -03:00
transfer(sender.clone(), recipient.clone(), balance_to_move)
2025-10-13 17:45:03 -03:00
}
_ => panic!("invalid params"),
2025-11-26 17:41:49 -03:00
};
write_nssa_outputs(instruction_words, pre_states, post_states);
2025-10-10 17:47:23 -03:00
}