async-graphql/src/scalars/any.rs

85 lines
2.3 KiB
Rust
Raw Normal View History

use crate::{Result, ScalarType, Value};
use async_graphql_derive::Scalar;
2020-04-09 14:03:09 +00:00
use itertools::Itertools;
2020-05-07 10:49:09 +00:00
use serde::de::DeserializeOwned;
2020-04-09 14:03:09 +00:00
/// Any scalar
///
/// The `Any` scalar is used to pass representations of entities from external services into the root `_entities` field for execution.
#[derive(Clone, PartialEq, Debug)]
pub struct Any(pub Value);
#[Scalar(internal)]
impl ScalarType for Any {
2020-04-09 14:03:09 +00:00
fn type_name() -> &'static str {
"_Any"
}
fn description() -> Option<&'static str> {
Some("The `_Any` scalar is used to pass representations of entities from external services into the root `_entities` field for execution.")
}
fn parse(value: &Value) -> Option<Self> {
Some(Self(value.clone()))
}
fn is_valid(_value: &Value) -> bool {
true
}
fn to_json(&self) -> Result<serde_json::Value> {
Ok(gql_value_to_json_value(&self.0))
}
}
2020-05-07 10:49:09 +00:00
impl Any {
/// Parse this `Any` value to T by `serde_json`.
fn parse_value<T: DeserializeOwned>(&self) -> std::result::Result<T, serde_json::Error> {
serde_json::from_value(self.to_json().unwrap())
}
}
2020-04-09 14:03:09 +00:00
fn gql_value_to_json_value(value: &Value) -> serde_json::Value {
match value {
Value::Null => serde_json::Value::Null,
Value::Variable(name) => name.clone().into(),
Value::Int(n) => n.as_i64().unwrap().into(),
Value::Float(n) => (*n).into(),
Value::String(s) => s.clone().into(),
Value::Boolean(v) => (*v).into(),
Value::Enum(e) => e.clone().into(),
Value::List(values) => values
.iter()
2020-04-09 14:03:09 +00:00
.map(|value| gql_value_to_json_value(value))
.collect_vec()
.into(),
Value::Object(obj) => serde_json::Value::Object(
obj.iter()
2020-04-09 14:03:09 +00:00
.map(|(k, v)| (k.clone(), gql_value_to_json_value(v)))
.collect(),
),
}
}
impl<T> From<T> for Any
where
T: Into<Value>,
{
fn from(value: T) -> Any {
Any(value.into())
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn test_conversion_ok() {
let value = Value::List(vec![Value::Int(1.into()), Value::Float(2.0), Value::Null]);
let expected = Any(value.clone());
let output: Any = value.into();
assert_eq!(output, expected);
}
}