2020-03-20 11:11:06 -05:00
|
|
|
#![allow(clippy::match_wild_err_arm)]
|
2020-05-01 21:24:01 -05:00
|
|
|
use crate::{
|
|
|
|
result::{GraphQLScope, UnsupportedAttribute},
|
2020-10-13 11:34:36 -05:00
|
|
|
util::{self, span_container::SpanContainer, RenameRule},
|
2020-05-01 21:24:01 -05:00
|
|
|
};
|
|
|
|
use proc_macro2::TokenStream;
|
2019-04-22 19:19:06 -05:00
|
|
|
use quote::{quote, ToTokens};
|
2020-05-01 21:24:01 -05:00
|
|
|
use syn::{self, ext::IdentExt, spanned::Spanned, Data, Fields};
|
|
|
|
|
2020-06-30 04:26:48 -05:00
|
|
|
pub fn impl_input_object(ast: syn::DeriveInput, error: GraphQLScope) -> syn::Result<TokenStream> {
|
2020-05-01 21:24:01 -05:00
|
|
|
let ast_span = ast.span();
|
2018-05-02 18:21:08 -05:00
|
|
|
let fields = match ast.data {
|
2020-05-01 21:24:01 -05:00
|
|
|
Data::Struct(data) => match data.fields {
|
|
|
|
Fields::Named(named) => named.named,
|
2017-08-06 13:48:46 -05:00
|
|
|
_ => {
|
2020-05-01 21:24:01 -05:00
|
|
|
return Err(
|
|
|
|
error.custom_error(ast_span, "all fields must be named, e.g., `test: String`")
|
|
|
|
)
|
2017-06-24 13:20:00 -05:00
|
|
|
}
|
|
|
|
},
|
2020-05-01 21:24:01 -05:00
|
|
|
_ => return Err(error.custom_error(ast_span, "can only be used on structs with fields")),
|
2017-06-24 13:20:00 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
// Parse attributes.
|
2020-05-01 21:24:01 -05:00
|
|
|
let attrs = util::ObjectAttributes::from_attrs(&ast.attrs)?;
|
2017-08-02 21:18:11 -05:00
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
// Parse attributes.
|
|
|
|
let ident = &ast.ident;
|
|
|
|
let name = attrs
|
|
|
|
.name
|
|
|
|
.clone()
|
|
|
|
.map(SpanContainer::into_inner)
|
|
|
|
.unwrap_or_else(|| ident.to_string());
|
|
|
|
|
|
|
|
let fields = fields
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|field| {
|
|
|
|
let span = field.span();
|
|
|
|
let field_attrs = match util::FieldAttributes::from_attrs(
|
|
|
|
&field.attrs,
|
|
|
|
util::FieldAttributeParseMode::Object,
|
|
|
|
) {
|
|
|
|
Ok(attrs) => attrs,
|
|
|
|
Err(e) => {
|
|
|
|
proc_macro_error::emit_error!(e);
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
};
|
2019-11-13 17:40:22 -06:00
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
let field_ident = field.ident.as_ref().unwrap();
|
|
|
|
let name = match field_attrs.name {
|
|
|
|
Some(ref name) => name.to_string(),
|
2020-10-13 11:34:36 -05:00
|
|
|
None => attrs
|
|
|
|
.rename
|
|
|
|
.unwrap_or(RenameRule::CamelCase)
|
|
|
|
.apply(&field_ident.unraw().to_string()),
|
2020-05-01 21:24:01 -05:00
|
|
|
};
|
2017-06-24 13:20:00 -05:00
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
if let Some(span) = field_attrs.skip {
|
|
|
|
error.unsupported_attribute_within(span.span(), UnsupportedAttribute::Skip)
|
2017-08-06 13:48:46 -05:00
|
|
|
}
|
2017-06-24 13:20:00 -05:00
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
if let Some(span) = field_attrs.deprecation {
|
|
|
|
error.unsupported_attribute_within(
|
|
|
|
span.span_ident(),
|
|
|
|
UnsupportedAttribute::Deprecation,
|
|
|
|
)
|
2017-12-02 08:02:17 -06:00
|
|
|
}
|
2017-06-24 13:20:00 -05:00
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
if name.starts_with("__") {
|
|
|
|
error.no_double_underscore(if let Some(name) = field_attrs.name {
|
|
|
|
name.span_ident()
|
|
|
|
} else {
|
|
|
|
name.span()
|
|
|
|
});
|
2017-06-24 13:20:00 -05:00
|
|
|
}
|
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
let resolver_code = quote!(#field_ident);
|
|
|
|
|
|
|
|
let default = field_attrs
|
|
|
|
.default
|
|
|
|
.map(|default| match default.into_inner() {
|
|
|
|
Some(expr) => expr.into_token_stream(),
|
|
|
|
None => quote! { Default::default() },
|
|
|
|
});
|
|
|
|
|
|
|
|
Some(util::GraphQLTypeDefinitionField {
|
|
|
|
name,
|
|
|
|
_type: field.ty,
|
|
|
|
args: Vec::new(),
|
|
|
|
description: field_attrs.description.map(SpanContainer::into_inner),
|
|
|
|
deprecation: None,
|
|
|
|
resolver_code,
|
|
|
|
is_type_inferred: true,
|
|
|
|
is_async: false,
|
|
|
|
default,
|
|
|
|
span,
|
|
|
|
})
|
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
|
|
|
proc_macro_error::abort_if_dirty();
|
|
|
|
|
|
|
|
if fields.is_empty() {
|
|
|
|
error.not_empty(ast_span);
|
|
|
|
}
|
2017-06-24 13:20:00 -05:00
|
|
|
|
2020-07-16 12:41:09 -05:00
|
|
|
if let Some(duplicates) =
|
|
|
|
crate::util::duplicate::Duplicate::find_by_key(&fields, |field| &field.name)
|
|
|
|
{
|
|
|
|
error.duplicate(duplicates.iter())
|
2020-05-01 21:24:01 -05:00
|
|
|
}
|
2017-06-24 13:20:00 -05:00
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
if !attrs.interfaces.is_empty() {
|
|
|
|
attrs.interfaces.iter().for_each(|elm| {
|
|
|
|
error.unsupported_attribute(elm.span(), UnsupportedAttribute::Interface)
|
2018-08-27 16:51:12 -05:00
|
|
|
});
|
2017-06-24 13:20:00 -05:00
|
|
|
}
|
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
if let Some(duplicates) =
|
|
|
|
crate::util::duplicate::Duplicate::find_by_key(&fields, |field| field.name.as_str())
|
|
|
|
{
|
|
|
|
error.duplicate(duplicates.iter());
|
|
|
|
}
|
2017-06-24 13:20:00 -05:00
|
|
|
|
2020-06-30 04:26:48 -05:00
|
|
|
if !attrs.is_internal && name.starts_with("__") {
|
2020-05-01 21:24:01 -05:00
|
|
|
error.no_double_underscore(if let Some(name) = attrs.name {
|
|
|
|
name.span_ident()
|
|
|
|
} else {
|
|
|
|
ident.span()
|
|
|
|
});
|
|
|
|
}
|
2017-06-24 13:20:00 -05:00
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
proc_macro_error::abort_if_dirty();
|
|
|
|
|
|
|
|
let definition = util::GraphQLTypeDefiniton {
|
|
|
|
name,
|
|
|
|
_type: syn::parse_str(&ast.ident.to_string()).unwrap(),
|
|
|
|
context: attrs.context.map(SpanContainer::into_inner),
|
|
|
|
scalar: attrs.scalar.map(SpanContainer::into_inner),
|
|
|
|
description: attrs.description.map(SpanContainer::into_inner),
|
|
|
|
fields,
|
|
|
|
generics: ast.generics,
|
Make interfaces great again! (#682)
* Bootstrap
* Upd
* Bootstrap macro
* Revert stuff
* Correct PoC to compile
* Bootstrap #[graphql_interface] expansion
* Bootstrap #[graphql_interface] meta parsing
* Bootstrap #[graphql_interface] very basic code generation [skip ci]
* Upd trait code generation and fix keywords usage [skip ci]
* Expand trait impls [skip ci]
* Tune up objects [skip ci]
* Finally! Complies at least... [skip ci]
* Parse meta for fields and its arguments [skip ci]
- also, refactor and bikeshed new macros code
* Impl filling fields meta and bootstrap field resolution [skip ci]
* Poking with fields resolution [skip ci]
* Solve Rust's teen async HRTB problems [skip ci]
* Start parsing trait methods [skip ci]
* Finish parsing fields from trait methods [skip ci]
* Autodetect trait asyncness and allow to specify it [skip ci]
* Allow to autogenerate trait object alias via attribute
* Support generics in trait definition and asyncify them correctly
* Temporary disable explicit async
* Cover arguments and custom names/descriptions in tests
* Re-enable tests with explicit async and fix the codegen to satisfy it
* Check implementers are registered in schema and vice versa
* Check argument camelCases
* Test argument defaults, and allow Into coercions for them
* Re-enable markers
* Re-enable markers and relax Sized requirement on IsInputType/IsOutputType marker traits
* Revert 'juniper_actix' fmt
* Fix missing marks for object
* Fix subscriptions marks
* Deduce result type correctly via traits
* Final fixes
* Fmt
* Restore marks checking
* Support custom ScalarValue
* Cover deprecations with tests
* Impl dowcasting via methods
* Impl dowcasting via external functions
* Support custom context, vol. 1
* Support custom context, vol. 2
* Cover fallible field with test
* Impl explicit generic ScalarValue, vol.1
* Impl explicit generic ScalarValue, vol.2
* Allow passing executor into methods
* Generating enum, vol.1
* Generating enum, vol.2
* Generating enum, vol.3
* Generating enum, vol.3
* Generating enum, vol.4
* Generating enum, vol.5
* Generating enum, vol.6
* Generating enum, vol.7
* Generating enum, vol.8
* Refactor juniper stuff
* Fix juniper tests, vol.1
* Fix juniper tests, vol.2
* Polish 'juniper' crate changes, vol.1
* Polish 'juniper' crate changes, vol.2
* Remove redundant stuf
* Polishing 'juniper_codegen', vol.1
* Polishing 'juniper_codegen', vol.2
* Polishing 'juniper_codegen', vol.3
* Polishing 'juniper_codegen', vol.4
* Polishing 'juniper_codegen', vol.5
* Polishing 'juniper_codegen', vol.6
* Polishing 'juniper_codegen', vol.7
* Polishing 'juniper_codegen', vol.8
* Polishing 'juniper_codegen', vol.9
* Fix other crates tests and make Clippy happier
* Fix examples
* Add codegen failure tests, vol. 1
* Add codegen failure tests, vol. 2
* Add codegen failure tests, vol.3
* Fix codegen failure tests accordingly to latest nightly Rust
* Fix codegen when interface has no implementers
* Fix warnings in book tests
* Describing new interfaces in Book, vol.1
Co-authored-by: Christian Legnitto <LegNeato@users.noreply.github.com>
2020-10-06 02:21:01 -05:00
|
|
|
interfaces: vec![],
|
2020-05-01 21:24:01 -05:00
|
|
|
include_type_generics: true,
|
|
|
|
generic_scalar: true,
|
|
|
|
no_async: attrs.no_async.is_some(),
|
2017-12-02 08:02:17 -06:00
|
|
|
};
|
|
|
|
|
2020-06-30 04:26:48 -05:00
|
|
|
Ok(definition.into_input_object_tokens())
|
2017-06-24 13:20:00 -05:00
|
|
|
}
|