multiversx-sc-codec-derive 0.25.0

Macro implementations of multiversx-sc-codec #[derive(NestedEncode, NestedDecode, TopEncode, TopDecode)]
Documentation
use crate::util::*;
use proc_macro::TokenStream;
use quote::quote;

pub fn dep_decode_snippet(
    _index: usize,
    field: &syn::Field,
    input_value: &proc_macro2::TokenStream,
) -> proc_macro2::TokenStream {
    let ty = sanitize_type_path(field.ty.clone());
    if let Some(ident) = &field.ident {
        quote! {
            #ident: <#ty as codec::NestedDecode>::dep_decode_or_handle_err(#input_value, __h__)?
        }
    } else {
        quote! {
            <#ty as codec::NestedDecode>::dep_decode_or_handle_err(#input_value, __h__)?
        }
    }
}

pub fn variant_dep_decode_snippets(
    name: &syn::Ident,
    data_enum: &syn::DataEnum,
    input_value: &proc_macro2::TokenStream,
) -> Vec<proc_macro2::TokenStream> {
    let mut previous_disc: Vec<ExplicitDiscriminant> = Vec::new();
    data_enum
        .variants
        .iter()
        .enumerate()
        .map(|(variant_index, variant)| {
            let variant_discriminant = get_discriminant(variant_index, variant, &mut previous_disc);
            let variant_ident = &variant.ident;
            let variant_field_snippets = fields_decl_syntax(&variant.fields, |index, field| {
                dep_decode_snippet(index, field, input_value)
            });
            quote! {
                #variant_discriminant => core::result::Result::Ok( #name::#variant_ident #variant_field_snippets ),
            }
        })
        .collect()
}

pub fn nested_decode_impl(ast: &syn::DeriveInput) -> TokenStream {
    let name = &ast.ident;
    let (impl_generics, ty_generics, where_clause) = &ast.generics.split_for_impl();
    let result = match &ast.data {
        syn::Data::Struct(data_struct) => {
            let field_dep_decode_snippets =
                fields_decl_syntax(&data_struct.fields, |index, field| {
                    dep_decode_snippet(index, field, &quote! {input})
                });

            let decode_body = dep_decode_body(name, &field_dep_decode_snippets);

            quote! {
                impl #impl_generics codec::NestedDecode for #name #ty_generics #where_clause {
                    fn dep_decode_or_handle_err<I, H>(input: &mut I, __h__: H) -> core::result::Result<Self, H::HandledErr>
                    where
                        I: codec::NestedDecodeInput,
                        H: codec::DecodeErrorHandler,
                    {
                        #decode_body
                    }
                }
            }
        }
        syn::Data::Enum(data_enum) => {
            validate_enum_variants(&data_enum.variants);

            let variant_dep_decode_snippets =
                variant_dep_decode_snippets(name, data_enum, &quote! {input});

            quote! {
                impl #impl_generics codec::NestedDecode for #name #ty_generics #where_clause {
                    fn dep_decode_or_handle_err<I, H>(input: &mut I, __h__: H) -> core::result::Result<Self, H::HandledErr>
                    where
                        I: codec::NestedDecodeInput,
                        H: codec::DecodeErrorHandler,
                    {
                        match <u8 as codec::NestedDecode>::dep_decode_or_handle_err(input, __h__)? {
                            #(#variant_dep_decode_snippets)*
                            _ => core::result::Result::Err(__h__.handle_error(codec::DecodeError::INVALID_VALUE)),
                        }
                    }
                }
            }
        }
        syn::Data::Union(_) => panic!("Union not supported"),
    };

    result.into()
}

fn dep_decode_body(
    name: &proc_macro2::Ident,
    field_dep_decode_snippets: &proc_macro2::TokenStream,
) -> proc_macro2::TokenStream {
    if field_dep_decode_snippets
        .to_string()
        .contains(BITFLAGS_PRIMITIVE)
    {
        return quote!(
            match #name::from_bits #field_dep_decode_snippets {
                Some(r) => core::result::Result::Ok(r),
                None => Err(__h__.handle_error(codec::DecodeError::INVALID_VALUE))
            }
        );
    }

    quote! (
        core::result::Result::Ok(
            #name #field_dep_decode_snippets
        )
    )
}