mirror of
https://github.com/lennart-k/rustical.git
synced 2025-12-13 19:22:26 +00:00
xml: Also implement unit variants
This commit is contained in:
@@ -22,6 +22,7 @@ pub struct VariantAttrs {
|
||||
#[darling(attributes(xml))]
|
||||
pub struct EnumAttrs {
|
||||
pub untagged: Flag,
|
||||
pub unit_variants_name: Option<String>,
|
||||
}
|
||||
|
||||
#[derive(Default, FromDeriveInput, Clone)]
|
||||
|
||||
@@ -67,9 +67,21 @@ pub fn derive_enum_variants(input: proc_macro::TokenStream) -> proc_macro::Token
|
||||
let input = parse_macro_input!(input as DeriveInput);
|
||||
|
||||
match &input.data {
|
||||
syn::Data::Struct(_) => panic!("Struct not supported, use XmlRootTag instead"),
|
||||
syn::Data::Struct(_) => panic!("Struct not supported"),
|
||||
syn::Data::Enum(e) => Enum::parse(&input, e).impl_enum_variants(),
|
||||
syn::Data::Union(_) => panic!("Union not supported as root"),
|
||||
syn::Data::Union(_) => panic!("Union not supported"),
|
||||
}
|
||||
.into()
|
||||
}
|
||||
|
||||
#[proc_macro_derive(EnumUnitVariants, attributes(xml))]
|
||||
pub fn derive_enum_unit_variants(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
|
||||
let input = parse_macro_input!(input as DeriveInput);
|
||||
|
||||
match &input.data {
|
||||
syn::Data::Struct(_) => panic!("Struct not supported"),
|
||||
syn::Data::Enum(e) => Enum::parse(&input, e).impl_enum_unit_variants(),
|
||||
syn::Data::Union(_) => panic!("Union not supported"),
|
||||
}
|
||||
.into()
|
||||
}
|
||||
|
||||
@@ -2,6 +2,7 @@ use super::{attrs::EnumAttrs, Variant};
|
||||
use crate::attrs::VariantAttrs;
|
||||
use core::panic;
|
||||
use darling::{FromDeriveInput, FromVariant};
|
||||
use proc_macro2::Span;
|
||||
use quote::quote;
|
||||
use syn::{DataEnum, DeriveInput};
|
||||
|
||||
@@ -238,4 +239,74 @@ impl Enum {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
pub fn impl_enum_unit_variants(&self) -> proc_macro2::TokenStream {
|
||||
let ident = &self.ident;
|
||||
if self.attrs.untagged.is_present() {
|
||||
panic!("EnumUnitVariants not implemented for untagged enums");
|
||||
}
|
||||
let unit_enum_ident = if let Some(name) = &self.attrs.unit_variants_name {
|
||||
syn::Ident::new(name, Span::call_site())
|
||||
} else {
|
||||
panic!("unit_variants_name not set");
|
||||
};
|
||||
|
||||
let tagged_variants: Vec<_> = self
|
||||
.variants
|
||||
.iter()
|
||||
.filter(|variant| !variant.attrs.other.is_present())
|
||||
.collect();
|
||||
|
||||
let variant_outputs: Vec<_> = tagged_variants
|
||||
.iter()
|
||||
.map(|variant| {
|
||||
let ns = match &variant.attrs.common.ns {
|
||||
Some(ns) => quote! { Some(#ns) },
|
||||
None => quote! { None },
|
||||
};
|
||||
let b_xml_name = variant.xml_name().value();
|
||||
let xml_name = String::from_utf8_lossy(&b_xml_name);
|
||||
quote! {(#ns, #xml_name)}
|
||||
})
|
||||
.collect();
|
||||
|
||||
let variant_idents: Vec<_> = tagged_variants
|
||||
.iter()
|
||||
.map(|variant| &variant.variant.ident)
|
||||
.collect();
|
||||
|
||||
let unit_to_output_branches =
|
||||
variant_idents
|
||||
.iter()
|
||||
.zip(&variant_outputs)
|
||||
.map(|(variant_ident, out)| {
|
||||
quote! { #unit_enum_ident::#variant_ident => #out }
|
||||
});
|
||||
|
||||
let from_enum_to_unit_branches = variant_idents.iter().map(|variant_ident| {
|
||||
quote! { #ident::#variant_ident { .. } => #unit_enum_ident::#variant_ident }
|
||||
});
|
||||
|
||||
quote! {
|
||||
enum #unit_enum_ident {
|
||||
#(#variant_idents),*
|
||||
}
|
||||
|
||||
impl From<#unit_enum_ident> for (Option<::quick_xml::name::Namespace<'static>>, &'static str) {
|
||||
fn from(val: #unit_enum_ident) -> Self {
|
||||
match val {
|
||||
#(#unit_to_output_branches),*
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl From<#ident> for #unit_enum_ident {
|
||||
fn from(val: #ident) -> Self {
|
||||
match val {
|
||||
#(#from_enum_to_unit_branches),*
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -14,6 +14,7 @@ pub use se::XmlSerialize;
|
||||
pub use se::XmlSerializeRoot;
|
||||
pub use unparsed::Unparsed;
|
||||
pub use value::{ParseValueError, ValueDeserialize, ValueSerialize};
|
||||
pub use xml_derive::EnumUnitVariants;
|
||||
pub use xml_derive::EnumVariants;
|
||||
pub use xml_derive::XmlRootTag;
|
||||
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
use quick_xml::name::Namespace;
|
||||
use rustical_xml::EnumVariants;
|
||||
use xml_derive::EnumUnitVariants;
|
||||
|
||||
pub const NS_DAV: Namespace = Namespace(b"DAV:");
|
||||
pub const NS_DAVPUSH: Namespace = Namespace(b"https://bitfire.at/webdav-push");
|
||||
@@ -14,7 +15,8 @@ enum ExtensionProp {
|
||||
Hello,
|
||||
}
|
||||
|
||||
#[derive(EnumVariants)]
|
||||
#[derive(EnumVariants, EnumUnitVariants)]
|
||||
#[xml(unit_variants_name = "CalendarPropName")]
|
||||
enum CalendarProp {
|
||||
// WebDAV (RFC 2518)
|
||||
#[xml(ns = "NS_DAV")]
|
||||
@@ -60,3 +62,15 @@ fn test_enum_untagged_variants() {
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_enum_unit_variants() {
|
||||
let displayname: (Option<Namespace>, &str) = CalendarPropName::Displayname.into();
|
||||
assert_eq!(displayname, (Some(NS_DAV), "displayname"));
|
||||
let topic: (Option<Namespace>, &str) = CalendarPropName::Topic.into();
|
||||
assert_eq!(topic, (None, "topic"));
|
||||
|
||||
let propname: CalendarPropName = CalendarProp::Displayname(None).into();
|
||||
let displayname: (Option<Namespace>, &str) = propname.into();
|
||||
assert_eq!(displayname, (Some(NS_DAV), "displayname"));
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user