mirror of
https://github.com/d0zingcat/ore.git
synced 2026-05-14 07:26:51 +00:00
279 lines
7.2 KiB
Rust
279 lines
7.2 KiB
Rust
use solana_program::{
|
|
account_info::AccountInfo, program_error::ProgramError, program_pack::Pack, pubkey::Pubkey,
|
|
system_program,
|
|
};
|
|
use spl_token::state::Mint;
|
|
|
|
use crate::{
|
|
state::{Bus, Proof},
|
|
utils::AccountDeserialize,
|
|
BUS_ADDRESSES, BUS_COUNT, MINT_ADDRESS, TREASURY_ADDRESS,
|
|
};
|
|
|
|
/// Errors if:
|
|
/// - Account is not a signer.
|
|
pub fn load_signer<'a, 'info>(info: &'a AccountInfo<'info>) -> Result<(), ProgramError> {
|
|
if !info.is_signer {
|
|
return Err(ProgramError::MissingRequiredSignature);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Account is not owned by Ore program.
|
|
/// - Data is empty.
|
|
/// - Data cannot deserialize into a bus account.
|
|
/// - Bus ID is not in 0-7 range.
|
|
/// - Address is not in set of valid bus address.
|
|
/// - Expected to be writable, but is not.
|
|
pub fn load_bus<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
is_writable: bool,
|
|
) -> Result<(), ProgramError> {
|
|
if info.owner.ne(&crate::id()) {
|
|
return Err(ProgramError::InvalidAccountOwner);
|
|
}
|
|
|
|
if info.data_is_empty() {
|
|
return Err(ProgramError::UninitializedAccount);
|
|
}
|
|
|
|
let bus_data = info.data.borrow();
|
|
let bus = Bus::try_from_bytes(&bus_data)?;
|
|
|
|
if bus.id.ge(&(BUS_COUNT as u64)) {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
if info.key.ne(&BUS_ADDRESSES[bus.id as usize]) {
|
|
return Err(ProgramError::InvalidSeeds);
|
|
}
|
|
|
|
if is_writable && !info.is_writable {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Account is not owned by Ore program.
|
|
/// - Data is empty.
|
|
/// - Data cannot deserialize into a proof account.
|
|
/// - Proof authority does not match the expected address.
|
|
/// - Expected to be writable, but is not.
|
|
pub fn load_proof<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
authority: &Pubkey,
|
|
is_writable: bool,
|
|
) -> Result<(), ProgramError> {
|
|
if info.owner.ne(&crate::id()) {
|
|
return Err(ProgramError::InvalidAccountOwner);
|
|
}
|
|
|
|
if info.data_is_empty() {
|
|
return Err(ProgramError::UninitializedAccount);
|
|
}
|
|
|
|
let proof_data = info.data.borrow();
|
|
let proof = Proof::try_from_bytes(&proof_data)?;
|
|
|
|
if proof.authority.ne(&authority) {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
if is_writable && !info.is_writable {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Account is not owned by Ore program.
|
|
/// - Data is empty.
|
|
/// - Data cannot deserialize into a treasury account.
|
|
/// - Address does not match the expected address.
|
|
/// - Expected to be writable, but is not.
|
|
pub fn load_treasury<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
is_writable: bool,
|
|
) -> Result<(), ProgramError> {
|
|
if info.owner.ne(&crate::id()) {
|
|
return Err(ProgramError::InvalidAccountOwner);
|
|
}
|
|
|
|
if info.data_is_empty() {
|
|
return Err(ProgramError::UninitializedAccount);
|
|
}
|
|
|
|
if info.key.ne(&TREASURY_ADDRESS) {
|
|
return Err(ProgramError::InvalidSeeds);
|
|
}
|
|
|
|
if is_writable && !info.is_writable {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Account is not owned by SPL token program.
|
|
/// - Data is empty.
|
|
/// - Data cannot deserialize into a mint account.
|
|
/// - Address does not match the expected mint address.
|
|
/// - Expected to be writable, but is not.
|
|
pub fn load_mint<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
is_writable: bool,
|
|
) -> Result<(), ProgramError> {
|
|
if info.owner.ne(&spl_token::id()) {
|
|
return Err(ProgramError::InvalidAccountOwner);
|
|
}
|
|
|
|
if info.data_is_empty() {
|
|
return Err(ProgramError::UninitializedAccount);
|
|
}
|
|
|
|
let mint_data = info.data.borrow();
|
|
if Mint::unpack_unchecked(&mint_data).is_err() {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
if info.key.ne(&MINT_ADDRESS) {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
if is_writable && !info.is_writable {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Account is not owned by SPL token program.
|
|
/// - Data is empty.
|
|
/// - Data cannot deserialize into a token account.
|
|
/// - Token account owner does not match the expected owner address.
|
|
/// - Token account mint does not match the expected mint address.
|
|
/// - Expected to be writable, but is not.
|
|
pub fn load_token_account<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
owner: Option<&Pubkey>,
|
|
mint: &Pubkey,
|
|
is_writable: bool,
|
|
) -> Result<(), ProgramError> {
|
|
if info.owner.ne(&spl_token::id()) {
|
|
return Err(ProgramError::InvalidAccountOwner);
|
|
}
|
|
|
|
if info.data_is_empty() {
|
|
return Err(ProgramError::UninitializedAccount);
|
|
}
|
|
|
|
let account_data = info.data.borrow();
|
|
let account = spl_token::state::Account::unpack_unchecked(&account_data)
|
|
.or(Err(ProgramError::InvalidAccountData))?;
|
|
|
|
if account.mint.ne(&mint) {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
if let Some(owner) = owner {
|
|
if account.owner.ne(owner) {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
}
|
|
|
|
if is_writable && !info.is_writable {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Address does not match PDA derived from provided seeds.
|
|
/// - Cannot load as an uninitialized account.
|
|
pub fn load_uninitialized_pda<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
seeds: &[&[u8]],
|
|
program_id: &Pubkey,
|
|
) -> Result<(), ProgramError> {
|
|
let key = Pubkey::create_program_address(seeds, program_id)?;
|
|
if info.key.ne(&key) {
|
|
return Err(ProgramError::InvalidSeeds);
|
|
}
|
|
load_uninitialized_account(info)
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Account is not owned by the system program.
|
|
/// - Data is not empty.
|
|
/// - Account is not writable.
|
|
pub fn load_uninitialized_account<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
) -> Result<(), ProgramError> {
|
|
if info.owner.ne(&system_program::id()) {
|
|
return Err(ProgramError::AccountAlreadyInitialized);
|
|
}
|
|
|
|
if !info.data_is_empty() {
|
|
return Err(ProgramError::AccountAlreadyInitialized);
|
|
}
|
|
|
|
if !info.is_writable {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Account cannot load with the expected address.
|
|
pub fn load_sysvar<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
key: Pubkey,
|
|
) -> Result<(), ProgramError> {
|
|
load_account(info, key, false)
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Account does not match the expected value.
|
|
/// - Expected to be writable, but is not.
|
|
pub fn load_account<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
key: Pubkey,
|
|
is_writable: bool,
|
|
) -> Result<(), ProgramError> {
|
|
if info.key.ne(&key) {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
if is_writable && !info.is_writable {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
/// Errors if:
|
|
/// - Address does not match the expected value.
|
|
/// - Account is not executable.
|
|
pub fn load_program<'a, 'info>(
|
|
info: &'a AccountInfo<'info>,
|
|
key: Pubkey,
|
|
) -> Result<(), ProgramError> {
|
|
if info.key.ne(&key) {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
if !info.executable {
|
|
return Err(ProgramError::InvalidAccountData);
|
|
}
|
|
|
|
Ok(())
|
|
}
|