microserde-derive 0.2.0

Derive macros for microserde. Use the re-exports from the microserde crate instead.
Documentation
use crate::{attr, bound};
use proc_macro2::{Span, TokenStream};
use quote::quote;
use syn::{
    parse_quote, Data, DataEnum, DataStruct, DeriveInput, Error, Fields, FieldsNamed, Result,
};

pub fn derive(input: DeriveInput) -> Result<TokenStream> {
    match &input.data {
        Data::Struct(DataStruct {
            fields: Fields::Named(fields),
            ..
        }) => derive_struct(&input, fields),
        Data::Enum(enumeration) => derive_enum(&input, enumeration),
        _ => Err(Error::new(
            Span::call_site(),
            "currently only structs with named fields are supported",
        )),
    }
}

pub fn derive_struct(input: &DeriveInput, fields: &FieldsNamed) -> Result<TokenStream> {
    let ident = &input.ident;
    let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
    let fieldname = fields.named.iter().map(|f| &f.ident).collect::<Vec<_>>();
    let fieldty = fields.named.iter().map(|f| &f.ty).collect::<Vec<_>>();
    let fieldattrs = fields
        .named
        .iter()
        .map(|f| attr::field_attrs(&f.attrs))
        .collect::<Result<Vec<_>>>()?;
    let fieldstr = fields
        .named
        .iter()
        .zip(&fieldattrs)
        .map(|(field, attrs)| {
            Ok(attrs
                .rename
                .clone()
                .unwrap_or_else(|| field.ident.as_ref().unwrap().to_string()))
        })
        .collect::<Result<Vec<_>>>()?;
    let fielddefault = fields
        .named
        .iter()
        .zip(&fieldattrs)
        .map(|(field, attrs)| {
            if attrs.default {
                let ty = &field.ty;
                quote!(microserde::export::Some(<#ty as microserde::export::Default>::default()))
            } else {
                let ty = &field.ty;
                quote!(<#ty as microserde::Deserialize>::default())
            }
        })
        .collect::<Vec<_>>();

    let wrapper_generics = bound::with_lifetime_bound(&input.generics, "'__a");
    let (wrapper_impl_generics, wrapper_ty_generics, _) = wrapper_generics.split_for_impl();
    let bound = parse_quote!(microserde::Deserialize);
    let bounded_where_clause = bound::where_clause_with_bound(&input.generics, bound);

    Ok(quote! {
        #[allow(non_local_definitions)]
        const _: () = {
            #[repr(C)]
            struct __Visitor #impl_generics #where_clause {
                __out: microserde::export::Option<#ident #ty_generics>,
            }

            impl #impl_generics microserde::Deserialize for #ident #ty_generics #bounded_where_clause {
                fn begin(__out: &mut microserde::export::Option<Self>) -> &mut dyn microserde::de::Visitor {
                    unsafe {
                        &mut *{
                            __out
                            as *mut microserde::export::Option<Self>
                            as *mut __Visitor #ty_generics
                        }
                    }
                }
            }

            impl #impl_generics microserde::de::Visitor for __Visitor #ty_generics #bounded_where_clause {
                fn map(&mut self) -> microserde::Result<microserde::export::Box<dyn microserde::de::Map + '_>> {
                    Ok(microserde::export::Box::new(__State {
                        #(
                            #fieldname: #fielddefault,
                        )*
                        __out: &mut self.__out,
                    }))
                }
            }

            struct __State #wrapper_impl_generics #where_clause {
                #(
                    #fieldname: microserde::export::Option<#fieldty>,
                )*
                __out: &'__a mut microserde::export::Option<#ident #ty_generics>,
            }

            impl #wrapper_impl_generics microserde::de::Map for __State #wrapper_ty_generics #bounded_where_clause {
                fn key(&mut self, __k: &microserde::export::str) -> microserde::Result<&mut dyn microserde::de::Visitor> {
                    match __k {
                        #(
                            #fieldstr => microserde::export::Ok(<#fieldty as microserde::Deserialize>::begin(&mut self.#fieldname)),
                        )*
                        _ => microserde::export::Ok(<dyn microserde::de::Visitor>::ignore()),
                    }
                }

                fn finish(&mut self) -> microserde::Result<()> {
                    #(
                        let #fieldname = self.#fieldname.take().ok_or(microserde::Error)?;
                    )*
                    *self.__out = microserde::export::Some(#ident {
                        #(
                            #fieldname,
                        )*
                    });
                    microserde::export::Ok(())
                }
            }
        };
    })
}

pub fn derive_enum(input: &DeriveInput, enumeration: &DataEnum) -> Result<TokenStream> {
    if input.generics.lt_token.is_some() || input.generics.where_clause.is_some() {
        return Err(Error::new(
            Span::call_site(),
            "Enums with generics are not supported",
        ));
    }

    let ident = &input.ident;

    let var_idents = enumeration
        .variants
        .iter()
        .map(|variant| match variant.fields {
            Fields::Unit => Ok(&variant.ident),
            _ => Err(Error::new_spanned(
                variant,
                "Invalid variant: only simple enum variants without fields are supported",
            )),
        })
        .collect::<Result<Vec<_>>>()?;
    let names = enumeration
        .variants
        .iter()
        .map(attr::name_of_variant)
        .collect::<Result<Vec<_>>>()?;

    Ok(quote! {
        #[allow(non_local_definitions)]
        const _: () = {
            #[repr(C)]
            struct __Visitor {
                __out: microserde::export::Option<#ident>,
            }

            impl microserde::Deserialize for #ident {
                fn begin(__out: &mut microserde::export::Option<Self>) -> &mut dyn microserde::de::Visitor {
                    unsafe {
                        &mut *{
                            __out
                            as *mut microserde::export::Option<Self>
                            as *mut __Visitor
                        }
                    }
                }
            }

            impl microserde::de::Visitor for __Visitor {
                fn string(&mut self, s: &microserde::export::str) -> microserde::Result<()> {
                    let value = match s {
                        #( #names => #ident::#var_idents, )*
                        _ => { return microserde::export::Err(microserde::Error) },
                    };
                    self.__out = microserde::export::Some(value);
                    microserde::export::Ok(())
                }
            }
        };
    })
}