1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
extern crate proc_macro; use proc_macro::TokenStream; use quote::quote; use syn::punctuated::Punctuated; use syn::token::Comma; use syn::{ parse_macro_input, DeriveInput, GenericParam, TypeParamBound, Generics, }; macro_rules! derive_as_json { ($tr:path, $name:ident, $generics:ident, $params:ident, $function:path) => { quote! { impl<#$params> $tr for #$name #$generics { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { let s = $function(&self) .map_err(|_| std::fmt::Error::default())?; write!(f, "{}", s) } } } }; } fn serialize_in_generics( g: &Generics, ) -> Punctuated<GenericParam, Comma> { let mut params = Punctuated::<GenericParam, Comma>::new(); for param in g.params.iter() { let param = match param { GenericParam::Type(typ) => { let mut typ = typ.clone(); typ.bounds.push(TypeParamBound::Trait( syn::parse_str("display_json::serde::Serialize").unwrap(), )); GenericParam::Type(typ.clone()) } _ => param.clone(), }; params.push(param); } params } #[proc_macro_derive(DisplayAsJson)] pub fn derive_display_as_json(input: TokenStream) -> TokenStream { let input = parse_macro_input!(input as DeriveInput); let name = &input.ident; let generics = &input.generics; let params = serialize_in_generics(generics); let result = derive_as_json!( std::fmt::Display, name, generics, params, display_json::serde_json::to_string ); TokenStream::from(result) } #[proc_macro_derive(DisplayAsJsonPretty)] pub fn derive_display_as_json_pretty( input: TokenStream, ) -> TokenStream { let input = parse_macro_input!(input as DeriveInput); let name = &input.ident; let generics = &input.generics; let params = serialize_in_generics(generics); let result = derive_as_json!( std::fmt::Display, name, generics, params, display_json::serde_json::to_string_pretty ); TokenStream::from(result) } #[proc_macro_derive(DebugAsJson)] pub fn derive_debug_as_json(input: TokenStream) -> TokenStream { let input = parse_macro_input!(input as DeriveInput); let name = &input.ident; let generics = &input.generics; let params = serialize_in_generics(generics); let result = derive_as_json!( std::fmt::Debug, name, generics, params, display_json::serde_json::to_string ); TokenStream::from(result) } #[proc_macro_derive(DebugAsJsonPretty)] pub fn derive_debug_as_json_pretty( input: TokenStream, ) -> TokenStream { let input = parse_macro_input!(input as DeriveInput); let name = &input.ident; let generics = &input.generics; let params = serialize_in_generics(generics); let result = derive_as_json!( std::fmt::Debug, name, generics, params, display_json::serde_json::to_string_pretty ); TokenStream::from(result) }