use std::convert::TryFrom; use std::num::ParseIntError; use std::ops::{Deref, DerefMut}; #[cfg(feature = "bson")] use bson::oid::{self, ObjectId}; use serde::{Deserialize, Serialize}; use crate::{InputValueError, InputValueResult, Scalar, ScalarType, Value}; /// ID scalar /// /// The input is a `&str`, `String`, `usize` or `uuid::UUID`, and the output is a string. #[derive(Clone, Ord, PartialOrd, Eq, PartialEq, Hash, Debug, Serialize, Deserialize)] #[serde(transparent)] pub struct ID(pub String); impl Deref for ID { type Target = String; fn deref(&self) -> &Self::Target { &self.0 } } impl DerefMut for ID { fn deref_mut(&mut self) -> &mut Self::Target { &mut self.0 } } impl From for ID { fn from(value: T) -> Self { ID(value.to_string()) } } impl From for String { fn from(id: ID) -> Self { id.0 } } macro_rules! try_from_integers { ($($ty:ty),*) => { $( impl TryFrom for $ty { type Error = ParseIntError; fn try_from(id: ID) -> Result { id.0.parse() } } )* }; } try_from_integers!(i8, i16, i32, i64, i128, u8, u16, u32, u64, u128, isize, usize); #[cfg(feature = "uuid")] impl TryFrom for uuid::Uuid { type Error = uuid::Error; fn try_from(id: ID) -> Result { uuid::Uuid::parse_str(&id.0) } } #[cfg(feature = "bson")] impl TryFrom for ObjectId { type Error = oid::Error; fn try_from(id: ID) -> std::result::Result { ObjectId::with_string(&id.0) } } impl PartialEq<&str> for ID { fn eq(&self, other: &&str) -> bool { self.0.as_str() == *other } } #[Scalar(internal, name = "ID")] impl ScalarType for ID { fn parse(value: Value) -> InputValueResult { match value { Value::Number(n) if n.is_i64() => Ok(ID(n.to_string())), Value::String(s) => Ok(ID(s)), _ => Err(InputValueError::expected_type(value)), } } fn is_valid(value: &Value) -> bool { match value { Value::Number(n) if n.is_i64() => true, Value::String(_) => true, _ => false, } } fn to_value(&self) -> Value { Value::String(self.0.clone()) } }