lssa/nssa/src/program.rs

164 lines
4.7 KiB
Rust
Raw Normal View History

2025-08-09 20:25:58 -03:00
use nssa_core::{
account::{Account, AccountWithMetadata},
2025-08-10 18:51:55 -03:00
program::{DEFAULT_PROGRAM_ID, InstructionData, ProgramId},
2025-08-09 20:25:58 -03:00
};
use program_methods::{AUTHENTICATED_TRANSFER_ELF, AUTHENTICATED_TRANSFER_ID};
2025-08-10 18:51:55 -03:00
use risc0_zkvm::{ExecutorEnv, ExecutorEnvBuilder, default_executor, serde::to_vec};
2025-08-10 18:59:29 -03:00
use serde::Serialize;
2025-08-09 20:25:58 -03:00
use crate::error::NssaError;
pub struct Program {
2025-08-10 00:53:53 -03:00
pub(crate) id: ProgramId,
pub(crate) elf: &'static [u8],
2025-08-09 20:25:58 -03:00
}
impl Program {
pub fn id(&self) -> ProgramId {
self.id
}
2025-08-09 20:35:44 -03:00
2025-08-10 19:53:05 -03:00
pub fn serialize_instruction<T: Serialize>(
2025-08-10 18:51:55 -03:00
instruction_data: T,
) -> Result<InstructionData, NssaError> {
to_vec(&instruction_data)
.map_err(|e| NssaError::InstructionDataSerializationError(e.to_string()))
}
2025-08-09 20:25:58 -03:00
pub(crate) fn execute(
&self,
pre_states: &[AccountWithMetadata],
2025-08-10 18:51:55 -03:00
instruction_data: &InstructionData,
2025-08-09 20:25:58 -03:00
) -> Result<Vec<Account>, NssaError> {
// Write inputs to the program
let mut env_builder = ExecutorEnv::builder();
Self::write_inputs(pre_states, instruction_data, &mut env_builder)?;
let env = env_builder.build().unwrap();
// Execute the program (without proving)
let executor = default_executor();
let session_info = executor
.execute(env, self.elf)
.map_err(|e| NssaError::ProgramExecutionFailed(e.to_string()))?;
// Get outputs
let mut post_states: Vec<Account> = session_info
.journal
.decode()
.map_err(|e| NssaError::ProgramExecutionFailed(e.to_string()))?;
// Claim any output account with default program owner field
for account in post_states.iter_mut() {
if account.program_owner == DEFAULT_PROGRAM_ID {
account.program_owner = self.id;
}
}
Ok(post_states)
}
/// Writes inputs to `env_builder` in the order expected by the programs
fn write_inputs(
pre_states: &[AccountWithMetadata],
2025-08-10 18:51:55 -03:00
instruction_data: &[u32],
2025-08-09 20:25:58 -03:00
env_builder: &mut ExecutorEnvBuilder,
) -> Result<(), NssaError> {
let pre_states = pre_states.to_vec();
env_builder
2025-08-10 18:51:55 -03:00
.write(&(pre_states, instruction_data))
2025-08-09 20:25:58 -03:00
.map_err(|e| NssaError::ProgramExecutionFailed(e.to_string()))?;
Ok(())
}
2025-08-10 00:53:53 -03:00
pub fn authenticated_transfer_program() -> Self {
Self {
id: AUTHENTICATED_TRANSFER_ID,
elf: AUTHENTICATED_TRANSFER_ELF,
}
}
}
// Test utils
#[cfg(test)]
impl Program {
/// A program that changes the nonce of an account
pub fn nonce_changer_program() -> Self {
use test_program_methods::{NONCE_CHANGER_ELF, NONCE_CHANGER_ID};
Program {
id: NONCE_CHANGER_ID,
elf: NONCE_CHANGER_ELF,
}
}
2025-08-10 09:57:10 -03:00
/// A program that produces more output accounts than the inputs it received
2025-08-10 10:09:23 -03:00
pub fn extra_output_program() -> Self {
use test_program_methods::{EXTRA_OUTPUT_ELF, EXTRA_OUTPUT_ID};
2025-08-10 09:57:10 -03:00
Program {
2025-08-10 10:09:23 -03:00
id: EXTRA_OUTPUT_ID,
elf: EXTRA_OUTPUT_ELF,
}
}
/// A program that produces less output accounts than the inputs it received
pub fn missing_output_program() -> Self {
use test_program_methods::{MISSING_OUTPUT_ELF, MISSING_OUTPUT_ID};
Program {
id: MISSING_OUTPUT_ID,
elf: MISSING_OUTPUT_ELF,
2025-08-10 09:57:10 -03:00
}
}
2025-08-10 10:19:59 -03:00
/// A program that changes the program owner of an account to [0, 1, 2, 3, 4, 5, 6, 7]
pub fn program_owner_changer() -> Self {
use test_program_methods::{PROGRAM_OWNER_CHANGER_ELF, PROGRAM_OWNER_CHANGER_ID};
Program {
id: PROGRAM_OWNER_CHANGER_ID,
elf: PROGRAM_OWNER_CHANGER_ELF,
}
}
2025-08-10 11:02:59 -03:00
/// A program that transfers balance without caring about authorizations
pub fn simple_balance_transfer() -> Self {
use test_program_methods::{SIMPLE_BALANCE_TRANSFER_ELF, SIMPLE_BALANCE_TRANSFER_ID};
Program {
id: SIMPLE_BALANCE_TRANSFER_ID,
elf: SIMPLE_BALANCE_TRANSFER_ELF,
}
}
2025-08-10 11:17:15 -03:00
/// A program that modifies the data of an account
pub fn data_changer() -> Self {
use test_program_methods::{DATA_CHANGER_ELF, DATA_CHANGER_ID};
Program {
id: DATA_CHANGER_ID,
elf: DATA_CHANGER_ELF,
}
}
/// A program that mints balance
pub fn minter() -> Self {
use test_program_methods::{MINTER_ELF, MINTER_ID};
Program {
id: MINTER_ID,
elf: MINTER_ELF,
}
}
2025-08-10 14:33:40 -03:00
/// A program that burns balance
pub fn burner() -> Self {
use test_program_methods::{BURNER_ELF, BURNER_ID};
Program {
id: BURNER_ID,
elf: BURNER_ELF,
}
}
2025-08-09 20:25:58 -03:00
}