blob: c1e56d91b540ae5c5f1490aad09f79d4a543b9e0 [file] [log] [blame]
//! Derive `Serialize` and `Deserialize` that delegates to the underlying repr
//! of a C-like enum.
//!
//! # Examples
//!
//! ```
//! use serde_repr::{Serialize_repr, Deserialize_repr};
//!
//! #[derive(Serialize_repr, Deserialize_repr, PartialEq, Debug)]
//! #[repr(u8)]
//! enum SmallPrime {
//! Two = 2,
//! Three = 3,
//! Five = 5,
//! Seven = 7,
//! }
//!
//! fn main() -> serde_json::Result<()> {
//! let j = serde_json::to_string(&SmallPrime::Seven)?;
//! assert_eq!(j, "7");
//!
//! let p: SmallPrime = serde_json::from_str("2")?;
//! assert_eq!(p, SmallPrime::Two);
//!
//! Ok(())
//! }
//! ```
#![recursion_limit = "128"]
extern crate proc_macro;
mod parse;
use proc_macro::TokenStream;
use quote::quote;
use syn::parse_macro_input;
use crate::parse::Input;
use std::iter;
#[proc_macro_derive(Serialize_repr)]
pub fn derive_serialize(input: TokenStream) -> TokenStream {
let input = parse_macro_input!(input as Input);
let ident = input.ident;
let repr = input.repr;
let match_variants = input.variants.iter().map(|variant| {
let variant = &variant.ident;
quote! {
#ident::#variant => #ident::#variant as #repr,
}
});
TokenStream::from(quote! {
impl serde::Serialize for #ident {
fn serialize<S>(&self, serializer: S) -> core::result::Result<S::Ok, S::Error>
where
S: serde::Serializer
{
let value: #repr = match *self {
#(#match_variants)*
};
serde::Serialize::serialize(&value, serializer)
}
}
})
}
#[proc_macro_derive(Deserialize_repr)]
pub fn derive_deserialize(input: TokenStream) -> TokenStream {
let input = parse_macro_input!(input as Input);
let ident = input.ident;
let repr = input.repr;
let variants = input.variants.iter().map(|variant| &variant.ident);
let declare_discriminants = input.variants.iter().map(|variant| {
let variant = &variant.ident;
quote! {
#[allow(non_upper_case_globals)]
const #variant: #repr = #ident::#variant as #repr;
}
});
let match_discriminants = input.variants.iter().map(|variant| {
let variant = &variant.ident;
quote! {
discriminant::#variant => core::result::Result::Ok(#ident::#variant),
}
});
let error_format = match input.variants.len() {
1 => "invalid value: {}, expected {}".to_owned(),
2 => "invalid value: {}, expected {} or {}".to_owned(),
n => {
"invalid value: {}, expected one of: {}".to_owned()
+ &iter::repeat(", {}").take(n - 1).collect::<String>()
}
};
TokenStream::from(quote! {
impl<'de> serde::Deserialize<'de> for #ident {
fn deserialize<D>(deserializer: D) -> core::result::Result<Self, D::Error>
where
D: serde::Deserializer<'de>,
{
struct discriminant;
impl discriminant {
#(#declare_discriminants)*
}
match <#repr as serde::Deserialize>::deserialize(deserializer)? {
#(#match_discriminants)*
other => core::result::Result::Err(serde::de::Error::custom(
format_args!(#error_format, other #(, discriminant::#variants)*)
)),
}
}
}
})
}