drt-sc-derive 0.0.1

DharitriOne smart contract API procedural macros
Documentation
use quote::quote;

use crate::{format::format_tokenize, generate::util::byte_str_literal};

use super::{count_args, parse_format_string, FormatPartType};

pub fn format_receiver_args_macro(input: proc_macro2::TokenStream) -> proc_macro2::TokenStream {
    let tokens = format_tokenize::tokenize(input);
    assert!(
        tokens.len() > 2,
        "format_receiver_args macro requires at least 2 arguments"
    );
    let num_arguments = tokens.len() - 2;
    let mut tokens_iter: std::vec::IntoIter<proc_macro2::TokenTree> = tokens.into_iter();

    let accumulator_expr = tokens_iter.next().unwrap();
    let format_string_token = tokens_iter.next().unwrap();
    let format_string = if let proc_macro2::TokenTree::Literal(lit) = format_string_token {
        lit.to_string()
    } else {
        panic!(
            "Formatting requires that the first argument is a string. Found: {format_string_token}"
        );
    };

    let format_str_parts = parse_format_string(&format_string);
    let num_placeholders = count_args(&format_str_parts);
    assert!(
        num_placeholders == num_arguments,
        "Number of placeholders ({num_placeholders}) does not match number of arguments ({num_arguments})."
    );

    format_tokens(format_str_parts, accumulator_expr, tokens_iter)
}

fn format_tokens(
    format_str_parts: Vec<FormatPartType>,
    accumulator_expr: proc_macro2::TokenTree,
    mut tokens_iter: std::vec::IntoIter<proc_macro2::TokenTree>,
) -> proc_macro2::TokenStream {
    format_str_parts.into_iter().map(|part| {
        match part {
            FormatPartType::StaticAscii(ascii_string) => {
                let str_as_bytes = byte_str_literal(ascii_string.as_bytes());
                quote! {
                    drt_sc::formatter::FormatBuffer::append_ascii(&mut #accumulator_expr, #str_as_bytes);
                }
            },
            FormatPartType::Display => {
                let arg_expr = tokens_iter.next().unwrap();
                quote! {
                    drt_sc::formatter::FormatBuffer::append_display(&mut #accumulator_expr, &#arg_expr);
                }
            },
            FormatPartType::LowerHex => {
                let arg_expr = tokens_iter.next().unwrap();
                quote! {
                    drt_sc::formatter::FormatBuffer::append_lower_hex(&mut #accumulator_expr, &#arg_expr);
                }
            },
            FormatPartType::Codec => {
                let arg_expr = tokens_iter.next().unwrap();
                quote! {
                    drt_sc::formatter::FormatBuffer::append_codec(&mut #accumulator_expr, &#arg_expr);
                }
            },
            FormatPartType::Bytes => {
                let arg_expr = tokens_iter.next().unwrap();
                quote! {
                    drt_sc::formatter::FormatBuffer::append_binary(&mut #accumulator_expr, &#arg_expr);
                }
            },
        }
    }).collect()
}