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> {
let is_unit_type = T::id() == Pubkey::default();
if !is_unit_type && 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<'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<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
}
}
impl crate::Id for () {
fn id() -> Pubkey {
Pubkey::default()
}
}