Building up codegen for scalars, vol.2
This commit is contained in:
parent
21c7a3a653
commit
8ea231f395
3 changed files with 185 additions and 22 deletions
|
@ -25,9 +25,8 @@ impl<Info: ?Sized, S: ScalarValue> resolve::Type<Info, S> for str {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<Info: ?Sized> resolve::TypeName<Info> for str {
|
impl<Info: ?Sized> resolve::TypeName<Info> for str {
|
||||||
fn type_name(_: &Info) -> &'static str {
|
fn type_name(info: &Info) -> &str {
|
||||||
// TODO: Reuse from `String`.
|
<String as resolve::TypeName<Info>>::type_name(info)
|
||||||
"String"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -69,8 +68,7 @@ where
|
||||||
|
|
||||||
impl<S: ScalarValue> resolve::ScalarToken<S> for str {
|
impl<S: ScalarValue> resolve::ScalarToken<S> for str {
|
||||||
fn parse_scalar_token(token: ScalarToken<'_>) -> Result<S, ParseError<'_>> {
|
fn parse_scalar_token(token: ScalarToken<'_>) -> Result<S, ParseError<'_>> {
|
||||||
// TODO: Replace with `resolve::ScalarToken<S>`
|
<String as resolve::ScalarToken<S>>::parse_scalar_token(token)
|
||||||
<String as crate::ParseScalarValue<S>>::from_str(token)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -63,15 +63,17 @@ pub(crate) enum Type {
|
||||||
/// [`ScalarValue`]: juniper::ScalarValue
|
/// [`ScalarValue`]: juniper::ScalarValue
|
||||||
Concrete(syn::Type),
|
Concrete(syn::Type),
|
||||||
|
|
||||||
/// One of type parameters of the original type is specified as [`ScalarValue`].
|
/// One of type parameters of the original type is specified as
|
||||||
|
/// [`ScalarValue`].
|
||||||
///
|
///
|
||||||
/// The original type is the type that the code is generated for.
|
/// The original type is the type that the code is generated for.
|
||||||
///
|
///
|
||||||
/// [`ScalarValue`]: juniper::ScalarValue
|
/// [`ScalarValue`]: juniper::ScalarValue
|
||||||
ExplicitGeneric(syn::Ident),
|
ExplicitGeneric(syn::Ident),
|
||||||
|
|
||||||
/// [`ScalarValue`] parametrization is assumed to be generic and is not specified
|
/// [`ScalarValue`] parametrization is assumed to be generic and is not
|
||||||
/// explicitly, or specified as bound predicate (like `S: ScalarValue + Send + Sync`).
|
/// specified explicitly, or specified as bound predicate (like
|
||||||
|
/// `S: ScalarValue + Send + Sync`).
|
||||||
///
|
///
|
||||||
/// [`ScalarValue`]: juniper::ScalarValue
|
/// [`ScalarValue`]: juniper::ScalarValue
|
||||||
ImplicitGeneric(Option<syn::PredicateType>),
|
ImplicitGeneric(Option<syn::PredicateType>),
|
||||||
|
|
|
@ -331,7 +331,9 @@ impl ToTokens for Definition {
|
||||||
self.impl_parse_scalar_value_tokens().to_tokens(into);
|
self.impl_parse_scalar_value_tokens().to_tokens(into);
|
||||||
self.impl_reflection_traits_tokens().to_tokens(into);
|
self.impl_reflection_traits_tokens().to_tokens(into);
|
||||||
////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////
|
||||||
self.impl_resolve_input_value_tokens().to_tokens(into);
|
self.impl_resolve_type_name().to_tokens(into);
|
||||||
|
self.impl_resolve_input_value().to_tokens(into);
|
||||||
|
self.impl_resolve_scalar_token().to_tokens(into);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -406,6 +408,30 @@ impl Definition {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns generated code implementing [`resolve::TypeName`] trait for this
|
||||||
|
/// [GraphQL scalar][1].
|
||||||
|
///
|
||||||
|
/// [`resolve::TypeName`]: juniper::resolve::TypeName
|
||||||
|
/// [1]: https://spec.graphql.org/October2021#sec-Scalars
|
||||||
|
fn impl_resolve_type_name(&self) -> TokenStream {
|
||||||
|
let name = &self.name;
|
||||||
|
|
||||||
|
let (ty, generics) = self.ty_and_generics();
|
||||||
|
let (info_ty, generics) = self.mix_info_ty(generics);
|
||||||
|
let (impl_gens, _, where_clause) = generics.split_for_impl();
|
||||||
|
|
||||||
|
quote! {
|
||||||
|
#[automatically_derived]
|
||||||
|
impl#impl_gens ::juniper::resolve::TypeName<#info_ty> for #ty
|
||||||
|
#where_clause
|
||||||
|
{
|
||||||
|
fn type_name(_: &#info_ty) -> &'static str {
|
||||||
|
#name
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns generated code implementing [`GraphQLValue`] trait for this
|
/// Returns generated code implementing [`GraphQLValue`] trait for this
|
||||||
/// [GraphQL scalar][1].
|
/// [GraphQL scalar][1].
|
||||||
///
|
///
|
||||||
|
@ -531,27 +557,28 @@ impl Definition {
|
||||||
///
|
///
|
||||||
/// [`resolve::InputValue`]: juniper::resolve::InputValue
|
/// [`resolve::InputValue`]: juniper::resolve::InputValue
|
||||||
/// [1]: https://spec.graphql.org/October2021#sec-Scalars
|
/// [1]: https://spec.graphql.org/October2021#sec-Scalars
|
||||||
fn impl_resolve_input_value_tokens(&self) -> TokenStream {
|
fn impl_resolve_input_value(&self) -> TokenStream {
|
||||||
let scalar = &self.scalar;
|
let conversion = self.methods.expand_try_from_input_value(&self.scalar);
|
||||||
|
|
||||||
let conversion = self.methods.expand_try_from_input_value(scalar);
|
let (ty, generics) = self.ty_and_generics();
|
||||||
|
let (scalar, mut generics) = self.mix_scalar_ty(generics);
|
||||||
let (ty, mut generics) = self.impl_self_and_generics(false);
|
let lt: syn::GenericParam = parse_quote! { '__inp };
|
||||||
generics.params.push(parse_quote! { '__inp });
|
generics.params.push(lt.clone());
|
||||||
let (impl_gens, _, where_clause) = generics.split_for_impl();
|
let (impl_gens, _, where_clause) = generics.split_for_impl();
|
||||||
|
|
||||||
quote! {
|
quote! {
|
||||||
#[automatically_derived]
|
#[automatically_derived]
|
||||||
impl#impl_gens ::juniper::resolve::InputValue<'__inp, #scalar> for #ty
|
impl#impl_gens ::juniper::resolve::InputValue<#lt, #scalar> for #ty
|
||||||
#where_clause
|
#where_clause
|
||||||
{
|
{
|
||||||
type Error = ::juniper::FieldError<#scalar>;
|
type Error = ::juniper::FieldError<#scalar>;
|
||||||
|
|
||||||
fn try_from_input_value(
|
fn try_from_input_value(
|
||||||
input: &'__inp ::juniper::graphql::InputValue<#scalar>,
|
input: &#lt ::juniper::graphql::InputValue<#scalar>,
|
||||||
) -> Result<Self, Self::Error> {
|
) -> Result<Self, Self::Error> {
|
||||||
#conversion
|
#conversion.map_err(
|
||||||
.map_err(::juniper::IntoFieldError::<#scalar>::into_field_error)
|
::juniper::IntoFieldError::<#scalar>::into_field_error,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -584,6 +611,35 @@ impl Definition {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns generated code implementing [`resolve::ScalarToken`] trait for
|
||||||
|
/// this [GraphQL scalar][1].
|
||||||
|
///
|
||||||
|
/// [`resolve::ScalarToken`]: juniper::resolve::ScalarToken
|
||||||
|
/// [1]: https://spec.graphql.org/October2021#sec-Scalars
|
||||||
|
fn impl_resolve_scalar_token(&self) -> TokenStream {
|
||||||
|
let body = self.methods.expand_parse_scalar_token(&self.scalar);
|
||||||
|
|
||||||
|
let (ty, generics) = self.ty_and_generics();
|
||||||
|
let (scalar, generics) = self.mix_scalar_ty(generics);
|
||||||
|
let (impl_gens, _, where_clause) = generics.split_for_impl();
|
||||||
|
|
||||||
|
quote! {
|
||||||
|
#[automatically_derived]
|
||||||
|
impl#impl_gens ::juniper::resolve::ScalarToken<#scalar> for #ty
|
||||||
|
#where_clause
|
||||||
|
{
|
||||||
|
fn parse_scalar_token(
|
||||||
|
token: ::juniper::parser::ScalarToken<'_>,
|
||||||
|
) -> ::std::result::Result<
|
||||||
|
#scalar,
|
||||||
|
::juniper::parser::ParseError<'_>,
|
||||||
|
> {
|
||||||
|
#body
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns generated code implementing [`BaseType`], [`BaseSubTypes`] and
|
/// Returns generated code implementing [`BaseType`], [`BaseSubTypes`] and
|
||||||
/// [`WrappedType`] traits for this [GraphQL scalar][1].
|
/// [`WrappedType`] traits for this [GraphQL scalar][1].
|
||||||
///
|
///
|
||||||
|
@ -700,6 +756,57 @@ impl Definition {
|
||||||
|
|
||||||
(ty, generics)
|
(ty, generics)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[must_use]
|
||||||
|
fn ty_and_generics(&self) -> (syn::Type, syn::Generics) {
|
||||||
|
let mut generics = self.generics.clone();
|
||||||
|
|
||||||
|
let ty = match &self.ty {
|
||||||
|
TypeOrIdent::Type(ty) => (**ty).clone(),
|
||||||
|
TypeOrIdent::Ident(ident) => {
|
||||||
|
let (_, ty_gen, _) = self.generics.split_for_impl();
|
||||||
|
parse_quote! { #ident#ty_gen }
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if !self.where_clause.is_empty() {
|
||||||
|
generics
|
||||||
|
.make_where_clause()
|
||||||
|
.predicates
|
||||||
|
.extend(self.where_clause.clone())
|
||||||
|
}
|
||||||
|
|
||||||
|
(ty, generics)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[must_use]
|
||||||
|
fn mix_info_ty(&self, mut generics: syn::Generics) -> (syn::Ident, syn::Generics) {
|
||||||
|
let ty = parse_quote! { __Info };
|
||||||
|
|
||||||
|
generics.params.push(parse_quote! { #ty: ?Sized });
|
||||||
|
|
||||||
|
(ty, generics)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[must_use]
|
||||||
|
fn mix_scalar_ty(&self, mut generics: syn::Generics) -> (&scalar::Type, syn::Generics) {
|
||||||
|
let scalar = &self.scalar;
|
||||||
|
|
||||||
|
if scalar.is_implicit_generic() {
|
||||||
|
generics.params.push(parse_quote! { #scalar });
|
||||||
|
}
|
||||||
|
if scalar.is_generic() {
|
||||||
|
generics
|
||||||
|
.make_where_clause()
|
||||||
|
.predicates
|
||||||
|
.push(parse_quote! { #scalar: ::juniper::ScalarValue });
|
||||||
|
}
|
||||||
|
if let Some(bound) = scalar.bounds() {
|
||||||
|
generics.make_where_clause().predicates.push(bound);
|
||||||
|
}
|
||||||
|
|
||||||
|
(scalar, generics)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Adds `__fa__` prefix to all lifetimes to avoid "lifetime name `'a` shadows a
|
/// Adds `__fa__` prefix to all lifetimes to avoid "lifetime name `'a` shadows a
|
||||||
|
@ -823,7 +930,8 @@ impl Methods {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Expands [`resolve::InputValue::try_from_input_value()`][0] method.
|
/// Expands body of [`resolve::InputValue::try_from_input_value()`][0]
|
||||||
|
/// method.
|
||||||
///
|
///
|
||||||
/// [0]: juniper::resolve::InputValue::try_from_input_value
|
/// [0]: juniper::resolve::InputValue::try_from_input_value
|
||||||
fn expand_try_from_input_value(&self, scalar: &scalar::Type) -> TokenStream {
|
fn expand_try_from_input_value(&self, scalar: &scalar::Type) -> TokenStream {
|
||||||
|
@ -841,8 +949,9 @@ impl Methods {
|
||||||
let self_constructor = field.closure_constructor();
|
let self_constructor = field.closure_constructor();
|
||||||
|
|
||||||
quote! {
|
quote! {
|
||||||
<#field_ty as ::juniper::resolve::InputValue<'__inp, #scalar>>::try_from_input_value(input)
|
<#field_ty as ::juniper::resolve::InputValue<'_, #scalar>>
|
||||||
.map(#self_constructor)
|
::try_from_input_value(input)
|
||||||
|
.map(#self_constructor)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -869,6 +978,27 @@ impl Methods {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Expands body of [`resolve::ScalarToken::parse_scalar_token()`][0]
|
||||||
|
/// method.
|
||||||
|
///
|
||||||
|
/// [0]: resolve::ScalarToken::parse_scalar_token
|
||||||
|
fn expand_parse_scalar_token(&self, scalar: &scalar::Type) -> TokenStream {
|
||||||
|
match self {
|
||||||
|
Self::Custom { parse_token, .. }
|
||||||
|
| Self::Delegated {
|
||||||
|
parse_token: Some(parse_token),
|
||||||
|
..
|
||||||
|
} => parse_token.expand_parse_scalar_token(scalar),
|
||||||
|
Self::Delegated { field, .. } => {
|
||||||
|
let field_ty = field.ty();
|
||||||
|
quote! {
|
||||||
|
<#field_ty as ::juniper::resolve::ScalarToken<#scalar>>
|
||||||
|
::parse_scalar_token(token)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Representation of [`ParseScalarValue::from_str`] method.
|
/// Representation of [`ParseScalarValue::from_str`] method.
|
||||||
|
@ -912,6 +1042,39 @@ impl ParseToken {
|
||||||
.unwrap_or_default(),
|
.unwrap_or_default(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Expands body of [`resolve::ScalarToken::parse_scalar_token()`][0]
|
||||||
|
/// method.
|
||||||
|
///
|
||||||
|
/// [0]: resolve::ScalarToken::parse_scalar_token
|
||||||
|
fn expand_parse_scalar_token(&self, scalar: &scalar::Type) -> TokenStream {
|
||||||
|
match self {
|
||||||
|
Self::Custom(parse_token) => {
|
||||||
|
quote! { #parse_token(token) }
|
||||||
|
}
|
||||||
|
Self::Delegated(delegated) => delegated
|
||||||
|
.iter()
|
||||||
|
.fold(None, |acc, ty| {
|
||||||
|
acc.map_or_else(
|
||||||
|
|| {
|
||||||
|
Some(quote! {
|
||||||
|
<#ty as ::juniper::resolve::ScalarToken<#scalar>>
|
||||||
|
::parse_scalar_token(token)
|
||||||
|
})
|
||||||
|
},
|
||||||
|
|prev| {
|
||||||
|
Some(quote! {
|
||||||
|
#prev.or_else(|_| {
|
||||||
|
<#ty as ::juniper::resolve::ScalarToken<#scalar>>
|
||||||
|
::parse_scalar_token(token)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
},
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.unwrap_or_default(),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Struct field to resolve not provided methods.
|
/// Struct field to resolve not provided methods.
|
||||||
|
|
Loading…
Reference in a new issue