281 lines
13 KiB
Rust
281 lines
13 KiB
Rust
//! The [`ProgramError`] type and related definitions.
|
|
|
|
#![allow(clippy::integer_arithmetic)]
|
|
use {
|
|
crate::{decode_error::DecodeError, instruction::InstructionError, msg, pubkey::PubkeyError},
|
|
borsh::maybestd::io::Error as BorshIoError,
|
|
num_traits::{FromPrimitive, ToPrimitive},
|
|
std::convert::TryFrom,
|
|
thiserror::Error,
|
|
};
|
|
|
|
/// Reasons the program may fail
|
|
#[derive(Clone, Debug, Deserialize, Eq, Error, PartialEq, Serialize)]
|
|
pub enum ProgramError {
|
|
/// Allows on-chain programs to implement program-specific error types and see them returned
|
|
/// by the Solana runtime. A program-specific error may be any type that is represented as
|
|
/// or serialized to a u32 integer.
|
|
#[error("Custom program error: {0:#x}")]
|
|
Custom(u32),
|
|
#[error("The arguments provided to a program instruction where invalid")]
|
|
InvalidArgument,
|
|
#[error("An instruction's data contents was invalid")]
|
|
InvalidInstructionData,
|
|
#[error("An account's data contents was invalid")]
|
|
InvalidAccountData,
|
|
#[error("An account's data was too small")]
|
|
AccountDataTooSmall,
|
|
#[error("An account's balance was too small to complete the instruction")]
|
|
InsufficientFunds,
|
|
#[error("The account did not have the expected program id")]
|
|
IncorrectProgramId,
|
|
#[error("A signature was required but not found")]
|
|
MissingRequiredSignature,
|
|
#[error("An initialize instruction was sent to an account that has already been initialized")]
|
|
AccountAlreadyInitialized,
|
|
#[error("An attempt to operate on an account that hasn't been initialized")]
|
|
UninitializedAccount,
|
|
#[error("The instruction expected additional account keys")]
|
|
NotEnoughAccountKeys,
|
|
#[error("Failed to borrow a reference to account data, already borrowed")]
|
|
AccountBorrowFailed,
|
|
#[error("Length of the seed is too long for address generation")]
|
|
MaxSeedLengthExceeded,
|
|
#[error("Provided seeds do not result in a valid address")]
|
|
InvalidSeeds,
|
|
#[error("IO Error: {0}")]
|
|
BorshIoError(String),
|
|
#[error("An account does not have enough lamports to be rent-exempt")]
|
|
AccountNotRentExempt,
|
|
#[error("Unsupported sysvar")]
|
|
UnsupportedSysvar,
|
|
#[error("Provided owner is not allowed")]
|
|
IllegalOwner,
|
|
#[error("Account data allocation exceeded the maximum accounts data size limit")]
|
|
MaxAccountsDataSizeExceeded,
|
|
#[error("Cannot close vote account unless it stopped voting at least one full epoch ago")]
|
|
ActiveVoteAccountClose,
|
|
}
|
|
|
|
pub trait PrintProgramError {
|
|
fn print<E>(&self)
|
|
where
|
|
E: 'static + std::error::Error + DecodeError<E> + PrintProgramError + FromPrimitive;
|
|
}
|
|
|
|
impl PrintProgramError for ProgramError {
|
|
fn print<E>(&self)
|
|
where
|
|
E: 'static + std::error::Error + DecodeError<E> + PrintProgramError + FromPrimitive,
|
|
{
|
|
match self {
|
|
Self::Custom(error) => {
|
|
if let Some(custom_error) = E::decode_custom_error_to_enum(*error) {
|
|
custom_error.print::<E>();
|
|
} else {
|
|
msg!("Error: Unknown");
|
|
}
|
|
}
|
|
Self::InvalidArgument => msg!("Error: InvalidArgument"),
|
|
Self::InvalidInstructionData => msg!("Error: InvalidInstructionData"),
|
|
Self::InvalidAccountData => msg!("Error: InvalidAccountData"),
|
|
Self::AccountDataTooSmall => msg!("Error: AccountDataTooSmall"),
|
|
Self::InsufficientFunds => msg!("Error: InsufficientFunds"),
|
|
Self::IncorrectProgramId => msg!("Error: IncorrectProgramId"),
|
|
Self::MissingRequiredSignature => msg!("Error: MissingRequiredSignature"),
|
|
Self::AccountAlreadyInitialized => msg!("Error: AccountAlreadyInitialized"),
|
|
Self::UninitializedAccount => msg!("Error: UninitializedAccount"),
|
|
Self::NotEnoughAccountKeys => msg!("Error: NotEnoughAccountKeys"),
|
|
Self::AccountBorrowFailed => msg!("Error: AccountBorrowFailed"),
|
|
Self::MaxSeedLengthExceeded => msg!("Error: MaxSeedLengthExceeded"),
|
|
Self::InvalidSeeds => msg!("Error: InvalidSeeds"),
|
|
Self::BorshIoError(_) => msg!("Error: BorshIoError"),
|
|
Self::AccountNotRentExempt => msg!("Error: AccountNotRentExempt"),
|
|
Self::UnsupportedSysvar => msg!("Error: UnsupportedSysvar"),
|
|
Self::IllegalOwner => msg!("Error: IllegalOwner"),
|
|
Self::MaxAccountsDataSizeExceeded => msg!("Error: MaxAccountsDataSizeExceeded"),
|
|
Self::ActiveVoteAccountClose => msg!("Error: ActiveVoteAccountClose"),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Builtin return values occupy the upper 32 bits
|
|
const BUILTIN_BIT_SHIFT: usize = 32;
|
|
macro_rules! to_builtin {
|
|
($error:expr) => {
|
|
($error as u64) << BUILTIN_BIT_SHIFT
|
|
};
|
|
}
|
|
|
|
pub const CUSTOM_ZERO: u64 = to_builtin!(1);
|
|
pub const INVALID_ARGUMENT: u64 = to_builtin!(2);
|
|
pub const INVALID_INSTRUCTION_DATA: u64 = to_builtin!(3);
|
|
pub const INVALID_ACCOUNT_DATA: u64 = to_builtin!(4);
|
|
pub const ACCOUNT_DATA_TOO_SMALL: u64 = to_builtin!(5);
|
|
pub const INSUFFICIENT_FUNDS: u64 = to_builtin!(6);
|
|
pub const INCORRECT_PROGRAM_ID: u64 = to_builtin!(7);
|
|
pub const MISSING_REQUIRED_SIGNATURES: u64 = to_builtin!(8);
|
|
pub const ACCOUNT_ALREADY_INITIALIZED: u64 = to_builtin!(9);
|
|
pub const UNINITIALIZED_ACCOUNT: u64 = to_builtin!(10);
|
|
pub const NOT_ENOUGH_ACCOUNT_KEYS: u64 = to_builtin!(11);
|
|
pub const ACCOUNT_BORROW_FAILED: u64 = to_builtin!(12);
|
|
pub const MAX_SEED_LENGTH_EXCEEDED: u64 = to_builtin!(13);
|
|
pub const INVALID_SEEDS: u64 = to_builtin!(14);
|
|
pub const BORSH_IO_ERROR: u64 = to_builtin!(15);
|
|
pub const ACCOUNT_NOT_RENT_EXEMPT: u64 = to_builtin!(16);
|
|
pub const UNSUPPORTED_SYSVAR: u64 = to_builtin!(17);
|
|
pub const ILLEGAL_OWNER: u64 = to_builtin!(18);
|
|
pub const MAX_ACCOUNTS_DATA_SIZE_EXCEEDED: u64 = to_builtin!(19);
|
|
pub const ACTIVE_VOTE_ACCOUNT_CLOSE: u64 = to_builtin!(20);
|
|
// Warning: Any new program errors added here must also be:
|
|
// - Added to the below conversions
|
|
// - Added as an equivalent to InstructionError
|
|
// - Be featureized in the BPF loader to return `InstructionError::InvalidError`
|
|
// until the feature is activated
|
|
|
|
impl From<ProgramError> for u64 {
|
|
fn from(error: ProgramError) -> Self {
|
|
match error {
|
|
ProgramError::InvalidArgument => INVALID_ARGUMENT,
|
|
ProgramError::InvalidInstructionData => INVALID_INSTRUCTION_DATA,
|
|
ProgramError::InvalidAccountData => INVALID_ACCOUNT_DATA,
|
|
ProgramError::AccountDataTooSmall => ACCOUNT_DATA_TOO_SMALL,
|
|
ProgramError::InsufficientFunds => INSUFFICIENT_FUNDS,
|
|
ProgramError::IncorrectProgramId => INCORRECT_PROGRAM_ID,
|
|
ProgramError::MissingRequiredSignature => MISSING_REQUIRED_SIGNATURES,
|
|
ProgramError::AccountAlreadyInitialized => ACCOUNT_ALREADY_INITIALIZED,
|
|
ProgramError::UninitializedAccount => UNINITIALIZED_ACCOUNT,
|
|
ProgramError::NotEnoughAccountKeys => NOT_ENOUGH_ACCOUNT_KEYS,
|
|
ProgramError::AccountBorrowFailed => ACCOUNT_BORROW_FAILED,
|
|
ProgramError::MaxSeedLengthExceeded => MAX_SEED_LENGTH_EXCEEDED,
|
|
ProgramError::InvalidSeeds => INVALID_SEEDS,
|
|
ProgramError::BorshIoError(_) => BORSH_IO_ERROR,
|
|
ProgramError::AccountNotRentExempt => ACCOUNT_NOT_RENT_EXEMPT,
|
|
ProgramError::UnsupportedSysvar => UNSUPPORTED_SYSVAR,
|
|
ProgramError::IllegalOwner => ILLEGAL_OWNER,
|
|
ProgramError::MaxAccountsDataSizeExceeded => MAX_ACCOUNTS_DATA_SIZE_EXCEEDED,
|
|
ProgramError::ActiveVoteAccountClose => ACTIVE_VOTE_ACCOUNT_CLOSE,
|
|
ProgramError::Custom(error) => {
|
|
if error == 0 {
|
|
CUSTOM_ZERO
|
|
} else {
|
|
error as u64
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<u64> for ProgramError {
|
|
fn from(error: u64) -> Self {
|
|
match error {
|
|
CUSTOM_ZERO => Self::Custom(0),
|
|
INVALID_ARGUMENT => Self::InvalidArgument,
|
|
INVALID_INSTRUCTION_DATA => Self::InvalidInstructionData,
|
|
INVALID_ACCOUNT_DATA => Self::InvalidAccountData,
|
|
ACCOUNT_DATA_TOO_SMALL => Self::AccountDataTooSmall,
|
|
INSUFFICIENT_FUNDS => Self::InsufficientFunds,
|
|
INCORRECT_PROGRAM_ID => Self::IncorrectProgramId,
|
|
MISSING_REQUIRED_SIGNATURES => Self::MissingRequiredSignature,
|
|
ACCOUNT_ALREADY_INITIALIZED => Self::AccountAlreadyInitialized,
|
|
UNINITIALIZED_ACCOUNT => Self::UninitializedAccount,
|
|
NOT_ENOUGH_ACCOUNT_KEYS => Self::NotEnoughAccountKeys,
|
|
ACCOUNT_BORROW_FAILED => Self::AccountBorrowFailed,
|
|
MAX_SEED_LENGTH_EXCEEDED => Self::MaxSeedLengthExceeded,
|
|
INVALID_SEEDS => Self::InvalidSeeds,
|
|
BORSH_IO_ERROR => Self::BorshIoError("Unknown".to_string()),
|
|
ACCOUNT_NOT_RENT_EXEMPT => Self::AccountNotRentExempt,
|
|
UNSUPPORTED_SYSVAR => Self::UnsupportedSysvar,
|
|
ILLEGAL_OWNER => Self::IllegalOwner,
|
|
MAX_ACCOUNTS_DATA_SIZE_EXCEEDED => Self::MaxAccountsDataSizeExceeded,
|
|
ACTIVE_VOTE_ACCOUNT_CLOSE => Self::ActiveVoteAccountClose,
|
|
_ => Self::Custom(error as u32),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl TryFrom<InstructionError> for ProgramError {
|
|
type Error = InstructionError;
|
|
|
|
fn try_from(error: InstructionError) -> Result<Self, Self::Error> {
|
|
match error {
|
|
Self::Error::Custom(err) => Ok(Self::Custom(err)),
|
|
Self::Error::InvalidArgument => Ok(Self::InvalidArgument),
|
|
Self::Error::InvalidInstructionData => Ok(Self::InvalidInstructionData),
|
|
Self::Error::InvalidAccountData => Ok(Self::InvalidAccountData),
|
|
Self::Error::AccountDataTooSmall => Ok(Self::AccountDataTooSmall),
|
|
Self::Error::InsufficientFunds => Ok(Self::InsufficientFunds),
|
|
Self::Error::IncorrectProgramId => Ok(Self::IncorrectProgramId),
|
|
Self::Error::MissingRequiredSignature => Ok(Self::MissingRequiredSignature),
|
|
Self::Error::AccountAlreadyInitialized => Ok(Self::AccountAlreadyInitialized),
|
|
Self::Error::UninitializedAccount => Ok(Self::UninitializedAccount),
|
|
Self::Error::NotEnoughAccountKeys => Ok(Self::NotEnoughAccountKeys),
|
|
Self::Error::AccountBorrowFailed => Ok(Self::AccountBorrowFailed),
|
|
Self::Error::MaxSeedLengthExceeded => Ok(Self::MaxSeedLengthExceeded),
|
|
Self::Error::InvalidSeeds => Ok(Self::InvalidSeeds),
|
|
Self::Error::BorshIoError(err) => Ok(Self::BorshIoError(err)),
|
|
Self::Error::AccountNotRentExempt => Ok(Self::AccountNotRentExempt),
|
|
Self::Error::UnsupportedSysvar => Ok(Self::UnsupportedSysvar),
|
|
Self::Error::IllegalOwner => Ok(Self::IllegalOwner),
|
|
Self::Error::MaxAccountsDataSizeExceeded => Ok(Self::MaxAccountsDataSizeExceeded),
|
|
Self::Error::ActiveVoteAccountClose => Ok(Self::ActiveVoteAccountClose),
|
|
_ => Err(error),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T> From<T> for InstructionError
|
|
where
|
|
T: ToPrimitive,
|
|
{
|
|
fn from(error: T) -> Self {
|
|
let error = error.to_u64().unwrap_or(0xbad_c0de);
|
|
match error {
|
|
CUSTOM_ZERO => Self::Custom(0),
|
|
INVALID_ARGUMENT => Self::InvalidArgument,
|
|
INVALID_INSTRUCTION_DATA => Self::InvalidInstructionData,
|
|
INVALID_ACCOUNT_DATA => Self::InvalidAccountData,
|
|
ACCOUNT_DATA_TOO_SMALL => Self::AccountDataTooSmall,
|
|
INSUFFICIENT_FUNDS => Self::InsufficientFunds,
|
|
INCORRECT_PROGRAM_ID => Self::IncorrectProgramId,
|
|
MISSING_REQUIRED_SIGNATURES => Self::MissingRequiredSignature,
|
|
ACCOUNT_ALREADY_INITIALIZED => Self::AccountAlreadyInitialized,
|
|
UNINITIALIZED_ACCOUNT => Self::UninitializedAccount,
|
|
NOT_ENOUGH_ACCOUNT_KEYS => Self::NotEnoughAccountKeys,
|
|
ACCOUNT_BORROW_FAILED => Self::AccountBorrowFailed,
|
|
MAX_SEED_LENGTH_EXCEEDED => Self::MaxSeedLengthExceeded,
|
|
INVALID_SEEDS => Self::InvalidSeeds,
|
|
BORSH_IO_ERROR => Self::BorshIoError("Unknown".to_string()),
|
|
ACCOUNT_NOT_RENT_EXEMPT => Self::AccountNotRentExempt,
|
|
UNSUPPORTED_SYSVAR => Self::UnsupportedSysvar,
|
|
ILLEGAL_OWNER => Self::IllegalOwner,
|
|
MAX_ACCOUNTS_DATA_SIZE_EXCEEDED => Self::MaxAccountsDataSizeExceeded,
|
|
ACTIVE_VOTE_ACCOUNT_CLOSE => Self::ActiveVoteAccountClose,
|
|
_ => {
|
|
// A valid custom error has no bits set in the upper 32
|
|
if error >> BUILTIN_BIT_SHIFT == 0 {
|
|
Self::Custom(error as u32)
|
|
} else {
|
|
Self::InvalidError
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<PubkeyError> for ProgramError {
|
|
fn from(error: PubkeyError) -> Self {
|
|
match error {
|
|
PubkeyError::MaxSeedLengthExceeded => Self::MaxSeedLengthExceeded,
|
|
PubkeyError::InvalidSeeds => Self::InvalidSeeds,
|
|
PubkeyError::IllegalOwner => Self::IllegalOwner,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<BorshIoError> for ProgramError {
|
|
fn from(error: BorshIoError) -> Self {
|
|
Self::BorshIoError(format!("{}", error))
|
|
}
|
|
}
|