agdb_derive 0.12.10

Agnesoft Graph Database - derive macros
Documentation
use super::generics;
use super::struct_def;
use super::tuple_def;
use proc_macro2::TokenStream;
use quote::quote;
use syn::DataEnum;
use syn::DeriveInput;
use syn::Fields;
use syn::Ident;
use syn::Token;
use syn::Variant;
use syn::punctuated::Punctuated;

pub(crate) fn parse_enum(e: &DataEnum, input: &DeriveInput) -> TokenStream {
    let name = &input.ident;
    let generic_names = generics::list_generics(&input.generics);
    let generics = generics::parse_generics(name, &input.generics);
    let variants = parse_variants(name, &e.variants, &generic_names);
    let (impl_generics, ty_generic, where_clause) = input.generics.split_for_impl();

    quote! {
        impl #impl_generics ::agdb::api_def::TypeDefinition for #name #ty_generic #where_clause {
            fn type_def() -> ::agdb::api_def::Type {
                ::agdb::api_def::Type::Enum(::agdb::api_def::Enum {
                    name: stringify!(#name),
                    generics: &[#(#generics),*],
                    variants: &[#(#variants),*],
                    functions: <#name #ty_generic as ::agdb::api_def::ImplDefinition>::functions(),
                })
            }
        }
    }
}

fn parse_variants(
    name: &Ident,
    variants: &Punctuated<Variant, Token![,]>,
    generics: &[String],
) -> Vec<TokenStream> {
    variants
        .iter()
        .map(|v| {
            let variant_name = &v.ident;
            match &v.fields {
                Fields::Named(fields_named) => {
                    let fields = struct_def::parse_named_fields(name, Some(fields_named), generics);

                    quote! {
                        ::agdb::api_def::NamedType {
                            name: stringify!(#variant_name),
                            ty: Some(|| ::agdb::api_def::Type::Struct(::agdb::api_def::Struct {
                                name: stringify!(#variant_name),
                                generics: &[],
                                fields: &[#(#fields),*],
                                functions: &[],
                            })),
                        }
                    }
                }
                Fields::Unnamed(fields_unnamed) => {
                    let fields = tuple_def::parse_unnamed_fields(Some(fields_unnamed), generics);

                    quote! {
                        ::agdb::api_def::NamedType {
                            name: stringify!(#variant_name),
                            ty: Some(|| ::agdb::api_def::Type::Tuple(::agdb::api_def::Tuple {
                                name: stringify!(#variant_name),
                                generics: &[],
                                fields: &[#(#fields),*],
                                functions: &[],
                            })),
                        }
                    }
                }
                Fields::Unit => {
                    quote! {
                        ::agdb::api_def::NamedType {
                            name: stringify!(#variant_name),
                            ty: None,
                        }
                    }
                }
            }
        })
        .collect()
}