2020-05-01 21:24:01 -05:00
|
|
|
use crate::{
|
|
|
|
result::{GraphQLScope, UnsupportedAttribute},
|
|
|
|
util::{self, span_container::SpanContainer},
|
|
|
|
};
|
2018-12-19 12:27:49 -06:00
|
|
|
use proc_macro2::TokenStream;
|
2019-04-22 19:19:06 -05:00
|
|
|
use quote::quote;
|
2020-05-01 21:24:01 -05:00
|
|
|
use syn::{self, ext::IdentExt, spanned::Spanned, Data, Fields};
|
|
|
|
|
|
|
|
pub fn impl_enum(
|
|
|
|
ast: syn::DeriveInput,
|
|
|
|
is_internal: bool,
|
|
|
|
error: GraphQLScope,
|
|
|
|
) -> syn::Result<TokenStream> {
|
|
|
|
let ast_span = ast.span();
|
2017-06-24 06:23:20 -05:00
|
|
|
|
2020-04-18 00:09:04 -05:00
|
|
|
if !ast.generics.params.is_empty() {
|
2020-05-01 21:24:01 -05:00
|
|
|
return Err(error.custom_error(ast_span, "does not support generics or lifetimes"));
|
2017-06-24 06:23:20 -05:00
|
|
|
}
|
2018-12-18 18:56:37 -06:00
|
|
|
|
2018-05-02 18:21:08 -05:00
|
|
|
let variants = match ast.data {
|
2020-04-18 00:09:04 -05:00
|
|
|
Data::Enum(enum_data) => enum_data.variants,
|
2020-05-01 21:24:01 -05:00
|
|
|
_ => return Err(error.custom_error(ast_span, "can only be applied to enums")),
|
2017-08-02 21:18:11 -05:00
|
|
|
};
|
2017-06-24 06:23:20 -05:00
|
|
|
|
2020-04-18 00:09:04 -05:00
|
|
|
// Parse attributes.
|
2020-05-01 21:24:01 -05:00
|
|
|
let attrs = util::ObjectAttributes::from_attrs(&ast.attrs)?;
|
2020-04-18 00:09:04 -05:00
|
|
|
let ident = &ast.ident;
|
2020-05-01 21:24:01 -05:00
|
|
|
let name = attrs
|
|
|
|
.name
|
|
|
|
.clone()
|
|
|
|
.map(SpanContainer::into_inner)
|
|
|
|
.unwrap_or_else(|| ident.unraw().to_string());
|
2020-04-18 00:09:04 -05:00
|
|
|
|
|
|
|
let fields = variants
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(|field| {
|
2020-05-01 21:24:01 -05:00
|
|
|
let span = field.span();
|
2020-04-18 00:09:04 -05:00
|
|
|
let field_attrs = match util::FieldAttributes::from_attrs(
|
2020-05-01 21:24:01 -05:00
|
|
|
&field.attrs,
|
2020-04-18 00:09:04 -05:00
|
|
|
util::FieldAttributeParseMode::Object,
|
|
|
|
) {
|
|
|
|
Ok(attrs) => attrs,
|
2020-05-01 21:24:01 -05:00
|
|
|
Err(err) => {
|
|
|
|
proc_macro_error::emit_error!(err);
|
|
|
|
return None;
|
|
|
|
}
|
2020-04-18 00:09:04 -05:00
|
|
|
};
|
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
let field_name = field.ident;
|
|
|
|
let name = field_attrs
|
|
|
|
.name
|
|
|
|
.clone()
|
|
|
|
.map(SpanContainer::into_inner)
|
|
|
|
.unwrap_or_else(|| util::to_upper_snake_case(&field_name.unraw().to_string()));
|
|
|
|
|
|
|
|
let resolver_code = quote!( #ident::#field_name );
|
|
|
|
|
|
|
|
let _type = match field.fields {
|
|
|
|
Fields::Unit => syn::parse_str(&field_name.to_string()).unwrap(),
|
|
|
|
_ => {
|
|
|
|
error.custom(
|
|
|
|
field.fields.span(),
|
|
|
|
"all fields of the enum must be unnamed, e.g., None",
|
|
|
|
);
|
|
|
|
return None;
|
2020-04-18 00:09:04 -05:00
|
|
|
}
|
2020-05-01 21:24:01 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
if let Some(skip) = field_attrs.skip {
|
|
|
|
error.unsupported_attribute(skip.span(), UnsupportedAttribute::Skip);
|
|
|
|
return None;
|
|
|
|
}
|
2017-06-24 06:23:20 -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 {
|
|
|
|
field_name.span()
|
|
|
|
});
|
2017-06-24 06:23:20 -05:00
|
|
|
}
|
2020-05-01 21:24:01 -05:00
|
|
|
|
|
|
|
if let Some(default) = field_attrs.default {
|
|
|
|
error.unsupported_attribute_within(
|
|
|
|
default.span_ident(),
|
|
|
|
UnsupportedAttribute::Default,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(util::GraphQLTypeDefinitionField {
|
|
|
|
name,
|
|
|
|
_type,
|
|
|
|
args: Vec::new(),
|
|
|
|
description: field_attrs.description.map(SpanContainer::into_inner),
|
|
|
|
deprecation: field_attrs.deprecation.map(SpanContainer::into_inner),
|
|
|
|
resolver_code,
|
|
|
|
is_type_inferred: true,
|
|
|
|
is_async: false,
|
|
|
|
default: None,
|
|
|
|
span,
|
|
|
|
})
|
2020-04-18 00:09:04 -05:00
|
|
|
})
|
|
|
|
.collect::<Vec<_>>();
|
2017-06-24 06:23:20 -05:00
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
proc_macro_error::abort_if_dirty();
|
|
|
|
|
|
|
|
if fields.is_empty() {
|
|
|
|
error.not_empty(ast_span);
|
|
|
|
}
|
|
|
|
|
|
|
|
match crate::util::duplicate::Duplicate::find_by_key(&fields, |field| &field.name) {
|
|
|
|
Some(duplicates) => error.duplicate(duplicates.iter()),
|
|
|
|
None => {}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !attrs.interfaces.is_empty() {
|
|
|
|
attrs.interfaces.iter().for_each(|elm| {
|
|
|
|
error.unsupported_attribute(elm.span(), UnsupportedAttribute::Interface)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(scalar) = attrs.scalar {
|
|
|
|
error.unsupported_attribute(scalar.span_ident(), UnsupportedAttribute::Scalar);
|
2020-04-18 00:09:04 -05:00
|
|
|
}
|
2019-08-18 14:36:44 -05:00
|
|
|
|
2020-05-01 21:24:01 -05:00
|
|
|
if name.starts_with("__") && !is_internal {
|
|
|
|
error.no_double_underscore(if let Some(name) = attrs.name {
|
|
|
|
name.span_ident()
|
|
|
|
} else {
|
|
|
|
ident.span()
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
proc_macro_error::abort_if_dirty();
|
|
|
|
|
2020-04-18 00:09:04 -05:00
|
|
|
let definition = util::GraphQLTypeDefiniton {
|
|
|
|
name,
|
|
|
|
_type: syn::parse_str(&ast.ident.to_string()).unwrap(),
|
2020-05-01 21:24:01 -05:00
|
|
|
context: attrs.context.map(SpanContainer::into_inner),
|
2020-04-18 00:09:04 -05:00
|
|
|
scalar: None,
|
2020-05-01 21:24:01 -05:00
|
|
|
description: attrs.description.map(SpanContainer::into_inner),
|
2020-04-18 00:09:04 -05:00
|
|
|
fields,
|
|
|
|
// NOTICE: only unit variants allow -> no generics possible
|
|
|
|
generics: syn::Generics::default(),
|
|
|
|
interfaces: None,
|
|
|
|
include_type_generics: true,
|
|
|
|
generic_scalar: true,
|
2020-05-01 21:24:01 -05:00
|
|
|
no_async: attrs.no_async.is_some(),
|
2017-12-02 05:42:07 -06:00
|
|
|
};
|
2019-11-15 14:23:59 -06:00
|
|
|
|
2020-04-18 00:09:04 -05:00
|
|
|
let juniper_crate_name = if is_internal { "crate" } else { "juniper" };
|
2020-05-01 21:24:01 -05:00
|
|
|
Ok(definition.into_enum_tokens(juniper_crate_name))
|
2017-06-24 06:23:20 -05:00
|
|
|
}
|