use proc_macro::TokenStream;
use quote::quote;
use syn::spanned::Spanned;
use syn::{FnArg, Ident, ItemFn, Pat, PatType, ReturnType, Signature, Type, parse_macro_input};
fn extract_program_param(sig: &Signature) -> syn::Result<(Pat, Type)> {
if sig.inputs.len() != 1 {
return Err(syn::Error::new(
sig.inputs.span(),
"Setup function must have exactly one parameter",
));
}
let arg = &sig.inputs[0];
match arg {
FnArg::Typed(PatType { pat, ty, .. }) => {
let param_pat = (**pat).clone();
let param_type = (**ty).clone();
Ok((param_pat, param_type))
}
FnArg::Receiver(_) => Err(syn::Error::new(
arg.span(),
"Setup function cannot have self parameter",
)),
}
}
fn extract_return_type(sig: &Signature) -> syn::Result<()> {
match &sig.output {
ReturnType::Type(_, ty) => {
match &**ty {
Type::Tuple(tuple) if tuple.elems.is_empty() => Ok(()),
_ => Err(syn::Error::new(
ty.span(),
"Setup function must return () or have no return type",
)),
}
}
ReturnType::Default => Ok(()),
}
}
pub fn setup_attr(attr: TokenStream, item: TokenStream) -> TokenStream {
let (program_name, use_crate_prefix) = if attr.is_empty() {
(crate::default_program_path(), true)
} else {
let path: syn::Path = parse_macro_input!(attr as syn::Path);
(quote! { #path }, false)
};
let input_fn = parse_macro_input!(item as ItemFn);
if input_fn.sig.asyncness.is_some() {
return syn::Error::new(input_fn.sig.span(), "Setup function cannot be async")
.to_compile_error()
.into();
}
let (program_param, program_type) = match extract_program_param(&input_fn.sig) {
Ok(info) => info,
Err(e) => return e.to_compile_error().into(),
};
if let Err(e) = extract_return_type(&input_fn.sig) {
return e.to_compile_error().into();
}
let fn_body = &input_fn.block;
let mut fn_attrs = input_fn.attrs.clone();
fn_attrs.retain(|attr| !attr.path().is_ident("setup"));
let vis = &input_fn.vis;
let fn_name = &input_fn.sig.ident;
let pascal_case_name = just_fmt::pascal_case!(fn_name.to_string());
let struct_name = Ident::new(&pascal_case_name, fn_name.span());
let expanded = if use_crate_prefix {
quote! {
#(#fn_attrs)*
#[doc(hidden)]
#vis struct #struct_name;
impl ::mingling::setup::ProgramSetup<#program_name> for #struct_name {
fn setup(&mut self, program: &mut ::mingling::Program<#program_name>) {
#fn_name(program);
}
}
#(#fn_attrs)*
#vis fn #fn_name(#program_param: #program_type) {
#fn_body
}
}
} else {
quote! {
#(#fn_attrs)*
#vis struct #struct_name;
impl ::mingling::setup::ProgramSetup<#program_name> for #struct_name {
fn setup(&mut self, program: &mut ::mingling::Program<#program_name>) {
#fn_name(program);
}
}
#(#fn_attrs)*
#vis fn #fn_name(#program_param: #program_type) {
#fn_body
}
}
};
expanded.into()
}