2023-06-13 22:56:15 +00:00
|
|
|
use proc_macro::TokenStream;
|
|
|
|
use quote::quote;
|
|
|
|
use syn::{parse_macro_input, ItemStruct};
|
|
|
|
|
|
|
|
#[proc_macro_derive(OptionalOptionalUser)]
|
|
|
|
pub fn derive_optional_optional_user(input: TokenStream) -> TokenStream {
|
|
|
|
let input = parse_macro_input!(input as ItemStruct);
|
2023-06-14 22:44:54 +00:00
|
|
|
let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
|
2023-06-13 22:56:15 +00:00
|
|
|
let name = &input.ident;
|
|
|
|
let has_user = input
|
|
|
|
.fields
|
|
|
|
.iter()
|
|
|
|
.find(|f| {
|
2023-06-14 22:44:54 +00:00
|
|
|
if let Some(ref ident) = f.ident {
|
2023-06-13 22:56:15 +00:00
|
|
|
ident == "user"
|
|
|
|
} else {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.is_some();
|
|
|
|
|
2023-06-14 22:44:54 +00:00
|
|
|
let (use_any, user_is_option_user) = if has_user {
|
|
|
|
(
|
|
|
|
quote!(
|
|
|
|
use ::std::any::Any;
|
|
|
|
),
|
|
|
|
quote!(
|
|
|
|
::std::any::TypeId::of::<::std::option::Option<crate::User>>()
|
|
|
|
== self.user.type_id()
|
|
|
|
),
|
2023-06-13 22:56:15 +00:00
|
|
|
)
|
|
|
|
} else {
|
2023-06-14 22:44:54 +00:00
|
|
|
(quote!(), quote!(false))
|
2023-06-13 22:56:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
let output = quote!(
|
2023-06-14 22:44:54 +00:00
|
|
|
impl #impl_generics #name #ty_generics #where_clause {
|
|
|
|
pub fn has_optional_user(&self) -> bool {
|
|
|
|
#use_any
|
|
|
|
#user_is_option_user
|
2023-06-13 22:56:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
TokenStream::from(output)
|
|
|
|
}
|