216 lines
7 KiB
Rust
216 lines
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/.
|
||
|
|
||
|
//! Compound (struct or enum variant) field types
|
||
|
|
||
|
use proc_macro2::TokenStream;
|
||
|
use quote::quote;
|
||
|
use syn::{spanned::Spanned, *};
|
||
|
|
||
|
use rxml_validation::NcName;
|
||
|
|
||
|
use crate::error_message::{self, ParentRef};
|
||
|
use crate::meta::{NameRef, XmlFieldMeta};
|
||
|
use crate::scope::{FromEventsScope, IntoEventsScope};
|
||
|
|
||
|
/// Code slices necessary for declaring and initializing a temporary variable
|
||
|
/// for parsing purposes.
|
||
|
pub(crate) struct FieldTempInit {
|
||
|
/// The type of the temporary variable.
|
||
|
pub(crate) ty: Type,
|
||
|
|
||
|
/// The initializer for the temporary variable.
|
||
|
pub(crate) init: TokenStream,
|
||
|
}
|
||
|
|
||
|
/// Describe how a struct or enum variant's member is parsed from XML data.
|
||
|
///
|
||
|
/// This struct is returned from [`FieldDef::make_builder_part`] and
|
||
|
/// contains code snippets and instructions for
|
||
|
/// [`Compound::make_from_events_statemachine`][`crate::compound::Compound::make_from_events_statemachine`]
|
||
|
/// to parse the field's data from XML.
|
||
|
pub(crate) enum FieldBuilderPart {
|
||
|
/// Parse a field from the item's element's start event.
|
||
|
Init {
|
||
|
/// Expression and type which extracts the field's data from the
|
||
|
/// element's start event.
|
||
|
value: FieldTempInit,
|
||
|
},
|
||
|
}
|
||
|
|
||
|
/// Describe how a struct or enum variant's member is converted to XML data.
|
||
|
///
|
||
|
/// This struct is returned from [`FieldDef::make_iterator_part`] and
|
||
|
/// contains code snippets and instructions for
|
||
|
/// [`Compound::make_into_events_statemachine`][`crate::compound::Compound::make_into_events_statemachine`]
|
||
|
/// to convert the field's data into XML.
|
||
|
pub(crate) enum FieldIteratorPart {
|
||
|
/// The field is emitted as part of StartElement.
|
||
|
Header {
|
||
|
/// A sequence of statements which updates the temporary variables
|
||
|
/// during the StartElement event's construction, consuming the
|
||
|
/// field's value.
|
||
|
setter: TokenStream,
|
||
|
},
|
||
|
}
|
||
|
|
||
|
/// Specify how the field is mapped to XML.
|
||
|
enum FieldKind {
|
||
|
/// The field maps to an attribute.
|
||
|
Attribute {
|
||
|
/// The XML name of the attribute.
|
||
|
xml_name: NameRef,
|
||
|
},
|
||
|
}
|
||
|
|
||
|
impl FieldKind {
|
||
|
/// Construct a new field implementation from the meta attributes.
|
||
|
///
|
||
|
/// `field_ident` is, for some field types, used to infer an XML name if
|
||
|
/// it is not specified explicitly.
|
||
|
fn from_meta(meta: XmlFieldMeta, field_ident: Option<&Ident>) -> Result<Self> {
|
||
|
match meta {
|
||
|
XmlFieldMeta::Attribute { span } => {
|
||
|
let Some(field_ident) = field_ident else {
|
||
|
return Err(Error::new(
|
||
|
span,
|
||
|
"attribute extraction not supported on unnamed fields",
|
||
|
));
|
||
|
};
|
||
|
|
||
|
let xml_name = match NcName::try_from(field_ident.to_string()) {
|
||
|
Ok(v) => v,
|
||
|
Err(e) => {
|
||
|
return Err(Error::new(
|
||
|
field_ident.span(),
|
||
|
format!("invalid XML attribute name: {}", e),
|
||
|
))
|
||
|
}
|
||
|
};
|
||
|
|
||
|
Ok(Self::Attribute {
|
||
|
xml_name: NameRef::Literal {
|
||
|
span: field_ident.span(),
|
||
|
value: xml_name,
|
||
|
},
|
||
|
})
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// Definition of a single field in a compound.
|
||
|
///
|
||
|
/// See [`Compound`][`crate::compound::Compound`] for more information on
|
||
|
/// compounds in general.
|
||
|
pub(crate) struct FieldDef {
|
||
|
/// The member identifying the field.
|
||
|
member: Member,
|
||
|
|
||
|
/// The type of the field.
|
||
|
ty: Type,
|
||
|
|
||
|
/// The way the field is mapped to XML.
|
||
|
kind: FieldKind,
|
||
|
}
|
||
|
|
||
|
impl FieldDef {
|
||
|
/// Create a new field definition from its declaration.
|
||
|
///
|
||
|
/// The `index` must be the zero-based index of the field even for named
|
||
|
/// fields.
|
||
|
pub(crate) fn from_field(field: &syn::Field, index: u32) -> Result<Self> {
|
||
|
let field_span = field.span();
|
||
|
let meta = XmlFieldMeta::parse_from_attributes(&field.attrs, &field_span)?;
|
||
|
|
||
|
let (member, ident) = match field.ident.as_ref() {
|
||
|
Some(v) => (Member::Named(v.clone()), Some(v)),
|
||
|
None => (
|
||
|
Member::Unnamed(Index {
|
||
|
index,
|
||
|
span: field_span,
|
||
|
}),
|
||
|
None,
|
||
|
),
|
||
|
};
|
||
|
|
||
|
let ty = field.ty.clone();
|
||
|
|
||
|
Ok(Self {
|
||
|
member,
|
||
|
ty,
|
||
|
kind: FieldKind::from_meta(meta, ident)?,
|
||
|
})
|
||
|
}
|
||
|
|
||
|
/// Access the [`syn::Member`] identifying this field in the original
|
||
|
/// type.
|
||
|
pub(crate) fn member(&self) -> &Member {
|
||
|
&self.member
|
||
|
}
|
||
|
|
||
|
/// Access the field's type.
|
||
|
pub(crate) fn ty(&self) -> &Type {
|
||
|
&self.ty
|
||
|
}
|
||
|
|
||
|
/// Construct the builder pieces for this field.
|
||
|
///
|
||
|
/// `container_name` must be a reference to the compound's type, so that
|
||
|
/// it can be used for error messages.
|
||
|
pub(crate) fn make_builder_part(
|
||
|
&self,
|
||
|
scope: &FromEventsScope,
|
||
|
container_name: &ParentRef,
|
||
|
) -> Result<FieldBuilderPart> {
|
||
|
match self.kind {
|
||
|
FieldKind::Attribute { ref xml_name } => {
|
||
|
let FromEventsScope { ref attrs, .. } = scope;
|
||
|
|
||
|
let missing_msg = error_message::on_missing_attribute(container_name, &self.member);
|
||
|
|
||
|
return Ok(FieldBuilderPart::Init {
|
||
|
value: FieldTempInit {
|
||
|
ty: self.ty.clone(),
|
||
|
init: quote! {
|
||
|
match #attrs.remove(::xso::exports::rxml::Namespace::none(), #xml_name) {
|
||
|
::core::option::Option::Some(v) => v,
|
||
|
::core::option::Option::None => return ::core::result::Result::Err(::xso::error::Error::Other(#missing_msg).into()),
|
||
|
}
|
||
|
},
|
||
|
},
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// Construct the iterator pieces for this field.
|
||
|
///
|
||
|
/// `bound_name` must be the name to which the field's value is bound in
|
||
|
/// the iterator code.
|
||
|
pub(crate) fn make_iterator_part(
|
||
|
&self,
|
||
|
scope: &IntoEventsScope,
|
||
|
bound_name: &Ident,
|
||
|
) -> Result<FieldIteratorPart> {
|
||
|
match self.kind {
|
||
|
FieldKind::Attribute { ref xml_name } => {
|
||
|
let IntoEventsScope { ref attrs, .. } = scope;
|
||
|
|
||
|
return Ok(FieldIteratorPart::Header {
|
||
|
setter: quote! {
|
||
|
#attrs.insert(
|
||
|
::xso::exports::rxml::Namespace::NONE,
|
||
|
#xml_name.to_owned(),
|
||
|
#bound_name,
|
||
|
);
|
||
|
},
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|