use std::fmt::{self, Debug, Display};
#[derive(PartialEq, Clone)]
pub struct Number {
n: N,
}
#[derive(Debug, PartialEq, Clone)]
enum N {
PosInt(u64),
NegInt(i64),
Float(f64),
}
impl Number {
#[inline]
pub fn is_i64(&self) -> bool {
match self.n {
N::PosInt(v) => v <= i64::max_value() as u64,
N::NegInt(_) => true,
N::Float(_) => false,
}
}
#[inline]
pub fn is_u64(&self) -> bool {
match self.n {
N::PosInt(_) => true,
N::NegInt(_) | N::Float(_) => false,
}
}
#[inline]
pub fn is_f64(&self) -> bool {
match self.n {
N::Float(_) => true,
N::PosInt(_) | N::NegInt(_) => false,
}
}
#[inline]
pub fn as_i64(&self) -> Option<i64> {
match self.n {
N::PosInt(n) => {
if n <= i64::max_value() as u64 {
Some(n as i64)
} else {
None
}
}
N::NegInt(n) => Some(n),
N::Float(_) => None,
}
}
#[inline]
pub fn as_u64(&self) -> Option<u64> {
match self.n {
N::PosInt(n) => Some(n),
N::NegInt(_) | N::Float(_) => None,
}
}
#[inline]
pub fn as_f64(&self) -> Option<f64> {
match self.n {
N::PosInt(n) => Some(n as f64),
N::NegInt(n) => Some(n as f64),
N::Float(n) => Some(n),
}
}
#[inline]
pub fn from_f64(f: f64) -> Option<Number> {
if f.is_finite() {
Some(Number { n: N::Float(f) })
} else {
None
}
}
pub fn visit<V>(&self, visitor: V) -> Result<V::Value, V::Error>
where
V: Visitor,
{
match self.n {
N::PosInt(n) => visitor.visit_u64(n),
N::NegInt(n) => visitor.visit_i64(n),
N::Float(n) => visitor.visit_f64(n),
}
}
}
pub trait Visitor {
type Value;
type Error;
fn error<T: Into<String>>(msg: T) -> Self::Error;
fn visit_u64(self, n: u64) -> Result<Self::Value, Self::Error>;
fn visit_i64(self, n: i64) -> Result<Self::Value, Self::Error>;
fn visit_f64(self, n: f64) -> Result<Self::Value, Self::Error>;
}
macro_rules! impl_from_unsigned {
(
$($ty:ty),*
) => {
$(
impl From<$ty> for Number {
#[inline]
fn from(u: $ty) -> Self {
Number { n: N::PosInt(u64::from(u)) }
}
}
)*
};
}
macro_rules! impl_from_signed {
(
$($ty:ty),*
) => {
$(
impl From<$ty> for Number {
#[inline]
fn from(n: $ty) -> Self {
let n = if n >= 0 {
N::PosInt(n as u64)
} else {
N::NegInt(i64::from(n))
};
Number { n }
}
}
)*
};
}
impl_from_unsigned!(u8, u16, u32, u64);
impl_from_signed!(i8, i16, i32, i64);
impl From<f32> for Number {
#[inline]
fn from(n: f32) -> Self {
Number {
n: N::Float(f64::from(n)),
}
}
}
impl From<f64> for Number {
#[inline]
fn from(n: f64) -> Self {
Number { n: N::Float(n) }
}
}
impl Display for Number {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
match self.n {
N::PosInt(i) => Display::fmt(&i, formatter),
N::NegInt(i) => Display::fmt(&i, formatter),
N::Float(f) => Display::fmt(&f, formatter),
}
}
}
impl Debug for Number {
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
Debug::fmt(&self.n, formatter)
}
}