mirror of
https://github.com/lennart-k/rustical.git
synced 2025-12-14 11:42:25 +00:00
xml: Work on struct serialization
This commit is contained in:
@@ -258,11 +258,10 @@ impl Field {
|
|||||||
writer.write_event(Event::Text(BytesText::new(&self.#field_ident)))?;
|
writer.write_event(Event::Text(BytesText::new(&self.#field_ident)))?;
|
||||||
}),
|
}),
|
||||||
FieldType::Tag => Some(quote! {
|
FieldType::Tag => Some(quote! {
|
||||||
self.#field_ident.serialize(Some(#field_name), writer)?;
|
self.#field_ident.serialize(None, Some(#field_name), writer)?;
|
||||||
}),
|
}),
|
||||||
FieldType::Untagged => Some(quote! {
|
FieldType::Untagged => Some(quote! {
|
||||||
// TODO: untag!
|
self.#field_ident.serialize(None, None, writer)?;
|
||||||
self.#field_ident.serialize(None, writer)?;
|
|
||||||
}),
|
}),
|
||||||
// TODO: Think about what to do here
|
// TODO: Think about what to do here
|
||||||
FieldType::TagName | FieldType::Namespace => None,
|
FieldType::TagName | FieldType::Namespace => None,
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
use crate::attrs::StructAttrs;
|
use crate::attrs::{FieldType, StructAttrs};
|
||||||
use crate::Field;
|
use crate::{field, Field};
|
||||||
use core::panic;
|
use core::panic;
|
||||||
use darling::FromDeriveInput;
|
use darling::FromDeriveInput;
|
||||||
use quote::quote;
|
use quote::quote;
|
||||||
@@ -181,11 +181,34 @@ impl NamedStruct {
|
|||||||
let ident = &self.ident;
|
let ident = &self.ident;
|
||||||
let tag_writers = self.fields.iter().map(Field::tag_writer);
|
let tag_writers = self.fields.iter().map(Field::tag_writer);
|
||||||
|
|
||||||
// TODO: Implement attributes
|
let untagged_attributes = self
|
||||||
|
.fields
|
||||||
|
.iter()
|
||||||
|
.filter(|field| field.attrs.xml_ty == FieldType::Untagged)
|
||||||
|
.map(|field| {
|
||||||
|
let field_ident = field.field_ident();
|
||||||
|
quote! { bytes_start.extend_attributes(self.#field_ident.attributes()); }
|
||||||
|
});
|
||||||
|
|
||||||
|
let attributes = self
|
||||||
|
.fields
|
||||||
|
.iter()
|
||||||
|
.filter(|field| field.attrs.xml_ty == FieldType::Attr)
|
||||||
|
.map(|field| {
|
||||||
|
let field_name = field.xml_name();
|
||||||
|
quote! {
|
||||||
|
::quick_xml::events::attributes::Attribute {
|
||||||
|
key: ::quick_xml::name::QName(#field_name),
|
||||||
|
value: b"hello".into()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
quote! {
|
quote! {
|
||||||
impl #impl_generics ::rustical_xml::XmlSerialize for #ident #type_generics #where_clause {
|
impl #impl_generics ::rustical_xml::XmlSerialize for #ident #type_generics #where_clause {
|
||||||
fn serialize<W: ::std::io::Write>(
|
fn serialize<W: ::std::io::Write>(
|
||||||
&self,
|
&self,
|
||||||
|
ns: Option<&[u8]>,
|
||||||
tag: Option<&[u8]>,
|
tag: Option<&[u8]>,
|
||||||
writer: &mut ::quick_xml::Writer<W>
|
writer: &mut ::quick_xml::Writer<W>
|
||||||
) -> ::std::io::Result<()> {
|
) -> ::std::io::Result<()> {
|
||||||
@@ -194,7 +217,10 @@ impl NamedStruct {
|
|||||||
let tag_str = tag.map(String::from_utf8_lossy);
|
let tag_str = tag.map(String::from_utf8_lossy);
|
||||||
|
|
||||||
if let Some(tag) = &tag_str {
|
if let Some(tag) = &tag_str {
|
||||||
writer.write_event(Event::Start(BytesStart::new(tag.to_owned())))?;
|
let mut bytes_start = BytesStart::new(tag.to_owned());
|
||||||
|
bytes_start.extend_attributes(self.attributes());
|
||||||
|
#(#untagged_attributes);*
|
||||||
|
writer.write_event(Event::Start(bytes_start))?;
|
||||||
}
|
}
|
||||||
#(#tag_writers);*
|
#(#tag_writers);*
|
||||||
if let Some(tag) = &tag_str {
|
if let Some(tag) = &tag_str {
|
||||||
@@ -202,6 +228,10 @@ impl NamedStruct {
|
|||||||
}
|
}
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn attributes<'a>(&self) -> Vec<::quick_xml::events::attributes::Attribute<'a>> {
|
||||||
|
vec![ #(#attributes),* ]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,11 +3,11 @@ use quick_xml::name::ResolveResult;
|
|||||||
use std::io::BufRead;
|
use std::io::BufRead;
|
||||||
pub use xml_derive::XmlDeserialize;
|
pub use xml_derive::XmlDeserialize;
|
||||||
pub use xml_derive::XmlDocument;
|
pub use xml_derive::XmlDocument;
|
||||||
pub use xml_derive::XmlRootTag;
|
|
||||||
|
|
||||||
use quick_xml::events::{BytesStart, Event};
|
use quick_xml::events::{BytesStart, Event};
|
||||||
|
|
||||||
use crate::XmlDeError;
|
use crate::XmlDeError;
|
||||||
|
use crate::XmlRootTag;
|
||||||
|
|
||||||
pub trait XmlDeserialize: Sized {
|
pub trait XmlDeserialize: Sized {
|
||||||
fn deserialize<R: BufRead>(
|
fn deserialize<R: BufRead>(
|
||||||
@@ -17,11 +17,6 @@ pub trait XmlDeserialize: Sized {
|
|||||||
) -> Result<Self, XmlDeError>;
|
) -> Result<Self, XmlDeError>;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait XmlRootTag {
|
|
||||||
fn root_tag() -> &'static [u8];
|
|
||||||
fn root_ns() -> Option<&'static [u8]>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub trait XmlDocument: XmlDeserialize {
|
pub trait XmlDocument: XmlDeserialize {
|
||||||
fn parse<R: BufRead>(reader: quick_xml::NsReader<R>) -> Result<Self, XmlDeError>;
|
fn parse<R: BufRead>(reader: quick_xml::NsReader<R>) -> Result<Self, XmlDeError>;
|
||||||
|
|
||||||
|
|||||||
@@ -16,6 +16,8 @@ pub enum XmlDeError {
|
|||||||
Eof,
|
Eof,
|
||||||
#[error("Unsupported xml event: {0}")]
|
#[error("Unsupported xml event: {0}")]
|
||||||
UnsupportedEvent(&'static str),
|
UnsupportedEvent(&'static str),
|
||||||
|
#[error("{0}")]
|
||||||
|
Other(String),
|
||||||
#[error("Invalid variant: {0}")]
|
#[error("Invalid variant: {0}")]
|
||||||
InvalidVariant(String),
|
InvalidVariant(String),
|
||||||
#[error("Invalid field name in {0}: {1}")]
|
#[error("Invalid field name in {0}: {1}")]
|
||||||
|
|||||||
@@ -8,10 +8,11 @@ mod value;
|
|||||||
|
|
||||||
pub use de::XmlDeserialize;
|
pub use de::XmlDeserialize;
|
||||||
pub use de::XmlDocument;
|
pub use de::XmlDocument;
|
||||||
pub use de::XmlRootTag;
|
|
||||||
pub use error::XmlDeError;
|
pub use error::XmlDeError;
|
||||||
pub use se::XmlSerialize;
|
pub use se::XmlSerialize;
|
||||||
|
pub use se::XmlSerializeRoot;
|
||||||
pub use value::Value;
|
pub use value::Value;
|
||||||
|
pub use xml_derive::XmlRootTag;
|
||||||
|
|
||||||
impl XmlDeserialize for () {
|
impl XmlDeserialize for () {
|
||||||
fn deserialize<R: BufRead>(
|
fn deserialize<R: BufRead>(
|
||||||
@@ -58,3 +59,8 @@ impl XmlDeserialize for Unparsed {
|
|||||||
Ok(Self(start.to_owned()))
|
Ok(Self(start.to_owned()))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub trait XmlRootTag {
|
||||||
|
fn root_tag() -> &'static [u8];
|
||||||
|
fn root_ns() -> Option<&'static [u8]>;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,9 +1,31 @@
|
|||||||
|
use quick_xml::events::attributes::Attribute;
|
||||||
pub use xml_derive::XmlSerialize;
|
pub use xml_derive::XmlSerialize;
|
||||||
|
|
||||||
|
use crate::XmlRootTag;
|
||||||
|
|
||||||
pub trait XmlSerialize {
|
pub trait XmlSerialize {
|
||||||
fn serialize<W: std::io::Write>(
|
fn serialize<W: std::io::Write>(
|
||||||
&self,
|
&self,
|
||||||
|
ns: Option<&[u8]>,
|
||||||
tag: Option<&[u8]>,
|
tag: Option<&[u8]>,
|
||||||
writer: &mut quick_xml::Writer<W>,
|
writer: &mut quick_xml::Writer<W>,
|
||||||
) -> std::io::Result<()>;
|
) -> std::io::Result<()>;
|
||||||
|
|
||||||
|
fn attributes<'a>(&self) -> impl IntoIterator<Item: Into<Attribute<'a>>>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait XmlSerializeRoot {
|
||||||
|
fn serialize_root<W: std::io::Write>(
|
||||||
|
&self,
|
||||||
|
writer: &mut quick_xml::Writer<W>,
|
||||||
|
) -> std::io::Result<()>;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: XmlSerialize + XmlRootTag> XmlSerializeRoot for T {
|
||||||
|
fn serialize_root<W: std::io::Write>(
|
||||||
|
&self,
|
||||||
|
writer: &mut quick_xml::Writer<W>,
|
||||||
|
) -> std::io::Result<()> {
|
||||||
|
self.serialize(Self::root_ns(), Some(Self::root_tag()), writer)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
use rustical_xml::{XmlRootTag, XmlSerialize};
|
use rustical_xml::{XmlRootTag, XmlSerialize, XmlSerializeRoot};
|
||||||
use xml_derive::XmlDeserialize;
|
use xml_derive::XmlDeserialize;
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_struct_document() {
|
fn test_struct_document() {
|
||||||
#[derive(Debug, XmlRootTag, XmlSerialize, XmlDeserialize, PartialEq)]
|
#[derive(Debug, XmlRootTag, XmlSerialize, PartialEq)]
|
||||||
#[xml(root = b"document")]
|
#[xml(root = b"document")]
|
||||||
struct Document {
|
struct Document {
|
||||||
child: Child,
|
child: Child,
|
||||||
@@ -22,7 +22,38 @@ fn test_struct_document() {
|
|||||||
text: "asd".to_owned(),
|
text: "asd".to_owned(),
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
.serialize(Some(Document::root_tag()), &mut writer)
|
.serialize_root(&mut writer)
|
||||||
|
.unwrap();
|
||||||
|
let out = String::from_utf8(buf).unwrap();
|
||||||
|
dbg!(out);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_struct_untagged_attr() {
|
||||||
|
#[derive(Debug, XmlRootTag, XmlSerialize, PartialEq)]
|
||||||
|
#[xml(root = b"document")]
|
||||||
|
struct Document {
|
||||||
|
#[xml(ty = "attr")]
|
||||||
|
name: String,
|
||||||
|
#[xml(ty = "untagged")]
|
||||||
|
child: Child,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, XmlSerialize, PartialEq, Default)]
|
||||||
|
struct Child {
|
||||||
|
#[xml(ty = "attr")]
|
||||||
|
text: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut buf = Vec::new();
|
||||||
|
let mut writer = quick_xml::Writer::new(&mut buf);
|
||||||
|
Document {
|
||||||
|
name: "okay".to_owned(),
|
||||||
|
child: Child {
|
||||||
|
text: "asd".to_owned(),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
.serialize_root(&mut writer)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
let out = String::from_utf8(buf).unwrap();
|
let out = String::from_utf8(buf).unwrap();
|
||||||
dbg!(out);
|
dbg!(out);
|
||||||
|
|||||||
Reference in New Issue
Block a user