use {
crate::{
error::{Error, ErrorCode},
solana_program::{
account_info::AccountInfo,
bpf_loader_upgradeable::{self, UpgradeableLoaderState},
instruction::AccountMeta,
pubkey::Pubkey,
},
AccountDeserialize, Accounts, AccountsExit, Id, Key, Result, ToAccountInfos,
ToAccountMetas,
},
std::{collections::BTreeSet, fmt, marker::PhantomData, ops::Deref},
};
#[derive(Clone)]
pub struct Program<'info, T = ()> {
info: &'info AccountInfo<'info>,
_phantom: PhantomData<T>,
}
impl<T: fmt::Debug> fmt::Debug for Program<'_, T> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("Program").field("info", &self.info).finish()
}
}
impl<'a, T> Program<'a, T> {
pub(crate) fn new(info: &'a AccountInfo<'a>) -> Program<'a, T> {
Self {
info,
_phantom: PhantomData,
}
}
pub fn programdata_address(&self) -> Result<Option<Pubkey>> {
if *self.info.owner == bpf_loader_upgradeable::ID {
let mut data: &[u8] = &self.info.try_borrow_data()?;
let upgradable_loader_state =
UpgradeableLoaderState::try_deserialize_unchecked(&mut data)?;
match upgradable_loader_state {
UpgradeableLoaderState::Uninitialized
| UpgradeableLoaderState::Buffer {
authority_address: _,
}
| UpgradeableLoaderState::ProgramData {
slot: _,
upgrade_authority_address: _,
} => {
unreachable!()
}
UpgradeableLoaderState::Program {
programdata_address,
} => Ok(Some(programdata_address)),
}
} else {
Ok(None)
}
}
}
impl<'a, T: Id> TryFrom<&'a AccountInfo<'a>> for Program<'a, T> {
type Error = Error;
fn try_from(info: &'a AccountInfo<'a>) -> Result<Self> {
if info.key != &T::id() {
return Err(Error::from(ErrorCode::InvalidProgramId).with_pubkeys((*info.key, T::id())));
}
if !info.executable {
return Err(ErrorCode::InvalidProgramExecutable.into());
}
Ok(Program::new(info))
}
}
impl<'a> TryFrom<&'a AccountInfo<'a>> for Program<'a, ()> {
type Error = Error;
fn try_from(info: &'a AccountInfo<'a>) -> Result<Self> {
if !info.executable {
return Err(ErrorCode::InvalidProgramExecutable.into());
}
Ok(Program::new(info))
}
}
impl<'info, B, T: Id> Accounts<'info, B> for Program<'info, T> {
#[inline(never)]
fn try_accounts(
_program_id: &Pubkey,
accounts: &mut &'info [AccountInfo<'info>],
_ix_data: &[u8],
_bumps: &mut B,
_reallocs: &mut BTreeSet<Pubkey>,
) -> Result<Self> {
if accounts.is_empty() {
return Err(ErrorCode::AccountNotEnoughKeys.into());
}
let account = &accounts[0];
*accounts = &accounts[1..];
Program::try_from(account)
}
}
impl<'info, B> Accounts<'info, B> for Program<'info, ()> {
#[inline(never)]
fn try_accounts(
_program_id: &Pubkey,
accounts: &mut &'info [AccountInfo<'info>],
_ix_data: &[u8],
_bumps: &mut B,
_reallocs: &mut BTreeSet<Pubkey>,
) -> Result<Self> {
if accounts.is_empty() {
return Err(ErrorCode::AccountNotEnoughKeys.into());
}
let account = &accounts[0];
*accounts = &accounts[1..];
Program::try_from(account)
}
}
impl<T> ToAccountMetas for Program<'_, T> {
fn to_account_metas(&self, is_signer: Option<bool>) -> Vec<AccountMeta> {
let is_signer = is_signer.unwrap_or(self.info.is_signer);
let meta = match self.info.is_writable {
false => AccountMeta::new_readonly(*self.info.key, is_signer),
true => AccountMeta::new(*self.info.key, is_signer),
};
vec![meta]
}
}
impl<'info, T> ToAccountInfos<'info> for Program<'info, T> {
fn to_account_infos(&self) -> Vec<AccountInfo<'info>> {
vec![self.info.clone()]
}
}
impl<'info, T> AsRef<AccountInfo<'info>> for Program<'info, T> {
fn as_ref(&self) -> &AccountInfo<'info> {
self.info
}
}
impl<'info, T> Deref for Program<'info, T> {
type Target = AccountInfo<'info>;
fn deref(&self) -> &Self::Target {
self.info
}
}
impl<'info, T: AccountDeserialize> AccountsExit<'info> for Program<'info, T> {}
impl<T: AccountDeserialize> Key for Program<'_, T> {
fn key(&self) -> Pubkey {
*self.info.key
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::system_program::System;
fn account_info<'a>(
key: &'a Pubkey,
owner: &'a Pubkey,
lamports: &'a mut u64,
data: &'a mut [u8],
executable: bool,
) -> AccountInfo<'a> {
AccountInfo::new(key, false, false, lamports, data, owner, executable)
}
#[test]
fn program_system_rejects_non_system_account() {
let wrong_key = Pubkey::new_from_array([7u8; 32]);
let owner = Pubkey::default();
let mut lamports = 0u64;
let mut data = [];
let info = account_info(&wrong_key, &owner, &mut lamports, &mut data, true);
let err = Program::<System>::try_from(&info)
.err()
.expect("non-system key must be rejected");
match err {
Error::AnchorError(e) => assert_eq!(
e.error_code_number,
ErrorCode::InvalidProgramId as u32,
),
other => panic!("unexpected error variant: {other:?}"),
}
}
#[test]
fn program_system_accepts_system_account() {
let key = System::id();
let owner = Pubkey::default();
let mut lamports = 0u64;
let mut data = [];
let info = account_info(&key, &owner, &mut lamports, &mut data, true);
Program::<System>::try_from(&info).expect("system program account must be accepted");
}
#[test]
fn program_system_rejects_non_executable_system_account() {
let key = System::id();
let owner = Pubkey::default();
let mut lamports = 0u64;
let mut data = [];
let info = account_info(&key, &owner, &mut lamports, &mut data, false);
let err = Program::<System>::try_from(&info)
.err()
.expect("non-executable account must be rejected");
match err {
Error::AnchorError(e) => assert_eq!(
e.error_code_number,
ErrorCode::InvalidProgramExecutable as u32,
),
other => panic!("unexpected error variant: {other:?}"),
}
}
#[test]
fn program_unit_accepts_arbitrary_executable() {
let key = Pubkey::new_from_array([7u8; 32]);
let owner = Pubkey::default();
let mut lamports = 0u64;
let mut data = [];
let info = account_info(&key, &owner, &mut lamports, &mut data, true);
Program::<()>::try_from(&info).expect("any executable account must be accepted");
}
}