solana-program-library/record/program/src/processor.rs

122 lines
4.8 KiB
Rust

//! Program state processor
use {
crate::{
error::RecordError,
instruction::RecordInstruction,
state::{Data, RecordData},
},
borsh::{BorshDeserialize, BorshSerialize},
solana_program::{
account_info::{next_account_info, AccountInfo},
entrypoint::ProgramResult,
msg,
program_error::ProgramError,
program_pack::IsInitialized,
pubkey::Pubkey,
},
};
fn check_authority(authority_info: &AccountInfo, expected_authority: &Pubkey) -> ProgramResult {
if expected_authority != authority_info.key {
msg!("Incorrect record authority provided");
return Err(RecordError::IncorrectAuthority.into());
}
if !authority_info.is_signer {
msg!("Record authority signature missing");
return Err(ProgramError::MissingRequiredSignature);
}
Ok(())
}
/// Instruction processor
pub fn process_instruction(
_program_id: &Pubkey,
accounts: &[AccountInfo],
input: &[u8],
) -> ProgramResult {
let instruction = RecordInstruction::try_from_slice(input)?;
let account_info_iter = &mut accounts.iter();
match instruction {
RecordInstruction::Initialize => {
msg!("RecordInstruction::Initialize");
let data_info = next_account_info(account_info_iter)?;
let authority_info = next_account_info(account_info_iter)?;
let mut account_data = RecordData::try_from_slice(*data_info.data.borrow())?;
if account_data.is_initialized() {
msg!("Record account already initialized");
return Err(ProgramError::AccountAlreadyInitialized);
}
account_data.authority = *authority_info.key;
account_data.version = RecordData::CURRENT_VERSION;
account_data
.serialize(&mut *data_info.data.borrow_mut())
.map_err(|e| e.into())
}
RecordInstruction::Write { offset, data } => {
msg!("RecordInstruction::Write");
let data_info = next_account_info(account_info_iter)?;
let authority_info = next_account_info(account_info_iter)?;
let account_data = RecordData::try_from_slice(&data_info.data.borrow())?;
if !account_data.is_initialized() {
msg!("Record account not initialized");
return Err(ProgramError::UninitializedAccount);
}
check_authority(authority_info, &account_data.authority)?;
let start = RecordData::WRITABLE_START_INDEX + offset as usize;
let end = start + data.len();
if end > data_info.data.borrow().len() {
Err(ProgramError::AccountDataTooSmall)
} else {
data_info.data.borrow_mut()[start..end].copy_from_slice(&data);
Ok(())
}
}
RecordInstruction::SetAuthority => {
msg!("RecordInstruction::SetAuthority");
let data_info = next_account_info(account_info_iter)?;
let authority_info = next_account_info(account_info_iter)?;
let new_authority_info = next_account_info(account_info_iter)?;
let mut account_data = RecordData::try_from_slice(&data_info.data.borrow())?;
if !account_data.is_initialized() {
msg!("Record account not initialized");
return Err(ProgramError::UninitializedAccount);
}
check_authority(authority_info, &account_data.authority)?;
account_data.authority = *new_authority_info.key;
account_data
.serialize(&mut *data_info.data.borrow_mut())
.map_err(|e| e.into())
}
RecordInstruction::CloseAccount => {
msg!("RecordInstruction::CloseAccount");
let data_info = next_account_info(account_info_iter)?;
let authority_info = next_account_info(account_info_iter)?;
let destination_info = next_account_info(account_info_iter)?;
let mut account_data = RecordData::try_from_slice(&data_info.data.borrow())?;
if !account_data.is_initialized() {
msg!("Record not initialized");
return Err(ProgramError::UninitializedAccount);
}
check_authority(authority_info, &account_data.authority)?;
let destination_starting_lamports = destination_info.lamports();
let data_lamports = data_info.lamports();
**data_info.lamports.borrow_mut() = 0;
**destination_info.lamports.borrow_mut() = destination_starting_lamports
.checked_add(data_lamports)
.ok_or(RecordError::Overflow)?;
account_data.data = Data::default();
account_data
.serialize(&mut *data_info.data.borrow_mut())
.map_err(|e| e.into())
}
}
}