Refactor & Optimize - Number and Array (#1868)

This commit is contained in:
Finn Bear 2023-04-25 14:52:02 -07:00 committed by GitHub
parent 38016a0ee6
commit c7633414b5
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 46 additions and 315 deletions

View file

@ -76,11 +76,7 @@ impl From<Array> for Vec<Value> {
impl FromIterator<Value> for Array { impl FromIterator<Value> for Array {
fn from_iter<I: IntoIterator<Item = Value>>(iter: I) -> Self { fn from_iter<I: IntoIterator<Item = Value>>(iter: I) -> Self {
let mut a: Vec<Value> = vec![]; Array(iter.into_iter().collect())
for v in iter {
a.push(v)
}
Array(a)
} }
} }

View file

@ -36,77 +36,19 @@ impl Default for Number {
} }
} }
impl From<i8> for Number { macro_rules! from_prim_ints {
fn from(i: i8) -> Self { ($($int: ty),*) => {
Self::Int(i as i64) $(
} impl From<$int> for Number {
fn from(i: $int) -> Self {
Self::Int(i as i64)
}
}
)*
};
} }
impl From<i16> for Number { from_prim_ints!(i8, i16, i32, i64, i128, isize, u8, u16, u32, u64, u128, usize);
fn from(i: i16) -> Self {
Self::Int(i as i64)
}
}
impl From<i32> for Number {
fn from(i: i32) -> Self {
Self::Int(i as i64)
}
}
impl From<i64> for Number {
fn from(i: i64) -> Self {
Self::Int(i)
}
}
impl From<i128> for Number {
fn from(i: i128) -> Self {
Self::Int(i as i64)
}
}
impl From<isize> for Number {
fn from(i: isize) -> Self {
Self::Int(i as i64)
}
}
impl From<u8> for Number {
fn from(i: u8) -> Self {
Self::Int(i as i64)
}
}
impl From<u16> for Number {
fn from(i: u16) -> Self {
Self::Int(i as i64)
}
}
impl From<u32> for Number {
fn from(i: u32) -> Self {
Self::Int(i as i64)
}
}
impl From<u64> for Number {
fn from(i: u64) -> Self {
Self::Int(i as i64)
}
}
impl From<u128> for Number {
fn from(i: u128) -> Self {
Self::Int(i as i64)
}
}
impl From<usize> for Number {
fn from(i: usize) -> Self {
Self::Int(i as i64)
}
}
impl From<f32> for Number { impl From<f32> for Number {
fn from(f: f32) -> Self { fn from(f: f32) -> Self {
@ -165,245 +107,38 @@ impl TryFrom<&str> for Number {
} }
} }
impl TryFrom<Number> for i8 { macro_rules! try_into_prim {
type Error = Error; // TODO: switch to one argument per int once https://github.com/rust-lang/rust/issues/29599 is stable
fn try_from(value: Number) -> Result<Self, Self::Error> { ($($int: ty => $to_int: ident),*) => {
match value { $(
Number::Int(v) => match v.to_i8() { impl TryFrom<Number> for $int {
Some(v) => Ok(v), type Error = Error;
None => Err(Error::TryFrom(value.to_string(), "i8")), fn try_from(value: Number) -> Result<Self, Self::Error> {
}, match value {
Number::Float(v) => match v.to_i8() { Number::Int(v) => match v.$to_int() {
Some(v) => Ok(v), Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i8")), None => Err(Error::TryFrom(value.to_string(), stringify!($int))),
}, },
Number::Decimal(ref v) => match v.to_i8() { Number::Float(v) => match v.$to_int() {
Some(v) => Ok(v), Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i8")), None => Err(Error::TryFrom(value.to_string(), stringify!($int))),
}, },
} Number::Decimal(ref v) => match v.$to_int() {
} Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), stringify!($int))),
},
}
}
}
)*
};
} }
impl TryFrom<Number> for i16 { try_into_prim!(
type Error = Error; i8 => to_i8, i16 => to_i16, i32 => to_i32, i64 => to_i64, i128 => to_i128,
fn try_from(value: Number) -> Result<Self, Self::Error> { u8 => to_u8, u16 => to_u16, u32 => to_u32, u64 => to_u64, u128 => to_u128,
match value { f32 => to_f32, f64 => to_f64
Number::Int(v) => match v.to_i16() { );
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i16")),
},
Number::Float(v) => match v.to_i16() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i16")),
},
Number::Decimal(ref v) => match v.to_i16() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i16")),
},
}
}
}
impl TryFrom<Number> for i32 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_i32() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i32")),
},
Number::Float(v) => match v.to_i32() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i32")),
},
Number::Decimal(ref v) => match v.to_i32() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i32")),
},
}
}
}
impl TryFrom<Number> for i64 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_i64() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i64")),
},
Number::Float(v) => match v.to_i64() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i64")),
},
Number::Decimal(ref v) => match v.to_i64() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i64")),
},
}
}
}
impl TryFrom<Number> for i128 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_i128() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i128")),
},
Number::Float(v) => match v.to_i128() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i128")),
},
Number::Decimal(ref v) => match v.to_i128() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "i128")),
},
}
}
}
impl TryFrom<Number> for u8 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_u8() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u8")),
},
Number::Float(v) => match v.to_u8() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u8")),
},
Number::Decimal(ref v) => match v.to_u8() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u8")),
},
}
}
}
impl TryFrom<Number> for u16 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_u16() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u16")),
},
Number::Float(v) => match v.to_u16() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u16")),
},
Number::Decimal(ref v) => match v.to_u16() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u16")),
},
}
}
}
impl TryFrom<Number> for u32 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_u32() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u32")),
},
Number::Float(v) => match v.to_u32() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u32")),
},
Number::Decimal(ref v) => match v.to_u32() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u32")),
},
}
}
}
impl TryFrom<Number> for u64 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_u64() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u64")),
},
Number::Float(v) => match v.to_u64() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u64")),
},
Number::Decimal(ref v) => match v.to_u64() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u64")),
},
}
}
}
impl TryFrom<Number> for u128 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_u128() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u128")),
},
Number::Float(v) => match v.to_u128() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u128")),
},
Number::Decimal(ref v) => match v.to_u128() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "u128")),
},
}
}
}
impl TryFrom<Number> for f32 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_f32() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "f32")),
},
Number::Float(v) => match v.to_f32() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "f32")),
},
Number::Decimal(ref v) => match v.to_f32() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "f32")),
},
}
}
}
impl TryFrom<Number> for f64 {
type Error = Error;
fn try_from(value: Number) -> Result<Self, Self::Error> {
match value {
Number::Int(v) => match v.to_f64() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "f64")),
},
Number::Float(v) => match v.to_f64() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "f64")),
},
Number::Decimal(ref v) => match v.to_f64() {
Some(v) => Ok(v),
None => Err(Error::TryFrom(value.to_string(), "f64")),
},
}
}
}
impl TryFrom<Number> for BigDecimal { impl TryFrom<Number> for BigDecimal {
type Error = Error; type Error = Error;
@ -496,7 +231,7 @@ impl Number {
match self { match self {
Number::Int(v) => v > &0, Number::Int(v) => v > &0,
Number::Float(v) => v > &0.0, Number::Float(v) => v > &0.0,
Number::Decimal(v) => v > &BigDecimal::from(0), Number::Decimal(v) => v > &BigDecimal::default(),
} }
} }
@ -504,7 +239,7 @@ impl Number {
match self { match self {
Number::Int(v) => v < &0, Number::Int(v) => v < &0,
Number::Float(v) => v < &0.0, Number::Float(v) => v < &0.0,
Number::Decimal(v) => v < &BigDecimal::from(0), Number::Decimal(v) => v < &BigDecimal::default(),
} }
} }
@ -512,7 +247,7 @@ impl Number {
match self { match self {
Number::Int(v) => v >= &0, Number::Int(v) => v >= &0,
Number::Float(v) => v >= &0.0, Number::Float(v) => v >= &0.0,
Number::Decimal(v) => v >= &BigDecimal::from(0), Number::Decimal(v) => v >= &BigDecimal::default(),
} }
} }
@ -520,7 +255,7 @@ impl Number {
match self { match self {
Number::Int(v) => v <= &0, Number::Int(v) => v <= &0,
Number::Float(v) => v <= &0.0, Number::Float(v) => v <= &0.0,
Number::Decimal(v) => v <= &BigDecimal::from(0), Number::Decimal(v) => v <= &BigDecimal::default(),
} }
} }