use darling::{util::parse_expr, FromDeriveInput, FromMeta};
use syn::{parse_quote, Expr};
#[derive(FromDeriveInput)]
#[darling(attributes(demo))]
pub struct Receiver {
#[darling(with = parse_expr::preserve_str_literal, map = Some)]
example1: Option<Expr>,
#[darling(
with = |m| Ok(String::from_meta(m)?.to_uppercase()),
map = Some
)]
example2: Option<String>,
#[darling(with = |m| Ok(m.path().clone()))]
example3: syn::Path,
#[darling(with = "example4_parser")]
example4: String,
}
fn example4_parser(meta: &syn::Meta) -> darling::Result<String> {
match meta {
syn::Meta::NameValue(nv) => match &nv.value {
syn::Expr::Lit(syn::ExprLit {
lit: syn::Lit::Str(s),
..
}) => Ok(s.value()),
other => Err(darling::Error::unexpected_expr_type(other)),
},
_ => Err(darling::Error::unexpected_type("name-value")),
}
}
#[test]
fn handles_all_cases() {
let input = Receiver::from_derive_input(&parse_quote! {
#[demo(example1 = test::path, example2 = "hello", example3, example4 = "world")]
struct Example;
})
.unwrap();
assert_eq!(input.example1, Some(parse_quote!(test::path)));
assert_eq!(input.example2, Some("HELLO".to_string()));
assert_eq!(input.example3, parse_quote!(example3));
assert_eq!(input.example4, "world".to_string());
}