use crate::{InputValueError, InputValueResult, Result, ScalarType, Value}; use async_graphql_derive::Scalar; #[cfg(feature = "bson")] use bson::oid::{self, ObjectId}; use std::convert::TryFrom; use std::num::ParseIntError; use std::ops::{Deref, DerefMut}; /// 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)] pub struct ID(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 where T: std::fmt::Display, { fn from(value: T) -> Self { ID(value.to_string()) } } impl Into for ID { fn into(self) -> String { self.0 } } impl TryFrom for usize { type Error = ParseIntError; fn try_from(id: ID) -> std::result::Result { id.0.parse() } } impl TryFrom for uuid::Uuid { type Error = uuid::Error; fn try_from(id: ID) -> std::result::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)] impl ScalarType for ID { fn parse(value: Value) -> InputValueResult { match value { Value::Int(n) => Ok(ID(n.to_string())), Value::String(s) => Ok(ID(s)), _ => Err(InputValueError::ExpectedType(value)), } } fn is_valid(value: &Value) -> bool { match value { Value::Int(_) | Value::String(_) => true, _ => false, } } fn to_json(&self) -> Result { Ok(self.0.clone().into()) } }