instant-xml/instant-xml-macros/src/ser.rs

175 lines
5.5 KiB
Rust
Raw Normal View History

use proc_macro2::TokenStream;
2022-09-07 20:32:24 +00:00
use quote::{quote, ToTokens};
use syn::spanned::Spanned;
use crate::Namespace;
use super::{discard_lifetimes, ContainerMeta, FieldMeta};
2022-09-05 11:14:31 +00:00
pub fn to_xml(input: &syn::DeriveInput) -> proc_macro2::TokenStream {
let meta = ContainerMeta::from_derive(input);
2022-09-05 11:14:31 +00:00
match &input.data {
syn::Data::Struct(ref data) => serialize_struct(input, data, meta),
2022-09-05 11:14:31 +00:00
_ => todo!(),
}
}
fn serialize_struct(
input: &syn::DeriveInput,
data: &syn::DataStruct,
meta: ContainerMeta,
) -> proc_macro2::TokenStream {
let mut body = TokenStream::new();
let mut attributes = TokenStream::new();
match data.fields {
syn::Fields::Named(ref fields) => {
fields.named.iter().for_each(|field| {
process_named_field(field, &mut body, &mut attributes, &meta);
});
}
syn::Fields::Unnamed(_) => todo!(),
syn::Fields::Unit => {}
2022-09-05 11:14:31 +00:00
};
let default_namespace = match &meta.ns.uri {
Some(ns) => quote!(#ns),
None => quote!(""),
};
let cx_len = meta.ns.prefixes.len();
let mut context = quote!(
let mut new = ::instant_xml::ser::Context::<#cx_len>::default();
new.default_ns = #default_namespace;
);
for (i, (prefix, ns)) in meta.ns.prefixes.iter().enumerate() {
context.extend(quote!(
new.prefixes[#i] = ::instant_xml::ser::Prefix { ns: #ns, prefix: #prefix };
));
}
2022-09-05 11:14:31 +00:00
let ident = &input.ident;
let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
2022-09-07 20:32:24 +00:00
let tag = match &meta.rename {
Some(rename) => quote!(#rename),
None => ident.to_string().into_token_stream(),
};
2022-09-05 11:14:31 +00:00
quote!(
impl #impl_generics ToXml for #ident #ty_generics #where_clause {
2022-09-05 11:14:31 +00:00
fn serialize<W: ::core::fmt::Write + ?::core::marker::Sized>(
&self,
serializer: &mut instant_xml::Serializer<W>,
) -> Result<(), instant_xml::Error> {
// Start tag
2022-09-07 20:32:24 +00:00
let prefix = serializer.write_start(#tag, #default_namespace, false)?;
debug_assert_eq!(prefix, None);
// Set up element context, this will also emit namespace declarations
#context
let old = serializer.push(new)?;
// Finalize start element
2022-09-05 11:14:31 +00:00
#attributes
serializer.end_start()?;
2022-09-05 11:14:31 +00:00
#body
// Close tag
2022-09-07 20:32:24 +00:00
serializer.write_close(prefix, #tag)?;
serializer.pop(old);
2022-09-05 11:14:31 +00:00
Ok(())
}
const KIND: ::instant_xml::Kind = ::instant_xml::Kind::Element(::instant_xml::Id {
ns: #default_namespace,
2022-09-07 20:32:24 +00:00
name: #tag,
});
2022-09-05 11:14:31 +00:00
};
)
}
fn process_named_field(
field: &syn::Field,
body: &mut TokenStream,
attributes: &mut TokenStream,
meta: &ContainerMeta,
) {
2022-09-07 20:32:24 +00:00
let field_name = field.ident.as_ref().unwrap();
let field_meta = FieldMeta::from_field(field);
2022-09-07 20:32:24 +00:00
let tag = match &field_meta.rename {
Some(rename) => quote!(#rename),
None => field_name.to_string().into_token_stream(),
};
let default_ns = &meta.ns.uri;
if field_meta.attribute {
let (ns, error) = match &field_meta.ns.uri {
Some(Namespace::Path(path)) => match path.get_ident() {
Some(prefix) => match &meta.ns.prefixes.get(&prefix.to_string()) {
Some(ns) => (quote!(#ns), quote!()),
None => (
quote!(""),
syn::Error::new(
field_meta.ns.uri.span(),
&format!("unknown prefix `{prefix}` (prefix must be defined on the field's type)"),
)
.into_compile_error(),
),
},
None => (
quote!(""),
syn::Error::new(
field_meta.ns.uri.span(),
"attribute namespace must be a prefix identifier",
)
.into_compile_error(),
),
},
Some(Namespace::Literal(_)) => (
quote!(""),
syn::Error::new(
field_meta.ns.uri.span(),
"attribute namespace must be a prefix identifier",
)
.into_compile_error(),
),
None => (match default_ns {
Some(ns) => quote!(#ns),
None => quote!(""),
}, quote!()),
};
attributes.extend(quote!(
#error
2022-09-07 20:32:24 +00:00
serializer.write_attr(#tag, #ns, &self.#field_name)?;
));
return;
}
let ns = match field_meta.ns.uri {
Some(ns) => quote!(#ns),
None => match &meta.ns.uri {
Some(ns) => quote!(#ns),
None => quote!(""),
},
};
let mut no_lifetime_type = field.ty.clone();
discard_lifetimes(&mut no_lifetime_type);
body.extend(quote!(
match <#no_lifetime_type as ToXml>::KIND {
::instant_xml::Kind::Element(_) => {
2022-09-07 20:32:24 +00:00
self.#field_name.serialize(serializer)?;
}
::instant_xml::Kind::Scalar => {
2022-09-07 20:32:24 +00:00
let prefix = serializer.write_start(#tag, #ns, true)?;
serializer.end_start()?;
2022-09-07 20:32:24 +00:00
self.#field_name.serialize(serializer)?;
serializer.write_close(prefix, #tag)?;
}
}
));
}