2019-02-22 11:16:11 +00:00
|
|
|
extern crate proc_macro;
|
|
|
|
|
|
|
|
use crate::proc_macro::TokenStream;
|
|
|
|
use quote::quote;
|
2019-02-22 14:54:18 +00:00
|
|
|
use syn::{parse_macro_input, DeriveInput};
|
2019-02-22 11:16:11 +00:00
|
|
|
|
2019-03-16 03:30:21 +00:00
|
|
|
/// Returns true if some field has an attribute declaring it should be generated from default (not
|
|
|
|
/// randomized).
|
|
|
|
///
|
|
|
|
/// The field attribute is: `#[test_random(default)]`
|
|
|
|
fn should_use_default(field: &syn::Field) -> bool {
|
2019-09-28 04:29:14 +00:00
|
|
|
field.attrs.iter().any(|attr| {
|
2022-02-25 00:10:17 +00:00
|
|
|
attr.path.is_ident("test_random") && attr.tokens.to_string().replace(' ', "") == "(default)"
|
2019-09-28 04:29:14 +00:00
|
|
|
})
|
2019-03-16 03:30:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[proc_macro_derive(TestRandom, attributes(test_random))]
|
2019-02-22 11:16:11 +00:00
|
|
|
pub fn test_random_derive(input: TokenStream) -> TokenStream {
|
2019-02-22 16:10:26 +00:00
|
|
|
let derived_input = parse_macro_input!(input as DeriveInput);
|
|
|
|
let name = &derived_input.ident;
|
2019-05-08 03:08:37 +00:00
|
|
|
let (impl_generics, ty_generics, where_clause) = &derived_input.generics.split_for_impl();
|
2019-02-22 11:16:11 +00:00
|
|
|
|
2019-02-22 16:10:26 +00:00
|
|
|
let struct_data = match &derived_input.data {
|
2019-02-22 14:54:18 +00:00
|
|
|
syn::Data::Struct(s) => s,
|
|
|
|
_ => panic!("test_random_derive only supports structs."),
|
|
|
|
};
|
|
|
|
|
2019-03-16 03:30:21 +00:00
|
|
|
// Build quotes for fields that should be generated and those that should be built from
|
|
|
|
// `Default`.
|
|
|
|
let mut quotes = vec![];
|
|
|
|
for field in &struct_data.fields {
|
|
|
|
match &field.ident {
|
|
|
|
Some(ref ident) => {
|
|
|
|
if should_use_default(field) {
|
|
|
|
quotes.push(quote! {
|
|
|
|
#ident: <_>::default(),
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
quotes.push(quote! {
|
|
|
|
#ident: <_>::random_for_test(rng),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => panic!("test_random_derive only supports named struct fields."),
|
|
|
|
};
|
|
|
|
}
|
2019-02-22 14:54:18 +00:00
|
|
|
|
|
|
|
let output = quote! {
|
2019-05-08 03:08:37 +00:00
|
|
|
impl #impl_generics TestRandom for #name #ty_generics #where_clause {
|
|
|
|
fn random_for_test(rng: &mut impl rand::RngCore) -> Self {
|
2019-02-22 14:54:18 +00:00
|
|
|
Self {
|
|
|
|
#(
|
2019-03-16 03:30:21 +00:00
|
|
|
#quotes
|
2019-02-22 14:54:18 +00:00
|
|
|
)*
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
output.into()
|
2019-02-22 11:16:11 +00:00
|
|
|
}
|