smart-config-derive 0.4.0-pre.2

Derive macros for `smart-config` library
Documentation
use proc_macro::TokenStream;
use proc_macro2::{Ident, Span};
use quote::{quote, quote_spanned};
use syn::{DeriveInput, spanned::Spanned};

use crate::utils::{
    ConfigContainer, ConfigContainerFields, ConfigEnumVariant, ConfigField, RenameRule,
    wrap_in_option,
};

impl ConfigField {
    /// Returns `Option<_>`.
    fn deserialize_param(&self, index: usize) -> proc_macro2::TokenStream {
        let name_span = self.name.span();
        if self.attrs.nest {
            if Self::is_option(&self.ty) {
                quote_spanned! {name_span=>
                    ctx.deserialize_nested_config_opt(#index)
                }
            } else {
                let default_fn = wrap_in_option(self.default_fn());
                quote_spanned! {name_span=>
                    ctx.deserialize_nested_config(#index, #default_fn)
                }
            }
        } else {
            quote_spanned! {name_span=>
                ctx.deserialize_param(#index)
            }
        }
    }
}

impl ConfigEnumVariant {
    fn matches(&self, rename_rule: Option<RenameRule>) -> proc_macro2::TokenStream {
        let mut all_names = self.expected_variants(rename_rule);
        let name = all_names.next().unwrap();
        let name_span = self.name.span();
        quote_spanned!(name_span=> #name #(| #all_names)*)
    }
}

impl ConfigContainer {
    fn process_fields<'a>(
        fields: &'a [ConfigField],
        param_index: &'a mut usize,
        nested_index: &'a mut usize,
    ) -> (proc_macro2::TokenStream, Vec<proc_macro2::TokenStream>) {
        let mut init = proc_macro2::TokenStream::default();
        let fields = fields.iter().enumerate().map(|(i, field)| {
            let index;
            if field.attrs.nest {
                index = *nested_index;
                *nested_index += 1;
            } else {
                index = *param_index;
                *param_index += 1;
            }

            let name = &field.name;
            let value = field.deserialize_param(index);
            let local_var = Ident::new(&format!("__{i}"), name.span());
            init.extend(quote_spanned! {name.span()=>
                let #local_var = #value;
            });
            quote_spanned!(name.span()=> #name: #local_var?)
        });
        let fields = fields.collect();
        (init, fields)
    }

    fn derive_deserialize_config(&self) -> proc_macro2::TokenStream {
        let name = &self.name;

        let mut param_index = 0;
        let mut nested_index = 0;
        let instance = match &self.fields {
            ConfigContainerFields::Struct(fields) => {
                let (init, fields) =
                    Self::process_fields(fields, &mut param_index, &mut nested_index);
                quote!({
                    #init
                    Self { #(#fields,)* }
                })
            }
            ConfigContainerFields::Enum { variants, .. } => {
                let match_hands = variants.iter().map(|variant| {
                    let name = &variant.name;
                    let matches = variant.matches(self.attrs.rename_all);
                    let (init, variant_fields) =
                        Self::process_fields(&variant.fields, &mut param_index, &mut nested_index);
                    quote!(#matches => {
                        #init
                        Self::#name { #(#variant_fields,)* }
                    })
                });
                let match_hands: Vec<_> = match_hands.collect();
                let tag_expr = quote! {
                    ctx.deserialize_param::<&'static str>(#param_index)
                };

                quote! {{
                    match #tag_expr? {
                        #(#match_hands)*
                        _ => ::core::unreachable!(),
                    }
                }}
            }
        };

        let cr = self.cr(Span::call_site());
        quote! {
            impl #cr::DeserializeConfig for #name {
                #[allow(unused_mut)]
                fn deserialize_config(
                    mut ctx: #cr::de::DeserializeContext<'_>,
                ) -> ::core::result::Result<Self, #cr::DeserializeConfigError> {
                    ::core::result::Result::Ok(#instance)
                }
            }
        }
    }
}

pub(crate) fn impl_deserialize_config(input: TokenStream) -> TokenStream {
    let input: DeriveInput = syn::parse(input).unwrap();
    let trait_impl = match ConfigContainer::new(&input) {
        Ok(trait_impl) => trait_impl,
        Err(err) => return err.into_compile_error().into(),
    };
    trait_impl.derive_deserialize_config().into()
}