use std::{convert::TryFrom, fmt};
use serde::{Deserialize, Serialize, de};
use crate::shared::models::{
SATS_PER_BTC, leverage::Leverage, margin::Margin, price::Price, quantity::Quantity,
};
use super::error::CrossLeverageValidationError;
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
pub struct CrossLeverage(u64);
impl CrossLeverage {
pub const MIN: Self = Self(1);
pub const MAX: Self = Self(100);
pub fn bounded<T>(value: T) -> Self
where
T: Into<f64>,
{
let as_f64: f64 = value.into();
let rounded = as_f64.round().max(0.0) as u64;
let clamped = rounded.clamp(Self::MIN.0, Self::MAX.0);
Self(clamped)
}
pub fn as_u64(&self) -> u64 {
self.0
}
pub fn try_calculate_rounded(
quantity: Quantity,
margin: Margin,
price: Price,
) -> Result<Self, CrossLeverageValidationError> {
let leverage_value = quantity.as_f64() * SATS_PER_BTC / (margin.as_f64() * price.as_f64());
Self::try_from(leverage_value.round())
}
}
impl From<CrossLeverage> for u64 {
fn from(value: CrossLeverage) -> u64 {
value.0
}
}
impl From<CrossLeverage> for Leverage {
fn from(value: CrossLeverage) -> Leverage {
Leverage::try_from(value.0 as f64).expect("Must be a valid `Leverage`")
}
}
impl TryFrom<u64> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: u64) -> Result<Self, Self::Error> {
if value < Self::MIN.0 {
return Err(CrossLeverageValidationError::TooLow { value });
}
if value > Self::MAX.0 {
return Err(CrossLeverageValidationError::TooHigh { value });
}
Ok(CrossLeverage(value))
}
}
impl TryFrom<u8> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: u8) -> Result<Self, Self::Error> {
Self::try_from(value as u64)
}
}
impl TryFrom<u16> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: u16) -> Result<Self, Self::Error> {
Self::try_from(value as u64)
}
}
impl TryFrom<u32> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: u32) -> Result<Self, Self::Error> {
Self::try_from(value as u64)
}
}
impl TryFrom<i8> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: i8) -> Result<Self, Self::Error> {
Self::try_from(value.max(0) as u64)
}
}
impl TryFrom<i16> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: i16) -> Result<Self, Self::Error> {
Self::try_from(value.max(0) as u64)
}
}
impl TryFrom<i32> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: i32) -> Result<Self, Self::Error> {
Self::try_from(value.max(0) as u64)
}
}
impl TryFrom<i64> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: i64) -> Result<Self, Self::Error> {
Self::try_from(value.max(0) as u64)
}
}
impl TryFrom<usize> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: usize) -> Result<Self, Self::Error> {
Self::try_from(value as u64)
}
}
impl TryFrom<isize> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: isize) -> Result<Self, Self::Error> {
Self::try_from(value.max(0) as u64)
}
}
impl TryFrom<f32> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: f32) -> Result<Self, Self::Error> {
Self::try_from(value as f64)
}
}
impl TryFrom<f64> for CrossLeverage {
type Error = CrossLeverageValidationError;
fn try_from(value: f64) -> Result<Self, Self::Error> {
if value.fract() != 0.0 {
return Err(CrossLeverageValidationError::NotAnInteger { value });
}
Self::try_from(value.max(0.) as u64)
}
}
impl fmt::Display for CrossLeverage {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
self.0.fmt(f)
}
}
impl Serialize for CrossLeverage {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
serializer.serialize_u64(self.0)
}
}
impl<'de> Deserialize<'de> for CrossLeverage {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
let leverage_u64 = u64::deserialize(deserializer)?;
CrossLeverage::try_from(leverage_u64).map_err(|e| de::Error::custom(e.to_string()))
}
}