[go: up one dir, main page]

darling 0.21.3

A proc-macro library for reading attributes into structs when implementing custom derives.
Documentation
use darling::{FromDeriveInput, FromMeta};
use syn::parse_quote;

#[derive(Debug, Default, PartialEq, Eq, FromMeta)]
#[darling(default)]
pub struct Amet {
    hello: bool,
    world: String,
}

#[derive(Debug, PartialEq, Eq, FromMeta)]
#[darling(rename_all = "snake_case")]
pub enum Lorem {
    Ipsum(bool),
    Dolor(String),
    OptDolor(Option<String>),
    Sit(Amet),
}

#[derive(Debug, PartialEq, Eq, FromDeriveInput)]
#[darling(attributes(hello))]
pub struct Holder {
    lorem: Lorem,
}

impl PartialEq<Lorem> for Holder {
    fn eq(&self, other: &Lorem) -> bool {
        self.lorem == *other
    }
}

#[test]
fn bool_word() {
    let di = parse_quote! {
        #[hello(lorem(ipsum))]
        pub struct Bar;
    };

    let pr = Holder::from_derive_input(&di).unwrap();
    assert_eq!(pr, Lorem::Ipsum(true));
}

#[test]
fn bool_literal() {
    let di = parse_quote! {
        #[hello(lorem(ipsum = false))]
        pub struct Bar;
    };

    let pr = Holder::from_derive_input(&di).unwrap();
    assert_eq!(pr, Lorem::Ipsum(false));
}

#[test]
fn string_literal() {
    let di = parse_quote! {
        #[hello(lorem(dolor = "Hello"))]
        pub struct Bar;
    };

    let pr = Holder::from_derive_input(&di).unwrap();
    assert_eq!(pr, Lorem::Dolor("Hello".to_string()));
}

#[test]
fn option_literal() {
    let holder = Holder::from_derive_input(&parse_quote! {
        #[hello(lorem(opt_dolor = "Hello"))]
        struct Bar;
    })
    .unwrap();

    assert_eq!(holder.lorem, Lorem::OptDolor(Some("Hello".into())));
}

/// Make sure newtype variants whose field's type's `from_none` return
/// a `Some` can be used in key-value form.
#[test]
fn option_word_only() {
    let holder = Holder::from_derive_input(&parse_quote! {
        #[hello(lorem = "opt_dolor")]
        struct Bar;
    })
    .unwrap();

    assert_eq!(holder.lorem, Lorem::OptDolor(None));
}

/// Make sure that newtype variants which don't produce a from_none value
/// do not allow the word form.
#[test]
#[should_panic]
fn word_only_fails_for_non_option() {
    Holder::from_derive_input(&parse_quote! {
        #[hello(lorem = "dolor")]
        struct Bar;
    })
    .unwrap();
}

#[test]
fn struct_nested() {
    let di = parse_quote! {
        #[hello(lorem(sit(world = "Hello", hello = false)))]
        pub struct Bar;
    };

    let pr = Holder::from_derive_input(&di).unwrap();
    assert_eq!(
        pr,
        Lorem::Sit(Amet {
            hello: false,
            world: "Hello".to_string(),
        })
    );
}

#[test]
#[should_panic]
fn format_mismatch() {
    let di = parse_quote! {
        #[hello(lorem(dolor(world = "Hello", hello = false)))]
        pub struct Bar;
    };

    Holder::from_derive_input(&di).unwrap();
}