use core::any::Any;
use alloc::{boxed::Box, string::String};
use crate::arithmetic::ArithmeticsError;
const MAX_USER_ERROR: u16 = 32767;
const USER_ERROR_TOO_HIGH: u16 = 32768;
const UNWRAP_ERROR: u16 = 1;
const CODE_ADDITION_OVERFLOW: u16 = 100;
const CODE_SUBTRACTION_OVERFLOW: u16 = 101;
const CODE_NON_PAYABLE: u16 = 102;
const CODE_TRANSFER_TO_CONTRACT: u16 = 103;
const CODE_REENTRANT_CALL: u16 = 104;
const CODE_CONTRACT_ALREADY_INSTALLED: u16 = 105;
const CODE_UNKNOWN_CONSTRUCTOR: u16 = 106;
const CODE_NATIVE_TRANSFER_ERROR: u16 = 107;
const CODE_INDEX_OUT_OF_BOUNDS: u16 = 108;
const CODE_ZERO_ADDRESS: u16 = 109;
const CODE_ADDRESS_CREATION_FAILED: u16 = 110;
#[derive(Clone, Debug, PartialEq)]
pub enum OdraError {
ExecutionError(ExecutionError),
VmError(VmError)
}
impl From<ArithmeticsError> for ExecutionError {
fn from(error: ArithmeticsError) -> Self {
match error {
ArithmeticsError::AdditionOverflow => Self::addition_overflow(),
ArithmeticsError::SubtractingOverflow => Self::subtraction_overflow()
}
}
}
impl From<ArithmeticsError> for OdraError {
fn from(error: ArithmeticsError) -> Self {
Into::<ExecutionError>::into(error).into()
}
}
impl From<Box<dyn Any + Send>> for OdraError {
fn from(_: Box<dyn Any + Send>) -> Self {
OdraError::VmError(VmError::Panic)
}
}
#[derive(Clone, Debug)]
pub struct ExecutionError(u16, String);
impl ExecutionError {
pub fn new(code: u16, msg: &str) -> Self {
if code > MAX_USER_ERROR {
Self(
USER_ERROR_TOO_HIGH,
String::from("User error too high. The code should be in range 0..32767.")
)
} else {
Self(code, String::from(msg))
}
}
fn sys(code: u16, msg: &str) -> Self {
ExecutionError(code + USER_ERROR_TOO_HIGH, String::from(msg))
}
pub fn code(&self) -> u16 {
self.0
}
pub fn unwrap_error() -> Self {
Self::sys(UNWRAP_ERROR, "Unwrap error")
}
pub fn non_payable() -> Self {
Self::sys(CODE_NON_PAYABLE, "Method does not accept deposit")
}
pub fn can_not_transfer_to_contract() -> Self {
Self::sys(
CODE_TRANSFER_TO_CONTRACT,
"Can't transfer tokens to contract."
)
}
pub fn reentrant_call() -> Self {
Self::sys(CODE_REENTRANT_CALL, "Reentrant call.")
}
pub fn contract_already_installed() -> Self {
Self::sys(
CODE_CONTRACT_ALREADY_INSTALLED,
"Contract already installed."
)
}
pub fn unknown_constructor() -> Self {
Self::sys(CODE_UNKNOWN_CONSTRUCTOR, "Unknown constructor.")
}
pub fn native_token_transfer_error() -> Self {
Self::sys(CODE_NATIVE_TRANSFER_ERROR, "Native token transfer error.")
}
pub fn addition_overflow() -> Self {
Self::sys(CODE_ADDITION_OVERFLOW, "Addition overflow")
}
pub fn subtraction_overflow() -> Self {
Self::sys(CODE_SUBTRACTION_OVERFLOW, "Subtraction overflow")
}
pub fn index_out_of_bounds() -> Self {
Self::sys(CODE_INDEX_OUT_OF_BOUNDS, "Index out of bounds")
}
pub fn zero_address() -> Self {
Self::sys(CODE_ZERO_ADDRESS, "Zero address")
}
pub fn address_creation_failed() -> Self {
Self::sys(CODE_ADDRESS_CREATION_FAILED, "Address creation failed")
}
}
impl PartialEq for ExecutionError {
fn eq(&self, other: &Self) -> bool {
self.0 == other.0
}
}
impl From<ExecutionError> for OdraError {
fn from(error: ExecutionError) -> Self {
Self::ExecutionError(error)
}
}
#[derive(Clone, Debug, PartialEq, Eq)]
pub enum VmError {
Serialization,
Deserialization,
BalanceExceeded,
NoSuchMethod(String),
InvalidContractAddress,
InvalidContext,
MissingArg,
Other(String),
Panic
}
pub enum CollectionError {
IndexOutOfBounds
}
impl From<CollectionError> for ExecutionError {
fn from(error: CollectionError) -> Self {
match error {
CollectionError::IndexOutOfBounds => Self::index_out_of_bounds()
}
}
}
impl From<CollectionError> for OdraError {
fn from(error: CollectionError) -> Self {
Into::<ExecutionError>::into(error).into()
}
}
#[derive(Clone, Debug, PartialEq)]
pub enum AddressError {
ZeroAddress,
AddressCreationError
}
impl From<AddressError> for ExecutionError {
fn from(error: AddressError) -> Self {
match error {
AddressError::ZeroAddress => Self::zero_address(),
AddressError::AddressCreationError => Self::address_creation_failed()
}
}
}
impl From<AddressError> for OdraError {
fn from(error: AddressError) -> Self {
Into::<ExecutionError>::into(error).into()
}
}