xml: Also implement unit variants

This commit is contained in:
Lennart
2025-01-18 19:35:19 +01:00
parent ea9f5a711d
commit 39beee2f69
5 changed files with 102 additions and 3 deletions

View File

@@ -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)]

View File

@@ -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()
}

View File

@@ -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),*
}
}
}
}
}
}

View File

@@ -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;

View File

@@ -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"));
}