From 1aafd3da5de5c6ea371b662d20b1f8363ef63524 Mon Sep 17 00:00:00 2001 From: tyranron Date: Sun, 19 Jun 2022 22:18:50 +0200 Subject: [PATCH] Impl macro for scalars, vol.4 [skip ci] --- juniper/src/ast.rs | 10 +- juniper/src/graphql/mod.rs | 1 + juniper_codegen/src/graphql_scalar/derive.rs | 16 +- juniper_codegen/src/graphql_scalar/mod.rs | 193 ++++++++++++++----- 4 files changed, 157 insertions(+), 63 deletions(-) diff --git a/juniper/src/ast.rs b/juniper/src/ast.rs index c5f9897f..40cceee4 100644 --- a/juniper/src/ast.rs +++ b/juniper/src/ast.rs @@ -464,12 +464,12 @@ impl InputValue { /// Maps the [`ScalarValue`] type of this [`InputValue`] into the specified /// one. - pub fn map_scalar_value(self) -> InputValue + pub fn map_scalar_value(self) -> InputValue where - To: From + 'static, - S: 'static, + S: ScalarValue, + Into: ScalarValue, { - if TypeId::of::() == TypeId::of::() { + if TypeId::of::() == TypeId::of::() { // SAFETY: This is safe, because we're transmuting the value into // itself, so no invariants may change and we're just // satisfying the type checker. @@ -481,7 +481,7 @@ impl InputValue { } else { match self { Self::Null => InputValue::Null, - Self::Scalar(s) => InputValue::Scalar(s.into()), + Self::Scalar(s) => InputValue::Scalar(s.into_another()), Self::Enum(v) => InputValue::Enum(v), Self::Variable(n) => InputValue::Variable(n), Self::List(l) => InputValue::List( diff --git a/juniper/src/graphql/mod.rs b/juniper/src/graphql/mod.rs index 9e0e1af9..a88b154d 100644 --- a/juniper/src/graphql/mod.rs +++ b/juniper/src/graphql/mod.rs @@ -6,6 +6,7 @@ pub use crate::{ macros::{input_value, value, vars}, resolve::Type, value::Value, + GraphQLScalar as Scalar, }; /* diff --git a/juniper_codegen/src/graphql_scalar/derive.rs b/juniper_codegen/src/graphql_scalar/derive.rs index 835ea37b..b4503796 100644 --- a/juniper_codegen/src/graphql_scalar/derive.rs +++ b/juniper_codegen/src/graphql_scalar/derive.rs @@ -1,5 +1,7 @@ //! Code generation for `#[derive(GraphQLScalar)]` macro. +use std::convert::TryFrom; + use proc_macro2::TokenStream; use quote::ToTokens; use syn::{parse_quote, spanned::Spanned}; @@ -84,11 +86,8 @@ pub(super) fn parse_derived_methods(ast: &syn::DeriveInput, attr: &Attr) -> syn: .first() .filter(|_| fields.unnamed.len() == 1) .cloned() - .map(|f| Field { - itself: f, - is_named: false, - behavior: None.unwrap_or_default(), // TODO: Parse attribute! - }) + .map(Field::try_from) + .transpose()? .ok_or_else(|| { ERR.custom_error( ast.span(), @@ -101,11 +100,8 @@ pub(super) fn parse_derived_methods(ast: &syn::DeriveInput, attr: &Attr) -> syn: .first() .filter(|_| fields.named.len() == 1) .cloned() - .map(|f| Field { - itself: f, - is_named: true, - behavior: None.unwrap_or_default(), // TODO: Parse attribute! - }) + .map(Field::try_from) + .transpose()? .ok_or_else(|| { ERR.custom_error( ast.span(), diff --git a/juniper_codegen/src/graphql_scalar/mod.rs b/juniper_codegen/src/graphql_scalar/mod.rs index 15c44703..0434ecff 100644 --- a/juniper_codegen/src/graphql_scalar/mod.rs +++ b/juniper_codegen/src/graphql_scalar/mod.rs @@ -2,6 +2,8 @@ //! //! [1]: https://spec.graphql.org/October2021#sec-Scalars +use std::convert::TryFrom; + use proc_macro2::{Literal, TokenStream}; use quote::{format_ident, quote, ToTokens, TokenStreamExt}; use syn::{ @@ -536,10 +538,10 @@ impl Definition { let predicates = &mut generics.make_where_clause().predicates; predicates.push(parse_quote! { #sv: Clone }); predicates.push(parse_quote! { - ::juniper::behavior::Coerce: - ::juniper::resolve::TypeName<#inf, #bh> - + ::juniper::resolve::ScalarToken<#sv, #bh> - + ::juniper::resolve::InputValueOwned<#sv, #bh> + ::juniper::behavior::Coerce: + ::juniper::resolve::TypeName<#inf> + + ::juniper::resolve::ScalarToken<#sv> + + ::juniper::resolve::InputValueOwned<#sv> }); let (impl_gens, _, where_clause) = generics.split_for_impl(); @@ -566,7 +568,7 @@ impl Definition { #sv: '__r, { registry.register_scalar_with::< - ::juniper::behavior::Coerce, _, _, + ::juniper::behavior::Coerce, _, _, >(type_info, |meta| { meta#description #specified_by_url @@ -808,27 +810,25 @@ impl Definition { let (sv, mut generics) = self.mix_scalar_value(generics); let lt: syn::GenericParam = parse_quote! { '__inp }; generics.params.push(lt.clone()); - generics - .make_where_clause() - .predicates - .extend(self.methods.bound_try_from_input_value(<, sv, bh)); + let predicates = &mut generics.make_where_clause().predicates; + predicates.push(parse_quote! { #sv: #lt }); + predicates.extend(self.methods.bound_try_from_input_value(<, sv, bh)); let (impl_gens, _, where_clause) = generics.split_for_impl(); - let conversion = self.methods.expand_try_from_input_value(sv, bh); + let error_ty = self.methods.expand_try_from_input_value_error(<, sv, bh); + let body = self.methods.expand_try_from_input_value(sv, bh); quote! { #[automatically_derived] impl#impl_gens ::juniper::resolve::InputValue<#lt, #sv, #bh> for #ty #where_clause { - type Error = ::juniper::FieldError<#sv>; + type Error = #error_ty; fn try_from_input_value( input: &#lt ::juniper::graphql::InputValue<#sv>, ) -> ::std::result::Result { - #conversion.map_err( - ::juniper::IntoFieldError::<#sv>::into_field_error, - ) + #body } } } @@ -1363,13 +1363,14 @@ impl Methods { from_input: Some(from_input), .. } => { - let map_sv = sv.custom.as_ref().map(|custom_ty| { - quote! { - .map_scalar_value() - } + let map_sv = sv.custom.is_some().then(|| { + quote! { .map_scalar_value() } }); quote! { #from_input(input#map_sv) + .map_err( + ::juniper::IntoFieldError::<#sv>::into_field_error, + ) } } @@ -1382,13 +1383,45 @@ impl Methods { <::juniper::behavior::Coerce<#field_ty, #bh> as ::juniper::resolve::InputValue<'_, #sv, #field_bh>> ::try_from_input_value(input) - .into_inner() + .map(::juniper::behavior::Coerce::into_inner) .map(#self_constructor) } } } } + /// Expands error type of [`resolve::InputValue`] trait. + /// + /// [`resolve::InputValue`]: juniper::resolve::InputValue + fn expand_try_from_input_value_error( + &self, + lt: &syn::GenericParam, + sv: &ScalarValue, + bh: &behavior::Type, + ) -> syn::Type { + match self { + Self::Custom { .. } + | Self::Delegated { + from_input: Some(_), + .. + } => { + parse_quote! { + ::juniper::FieldError<#sv> + } + } + + Self::Delegated { field, .. } => { + let field_ty = field.ty(); + let field_bh = &field.behavior; + + parse_quote! { + <::juniper::behavior::Coerce<#field_ty, #bh> as + ::juniper::resolve::InputValue<#lt, #sv, #field_bh>>::Error + } + } + } + } + /// Generates additional trait bounds for [`resolve::InputValue`] /// implementation allowing to execute /// [`resolve::InputValue::try_from_input_value()`][0] method. @@ -1407,15 +1440,15 @@ impl Methods { from_input: Some(_), .. } => { + let mut bounds = vec![parse_quote! { + #sv: ::juniper::ScalarValue + }]; if let Some(custom_sv) = &sv.custom { - vec![ - parse_quote! { #custom_sv: ::std::convert::From<#sv> }, - parse_quote! { #custom_sv: 'static }, - parse_quote! { #sv: 'static }, - ] - } else { - vec![parse_quote! { #sv: ::juniper::ScalarValue }] + bounds.push(parse_quote! { + #custom_sv: ::juniper::ScalarValue + }); } + bounds } Self::Delegated { field, .. } => { @@ -1557,10 +1590,8 @@ impl ParseToken { fn expand_parse_scalar_token(&self, sv: &ScalarValue, bh: &behavior::Type) -> TokenStream { match self { Self::Custom(parse_token) => { - let into = sv.custom.as_ref().map(|custom_ty| { - quote! { - .map(<#sv as ::std::convert::From<#custom_ty>>::from) - } + let into = sv.custom.is_some().then(|| { + quote! { .map(::juniper::ScalarValue::into_another) } }); quote! { #parse_token(token)#into @@ -1606,15 +1637,15 @@ impl ParseToken { ) -> Vec { match self { Self::Custom(_) => { - vec![if let Some(custom_sv) = &sv.custom { - parse_quote! { - #sv: ::std::convert::From<#custom_sv> - } - } else { - parse_quote! { - #sv: ::juniper::ScalarValue - } - }] + let mut bounds = vec![parse_quote! { + #sv: ::juniper::ScalarValue + }]; + if let Some(custom_sv) = &sv.custom { + bounds.push(parse_quote! { + #custom_sv: ::juniper::ScalarValue + }); + } + bounds } Self::Delegated(delegated) => delegated @@ -1630,6 +1661,65 @@ impl ParseToken { } } +/// Available arguments behind `#[graphql]` attribute on a [`Field`] when +/// generating code for a [GraphQL scalar][0] implementation. +/// +/// [0]: https://spec.graphql.org/October2021#sec-Scalars +#[derive(Debug, Default)] +struct FieldAttr { + /// Explicitly specified type of the custom [`Behavior`] used for + /// [GraphQL scalar][0] implementation by the [`Field`]. + /// + /// If [`None`], then [`behavior::Standard`] will be used for the generated + /// code. + /// + /// [`Behavior`]: juniper::behavior + /// [`behavior::Standard`]: juniper::behavior::Standard + /// [0]: https://spec.graphql.org/October2021#sec-Scalars + behavior: Option>, +} + +impl Parse for FieldAttr { + fn parse(input: ParseStream<'_>) -> syn::Result { + let mut out = Self::default(); + while !input.is_empty() { + let ident = input.parse_any_ident()?; + match ident.to_string().as_str() { + "behave" | "behavior" => { + input.parse::()?; + let bh = input.parse::()?; + out.behavior + .replace(SpanContainer::new(ident.span(), Some(bh.span()), bh)) + .none_or_else(|_| err::dup_arg(&ident))? + } + name => { + return Err(err::unknown_arg(&ident, name)); + } + } + input.try_parse::()?; + } + Ok(out) + } +} + +impl FieldAttr { + /// Tries to merge two [`FieldAttr`]s into a single one, reporting about + /// duplicates, if any. + fn try_merge(self, mut another: Self) -> syn::Result { + Ok(Self { + behavior: try_merge_opt!(behavior: self, another), + }) + } + + /// Parses [`FieldAttr`] from the given multiple `name`d [`syn::Attribute`]s + /// placed on a field definition. + fn from_attrs(name: &str, attrs: &[syn::Attribute]) -> syn::Result { + filter_attrs(name, attrs) + .map(|attr| attr.parse_args()) + .try_fold(Self::default(), |prev, curr| prev.try_merge(curr?)) + } +} + /// Inner field of a type implementing [GraphQL scalar][0], that the /// implementation delegates calls to. /// @@ -1638,19 +1728,28 @@ struct Field { /// This [`Field`] itself. itself: syn::Field, - /// Indicator whether this [`Field`] is named. - is_named: bool, - /// [`Behavior`] parametrization of this [`Field`]. /// /// [`Behavior`]: juniper::behavior behavior: behavior::Type, } +impl TryFrom for Field { + type Error = syn::Error; + + fn try_from(field: syn::Field) -> syn::Result { + let attr = FieldAttr::from_attrs("graphql", &field.attrs)?; + Ok(Self { + itself: field, + behavior: attr.behavior.map(|bh| bh.into_inner()).unwrap_or_default(), + }) + } +} + impl ToTokens for Field { fn to_tokens(&self, tokens: &mut TokenStream) { - if self.is_named { - self.itself.ident.to_tokens(tokens) + if let Some(name) = &self.itself.ident { + name.to_tokens(tokens) } else { tokens.append(Literal::u8_unsuffixed(0)) } @@ -1668,10 +1767,8 @@ impl Field { /// /// [0]: https://spec.graphql.org/October2021#sec-Scalars fn closure_constructor(&self) -> TokenStream { - if self.is_named { - let ident = &self.itself.ident; - - quote! { |v| Self { #ident: v } } + if let Some(name) = &self.itself.ident { + quote! { |v| Self { #name: v } } } else { quote! { Self } }