juniper_codegen/graphql_input_object/
derive.rs

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
//! Code generation for `#[derive(GraphQLInputObject)]` macro.

use std::collections::HashSet;

use proc_macro2::TokenStream;
use quote::ToTokens as _;
use syn::{ext::IdentExt as _, parse_quote, spanned::Spanned};

use crate::common::{diagnostic, rename, scalar, SpanContainer};

use super::{ContainerAttr, Definition, FieldAttr, FieldDefinition};

/// [`diagnostic::Scope`] of errors for `#[derive(GraphQLInputObject)]` macro.
const ERR: diagnostic::Scope = diagnostic::Scope::InputObjectDerive;

/// Expands `#[derive(GraphQLInputObject)]` macro into generated code.
pub fn expand(input: TokenStream) -> syn::Result<TokenStream> {
    let ast = syn::parse2::<syn::DeriveInput>(input)?;
    let attr = ContainerAttr::from_attrs("graphql", &ast.attrs)?;

    let data = if let syn::Data::Struct(data) = &ast.data {
        data
    } else {
        return Err(ERR.custom_error(ast.span(), "can only be derived on structs"));
    };

    let renaming = attr
        .rename_fields
        .map(SpanContainer::into_inner)
        .unwrap_or(rename::Policy::CamelCase);

    let is_internal = attr.is_internal;
    let fields = data
        .fields
        .iter()
        .filter_map(|f| parse_field(f, renaming, is_internal))
        .collect::<Vec<_>>();

    diagnostic::abort_if_dirty();

    if !fields.iter().any(|f| !f.ignored) {
        return Err(ERR.custom_error(data.fields.span(), "expected at least 1 non-ignored field"));
    }

    let unique_fields = fields.iter().map(|v| &v.name).collect::<HashSet<_>>();
    if unique_fields.len() != fields.len() {
        return Err(ERR.custom_error(
            data.fields.span(),
            "expected all fields to have unique names",
        ));
    }

    let name = attr
        .name
        .clone()
        .map(SpanContainer::into_inner)
        .unwrap_or_else(|| ast.ident.unraw().to_string())
        .into_boxed_str();
    if !attr.is_internal && name.starts_with("__") {
        ERR.no_double_underscore(
            attr.name
                .as_ref()
                .map(SpanContainer::span_ident)
                .unwrap_or_else(|| ast.ident.span()),
        );
    }

    let context = attr
        .context
        .map_or_else(|| parse_quote! { () }, SpanContainer::into_inner);

    let scalar = scalar::Type::parse(attr.scalar.as_deref(), &ast.generics);

    diagnostic::abort_if_dirty();

    let definition = Definition {
        ident: ast.ident,
        generics: ast.generics,
        name,
        description: attr.description.map(SpanContainer::into_inner),
        context,
        scalar,
        fields,
    };

    Ok(definition.into_token_stream())
}

/// Parses a [`FieldDefinition`] from the given struct field definition.
///
/// Returns [`None`] if the parsing fails.
fn parse_field(
    f: &syn::Field,
    renaming: rename::Policy,
    is_internal: bool,
) -> Option<FieldDefinition> {
    let field_attr = FieldAttr::from_attrs("graphql", &f.attrs)
        .map_err(diagnostic::emit_error)
        .ok()?;

    let ident = f.ident.as_ref().or_else(|| err_unnamed_field(f))?;

    let name = field_attr
        .name
        .map_or_else(
            || renaming.apply(&ident.unraw().to_string()),
            SpanContainer::into_inner,
        )
        .into_boxed_str();
    if !is_internal && name.starts_with("__") {
        ERR.no_double_underscore(f.span());
    }

    Some(FieldDefinition {
        ident: ident.clone(),
        ty: f.ty.clone(),
        default: field_attr.default.map(SpanContainer::into_inner),
        name,
        description: field_attr.description.map(SpanContainer::into_inner),
        ignored: field_attr.ignore.is_some(),
    })
}

/// Emits "expected named struct field" [`syn::Error`] pointing to the given
/// `span`.
pub(crate) fn err_unnamed_field<T, S: Spanned>(span: &S) -> Option<T> {
    ERR.emit_custom(span.span(), "expected named struct field");
    None
}