mirror of
https://github.com/lennart-k/rustical.git
synced 2025-12-14 05:52:19 +00:00
xml: Also implement unit variants
This commit is contained in:
@@ -22,6 +22,7 @@ pub struct VariantAttrs {
|
|||||||
#[darling(attributes(xml))]
|
#[darling(attributes(xml))]
|
||||||
pub struct EnumAttrs {
|
pub struct EnumAttrs {
|
||||||
pub untagged: Flag,
|
pub untagged: Flag,
|
||||||
|
pub unit_variants_name: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Default, FromDeriveInput, Clone)]
|
#[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);
|
let input = parse_macro_input!(input as DeriveInput);
|
||||||
|
|
||||||
match &input.data {
|
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::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()
|
.into()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ use super::{attrs::EnumAttrs, Variant};
|
|||||||
use crate::attrs::VariantAttrs;
|
use crate::attrs::VariantAttrs;
|
||||||
use core::panic;
|
use core::panic;
|
||||||
use darling::{FromDeriveInput, FromVariant};
|
use darling::{FromDeriveInput, FromVariant};
|
||||||
|
use proc_macro2::Span;
|
||||||
use quote::quote;
|
use quote::quote;
|
||||||
use syn::{DataEnum, DeriveInput};
|
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 se::XmlSerializeRoot;
|
||||||
pub use unparsed::Unparsed;
|
pub use unparsed::Unparsed;
|
||||||
pub use value::{ParseValueError, ValueDeserialize, ValueSerialize};
|
pub use value::{ParseValueError, ValueDeserialize, ValueSerialize};
|
||||||
|
pub use xml_derive::EnumUnitVariants;
|
||||||
pub use xml_derive::EnumVariants;
|
pub use xml_derive::EnumVariants;
|
||||||
pub use xml_derive::XmlRootTag;
|
pub use xml_derive::XmlRootTag;
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
use quick_xml::name::Namespace;
|
use quick_xml::name::Namespace;
|
||||||
use rustical_xml::EnumVariants;
|
use rustical_xml::EnumVariants;
|
||||||
|
use xml_derive::EnumUnitVariants;
|
||||||
|
|
||||||
pub const NS_DAV: Namespace = Namespace(b"DAV:");
|
pub const NS_DAV: Namespace = Namespace(b"DAV:");
|
||||||
pub const NS_DAVPUSH: Namespace = Namespace(b"https://bitfire.at/webdav-push");
|
pub const NS_DAVPUSH: Namespace = Namespace(b"https://bitfire.at/webdav-push");
|
||||||
@@ -14,7 +15,8 @@ enum ExtensionProp {
|
|||||||
Hello,
|
Hello,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(EnumVariants)]
|
#[derive(EnumVariants, EnumUnitVariants)]
|
||||||
|
#[xml(unit_variants_name = "CalendarPropName")]
|
||||||
enum CalendarProp {
|
enum CalendarProp {
|
||||||
// WebDAV (RFC 2518)
|
// WebDAV (RFC 2518)
|
||||||
#[xml(ns = "NS_DAV")]
|
#[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