320 lines
9.7 KiB
Rust
320 lines
9.7 KiB
Rust
|
// Copyright (c) 2024 Jonas Schäfer <jonas@zombofant.net>
|
||
|
//
|
||
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
||
|
// License, v. 2.0. If a copy of the MPL was not distributed with this
|
||
|
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
||
|
|
||
|
//! Handling of enums
|
||
|
|
||
|
use std::collections::HashMap;
|
||
|
|
||
|
use proc_macro2::Span;
|
||
|
use quote::quote;
|
||
|
use syn::*;
|
||
|
|
||
|
use crate::common::{AsXmlParts, FromXmlParts, ItemDef};
|
||
|
use crate::compound::Compound;
|
||
|
use crate::error_message::ParentRef;
|
||
|
use crate::meta::{NameRef, NamespaceRef, XmlCompoundMeta};
|
||
|
use crate::state::{AsItemsStateMachine, FromEventsStateMachine};
|
||
|
|
||
|
/// The definition of an enum variant, switched on the XML element's name.
|
||
|
struct NameVariant {
|
||
|
/// The XML name of the element to map the enum variant to.
|
||
|
name: NameRef,
|
||
|
|
||
|
/// The name of the variant
|
||
|
ident: Ident,
|
||
|
|
||
|
/// The field(s) of this struct.
|
||
|
inner: Compound,
|
||
|
}
|
||
|
|
||
|
impl NameVariant {
|
||
|
/// Construct a new name-selected variant from its declaration.
|
||
|
fn new(decl: &Variant) -> Result<Self> {
|
||
|
let meta = XmlCompoundMeta::parse_from_attributes(&decl.attrs)?;
|
||
|
|
||
|
if let Some(namespace) = meta.namespace {
|
||
|
return Err(Error::new_spanned(
|
||
|
namespace,
|
||
|
"`namespace` is not allowed on enum variants (only on enums and structs)",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
let Some(name) = meta.name else {
|
||
|
return Err(Error::new(meta.span, "`name` is required on enum variants"));
|
||
|
};
|
||
|
|
||
|
Ok(Self {
|
||
|
name,
|
||
|
ident: decl.ident.clone(),
|
||
|
inner: Compound::from_fields(&decl.fields)?,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
fn make_from_events_statemachine(
|
||
|
&self,
|
||
|
enum_ident: &Ident,
|
||
|
state_ty_ident: &Ident,
|
||
|
) -> Result<FromEventsStateMachine> {
|
||
|
let xml_name = &self.name;
|
||
|
|
||
|
Ok(self
|
||
|
.inner
|
||
|
.make_from_events_statemachine(
|
||
|
state_ty_ident,
|
||
|
&ParentRef::Named(Path {
|
||
|
leading_colon: None,
|
||
|
segments: [
|
||
|
PathSegment::from(enum_ident.clone()),
|
||
|
self.ident.clone().into(),
|
||
|
]
|
||
|
.into_iter()
|
||
|
.collect(),
|
||
|
}),
|
||
|
&self.ident.to_string(),
|
||
|
)?
|
||
|
.with_augmented_init(|init| {
|
||
|
quote! {
|
||
|
if name.1 != #xml_name {
|
||
|
::core::result::Result::Err(::xso::error::FromEventsError::Mismatch {
|
||
|
name,
|
||
|
attrs,
|
||
|
})
|
||
|
} else {
|
||
|
#init
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
.compile())
|
||
|
}
|
||
|
|
||
|
fn make_as_item_iter_statemachine(
|
||
|
&self,
|
||
|
xml_namespace: &NamespaceRef,
|
||
|
enum_ident: &Ident,
|
||
|
item_iter_ty_lifetime: &Lifetime,
|
||
|
) -> Result<AsItemsStateMachine> {
|
||
|
let xml_name = &self.name;
|
||
|
|
||
|
Ok(self
|
||
|
.inner
|
||
|
.make_as_item_iter_statemachine(
|
||
|
&ParentRef::Named(Path {
|
||
|
leading_colon: None,
|
||
|
segments: [
|
||
|
PathSegment::from(enum_ident.clone()),
|
||
|
self.ident.clone().into(),
|
||
|
]
|
||
|
.into_iter()
|
||
|
.collect(),
|
||
|
}),
|
||
|
&self.ident.to_string(),
|
||
|
&item_iter_ty_lifetime,
|
||
|
)?
|
||
|
.with_augmented_init(|init| {
|
||
|
quote! {
|
||
|
let name = (
|
||
|
::xso::exports::rxml::Namespace::from(#xml_namespace),
|
||
|
::std::borrow::Cow::Borrowed(#xml_name),
|
||
|
);
|
||
|
#init
|
||
|
}
|
||
|
})
|
||
|
.compile())
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// Definition of an enum and how to parse it.
|
||
|
pub(crate) struct EnumDef {
|
||
|
/// The XML namespace of the element to map the enum to.
|
||
|
namespace: NamespaceRef,
|
||
|
|
||
|
/// The variants of the enum.
|
||
|
variants: Vec<NameVariant>,
|
||
|
|
||
|
/// Name of the target type.
|
||
|
target_ty_ident: Ident,
|
||
|
|
||
|
/// Name of the builder type.
|
||
|
builder_ty_ident: Ident,
|
||
|
|
||
|
/// Name of the iterator type.
|
||
|
item_iter_ty_ident: Ident,
|
||
|
|
||
|
/// Flag whether debug mode is enabled.
|
||
|
debug: bool,
|
||
|
}
|
||
|
|
||
|
impl EnumDef {
|
||
|
/// Create a new enum from its name, meta, and variants.
|
||
|
pub(crate) fn new<'x, I: IntoIterator<Item = &'x Variant>>(
|
||
|
ident: &Ident,
|
||
|
meta: XmlCompoundMeta,
|
||
|
variant_iter: I,
|
||
|
) -> Result<Self> {
|
||
|
if let Some(name) = meta.name {
|
||
|
return Err(Error::new_spanned(
|
||
|
name,
|
||
|
"`name` is not allowed on enums (only on their variants)",
|
||
|
));
|
||
|
}
|
||
|
|
||
|
let Some(namespace) = meta.namespace else {
|
||
|
return Err(Error::new(meta.span, "`namespace` is required on enums"));
|
||
|
};
|
||
|
|
||
|
let mut variants = Vec::new();
|
||
|
let mut seen_names = HashMap::new();
|
||
|
for variant in variant_iter {
|
||
|
let variant = NameVariant::new(variant)?;
|
||
|
if let Some(other) = seen_names.get(&variant.name) {
|
||
|
return Err(Error::new_spanned(
|
||
|
variant.name,
|
||
|
format!(
|
||
|
"duplicate `name` in enum: variants {} and {} have the same XML name",
|
||
|
other, variant.ident
|
||
|
),
|
||
|
));
|
||
|
}
|
||
|
seen_names.insert(variant.name.clone(), variant.ident.clone());
|
||
|
variants.push(variant);
|
||
|
}
|
||
|
|
||
|
let builder_ty_ident = match meta.builder {
|
||
|
Some(v) => v,
|
||
|
None => quote::format_ident!("{}FromXmlBuilder", ident.to_string()),
|
||
|
};
|
||
|
|
||
|
let item_iter_ty_ident = match meta.iterator {
|
||
|
Some(v) => v,
|
||
|
None => quote::format_ident!("{}AsXmlIterator", ident.to_string()),
|
||
|
};
|
||
|
|
||
|
Ok(Self {
|
||
|
namespace,
|
||
|
variants,
|
||
|
target_ty_ident: ident.clone(),
|
||
|
builder_ty_ident,
|
||
|
item_iter_ty_ident,
|
||
|
debug: meta.debug.is_set(),
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
|
||
|
impl ItemDef for EnumDef {
|
||
|
fn make_from_events_builder(
|
||
|
&self,
|
||
|
vis: &Visibility,
|
||
|
name_ident: &Ident,
|
||
|
attrs_ident: &Ident,
|
||
|
) -> Result<FromXmlParts> {
|
||
|
let xml_namespace = &self.namespace;
|
||
|
let target_ty_ident = &self.target_ty_ident;
|
||
|
let builder_ty_ident = &self.builder_ty_ident;
|
||
|
let state_ty_ident = quote::format_ident!("{}State", builder_ty_ident);
|
||
|
|
||
|
let mut statemachine = FromEventsStateMachine::new();
|
||
|
for variant in self.variants.iter() {
|
||
|
statemachine
|
||
|
.merge(variant.make_from_events_statemachine(target_ty_ident, &state_ty_ident)?);
|
||
|
}
|
||
|
|
||
|
statemachine.set_pre_init(quote! {
|
||
|
if name.0 != #xml_namespace {
|
||
|
return ::core::result::Result::Err(::xso::error::FromEventsError::Mismatch {
|
||
|
name,
|
||
|
attrs,
|
||
|
})
|
||
|
}
|
||
|
});
|
||
|
|
||
|
let defs = statemachine.render(
|
||
|
vis,
|
||
|
builder_ty_ident,
|
||
|
&state_ty_ident,
|
||
|
&TypePath {
|
||
|
qself: None,
|
||
|
path: target_ty_ident.clone().into(),
|
||
|
}
|
||
|
.into(),
|
||
|
)?;
|
||
|
|
||
|
Ok(FromXmlParts {
|
||
|
defs,
|
||
|
from_events_body: quote! {
|
||
|
#builder_ty_ident::new(#name_ident, #attrs_ident)
|
||
|
},
|
||
|
builder_ty_ident: builder_ty_ident.clone(),
|
||
|
})
|
||
|
}
|
||
|
|
||
|
fn make_as_xml_iter(&self, vis: &Visibility) -> Result<AsXmlParts> {
|
||
|
let target_ty_ident = &self.target_ty_ident;
|
||
|
let item_iter_ty_ident = &self.item_iter_ty_ident;
|
||
|
let item_iter_ty_lifetime = Lifetime {
|
||
|
apostrophe: Span::call_site(),
|
||
|
ident: Ident::new("xso_proc_as_xml_iter_lifetime", Span::call_site()),
|
||
|
};
|
||
|
let item_iter_ty = Type::Path(TypePath {
|
||
|
qself: None,
|
||
|
path: Path {
|
||
|
leading_colon: None,
|
||
|
segments: [PathSegment {
|
||
|
ident: item_iter_ty_ident.clone(),
|
||
|
arguments: PathArguments::AngleBracketed(AngleBracketedGenericArguments {
|
||
|
colon2_token: None,
|
||
|
lt_token: token::Lt {
|
||
|
spans: [Span::call_site()],
|
||
|
},
|
||
|
args: [GenericArgument::Lifetime(item_iter_ty_lifetime.clone())]
|
||
|
.into_iter()
|
||
|
.collect(),
|
||
|
gt_token: token::Gt {
|
||
|
spans: [Span::call_site()],
|
||
|
},
|
||
|
}),
|
||
|
}]
|
||
|
.into_iter()
|
||
|
.collect(),
|
||
|
},
|
||
|
});
|
||
|
let state_ty_ident = quote::format_ident!("{}State", item_iter_ty_ident);
|
||
|
|
||
|
let mut statemachine = AsItemsStateMachine::new();
|
||
|
for variant in self.variants.iter() {
|
||
|
statemachine.merge(variant.make_as_item_iter_statemachine(
|
||
|
&self.namespace,
|
||
|
target_ty_ident,
|
||
|
&item_iter_ty_lifetime,
|
||
|
)?);
|
||
|
}
|
||
|
|
||
|
let defs = statemachine.render(
|
||
|
vis,
|
||
|
&TypePath {
|
||
|
qself: None,
|
||
|
path: target_ty_ident.clone().into(),
|
||
|
}
|
||
|
.into(),
|
||
|
&state_ty_ident,
|
||
|
&item_iter_ty_lifetime,
|
||
|
&item_iter_ty,
|
||
|
)?;
|
||
|
|
||
|
Ok(AsXmlParts {
|
||
|
defs,
|
||
|
as_xml_iter_body: quote! {
|
||
|
#item_iter_ty_ident::new(self)
|
||
|
},
|
||
|
item_iter_ty,
|
||
|
item_iter_ty_lifetime,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
fn debug(&self) -> bool {
|
||
|
self.debug
|
||
|
}
|
||
|
}
|