Merge pull request #36 from DmitrySamoylov/feature-flatten

Feature flatten (closes #2)
This commit is contained in:
Marc-Antoine ARNAUD 2020-02-06 10:25:42 +01:00 committed by GitHub
commit 8cc97f90ef
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 292 additions and 17 deletions

View File

@ -28,7 +28,7 @@ This library will support XML de/ser-ializing with all specific features.
- [x] **attribute**: this field is defined as an attribute
- [x] **default**: defines the default function to init the field
- [ ] **flatten**: Flatten the contents of the field
- [x] **flatten**: Flatten the contents of the field
- [x] **namespace**: defines the namespace of the field
- [x] **rename**: be able to rename a field
- [x] **root**: rename the based element. Used only at the XML root.

View File

@ -647,3 +647,89 @@ fn de_custom() {
}
);
}
#[test]
fn de_flatten() {
#[derive(Default, PartialEq, Debug, YaDeserialize)]
struct DateTime {
#[yaserde(flatten)]
date: Date,
time: String,
#[yaserde(flatten)]
kind: DateKind,
}
#[derive(Default, PartialEq, Debug, YaDeserialize)]
struct Date {
year: i32,
month: i32,
day: i32,
#[yaserde(flatten)]
extra: Extra,
#[yaserde(flatten)]
optional_extra: Option<OptionalExtra>,
}
#[derive(Default, PartialEq, Debug, YaDeserialize)]
pub struct Extra {
week: i32,
century: i32,
}
#[derive(Default, PartialEq, Debug, YaDeserialize)]
pub struct OptionalExtra {
lunar_day: i32,
}
#[derive(PartialEq, Debug, YaDeserialize)]
pub enum DateKind {
#[yaserde(rename = "holidays")]
Holidays(Vec<String>),
#[yaserde(rename = "working")]
Working,
}
impl Default for DateKind {
fn default() -> Self {
DateKind::Working
}
};
let content = r#"
<?xml version="1.0" encoding="utf-8"?>
<DateTime>
<year>2020</year>
<month>1</month>
<day>1</day>
<week>1</week>
<century>21</century>
<lunar_day>1</lunar_day>
<time>10:40:03</time>
<holidays>New Year's Day</holidays>
<holidays>Novy God Day</holidays>
<holidays>Polar Bear Swim Day</holidays>
</DateTime>
"#;
convert_and_validate!(
content,
DateTime,
DateTime {
date: Date {
year: 2020,
month: 1,
day: 1,
extra: Extra {
week: 1,
century: 21,
},
optional_extra: Some(OptionalExtra { lunar_day: 1 }),
},
time: "10:40:03".to_string(),
kind: DateKind::Holidays(vec![
"New Year's Day".into(),
"Novy God Day".into(),
"Polar Bear Swim Day".into()
])
}
);
}

View File

@ -11,7 +11,15 @@ use yaserde::YaSerialize;
macro_rules! convert_and_validate {
($model: expr, $content: expr) => {
let data: Result<String, String> = to_string(&$model);
assert_eq!(data, Ok(String::from($content)));
assert_eq!(
data,
Ok(
String::from($content)
.split("\n")
.map(|s| s.trim())
.collect::<String>()
)
);
};
}
@ -489,3 +497,87 @@ fn ser_custom() {
let content = "<?xml version=\"1.0\" encoding=\"utf-8\"?><Date><Year>2020</Year><Month>1</Month><DoubleDay>10</DoubleDay></Date>";
convert_and_validate!(model, content);
}
#[test]
fn ser_flatten() {
#[derive(Default, PartialEq, Debug, YaSerialize)]
struct DateTime {
#[yaserde(flatten)]
date: Date,
time: String,
#[yaserde(flatten)]
kind: DateKind,
}
#[derive(Default, PartialEq, Debug, YaSerialize)]
struct Date {
year: i32,
month: i32,
day: i32,
#[yaserde(flatten)]
extra: Extra,
#[yaserde(flatten)]
optional_extra: Option<OptionalExtra>,
}
#[derive(Default, PartialEq, Debug, YaSerialize)]
pub struct Extra {
week: i32,
century: i32,
}
#[derive(Default, PartialEq, Debug, YaSerialize)]
pub struct OptionalExtra {
lunar_day: i32,
}
#[derive(PartialEq, Debug, YaSerialize)]
pub enum DateKind {
#[yaserde(rename = "holidays")]
Holidays(Vec<String>),
#[yaserde(rename = "working")]
Working,
}
impl Default for DateKind {
fn default() -> Self {
DateKind::Working
}
};
let model = DateTime {
date: Date {
year: 2020,
month: 1,
day: 1,
extra: Extra {
week: 1,
century: 21,
},
optional_extra: Some(OptionalExtra { lunar_day: 1 }),
},
time: "10:40:03".to_string(),
kind: DateKind::Holidays(vec![
"New Year's Day".into(),
"Novy God Day".into(),
"Polar Bear Swim Day".into(),
]),
};
let content = r#"
<?xml version="1.0" encoding="utf-8"?>
<DateTime>
<year>2020</year>
<month>1</month>
<day>1</day>
<week>1</week>
<century>21</century>
<lunar_day>1</lunar_day>
<time>10:40:03</time>
<holidays>New Year's Day</holidays>
<holidays>Novy God Day</holidays>
<holidays>Polar Bear Swim Day</holidays>
</DateTime>"#;
convert_and_validate!(model, content);
}

View File

@ -13,6 +13,7 @@ pub struct YaSerdeAttribute {
pub namespaces: BTreeMap<String, String>,
pub attribute: bool,
pub text: bool,
pub flatten: bool,
}
fn get_value(iter: &mut IntoIter) -> Option<String> {
@ -38,6 +39,7 @@ impl YaSerdeAttribute {
let mut root = None;
let mut default = None;
let mut text = false;
let mut flatten = false;
for attr in attrs.iter() {
let mut attr_iter = attr.clone().tokens.into_iter();
@ -78,6 +80,9 @@ impl YaSerdeAttribute {
"text" => {
text = true;
}
"flatten" => {
flatten = true;
}
_ => {}
}
}
@ -95,6 +100,7 @@ impl YaSerdeAttribute {
root,
default,
text,
flatten,
}
}
}
@ -113,6 +119,7 @@ fn parse_empty_attributes() {
namespaces: BTreeMap::new(),
attribute: false,
text: false,
flatten: false,
},
attrs
);
@ -160,6 +167,7 @@ fn parse_attributes() {
namespaces: BTreeMap::new(),
attribute: true,
text: false,
flatten: false,
},
attrs
);

View File

@ -186,7 +186,7 @@ pub fn parse(
let label = &field.ident;
let value_label = &get_value_label(&field.ident);
if field_attrs.attribute {
if field_attrs.attribute || field_attrs.flatten {
return None;
}
@ -288,6 +288,33 @@ pub fn parse(
.filter_map(|x| x)
.collect();
let call_flatten_visitors: TokenStream = data_struct
.fields
.iter()
.map(|field| {
let field_attrs = YaSerdeAttribute::parse(&field.attrs);
let value_label = &get_value_label(&field.ident);
if field_attrs.attribute || !field_attrs.flatten {
return None;
}
get_field_type(field).and_then(|f| match f {
FieldType::FieldTypeStruct { .. } => Some(quote! {
#value_label = yaserde::de::from_str(&unused_xml_elements)?;
}),
FieldType::FieldTypeOption { data_type } => match *data_type {
FieldType::FieldTypeStruct { .. } => Some(quote! {
#value_label = yaserde::de::from_str(&unused_xml_elements).ok();
}),
field_type => unimplemented!("\"flatten\" is not implemented for {:?}", field_type),
},
field_type => unimplemented!("\"flatten\" is not implemented for {:?}", field_type),
})
})
.filter_map(|x| x)
.collect();
let attributes_loading: TokenStream = data_struct
.fields
.iter()
@ -410,8 +437,15 @@ pub fn parse(
.filter_map(|x| x)
.collect();
let (init_unused, write_unused, visit_unused) = if call_flatten_visitors.is_empty() {
(None, None, None)
} else {
build_code_for_unused_xml_events(&call_flatten_visitors)
};
quote! {
use xml::reader::XmlEvent;
use xml::reader::{XmlEvent, EventReader};
use xml::writer::EventWriter;
use yaserde::Visitor;
#[allow(unknown_lints, unused_imports)]
use std::str::FromStr;
@ -439,15 +473,19 @@ pub fn parse(
#variables
#field_visitors
#init_unused
loop {
match reader.peek()?.to_owned() {
let event = reader.peek()?.to_owned();
match event {
XmlEvent::StartElement{ref name, ref attributes, ..} => {
match name.local_name.as_str() {
#call_visitors
named_element => {
let _root = reader.next_event();
let event = reader.next_event()?;
#write_unused
}
// name => {
// return Err(format!("unknown key {}", name))
@ -457,13 +495,16 @@ pub fn parse(
}
XmlEvent::EndElement{ref name} => {
if name.local_name == named_element {
#write_unused
break;
}
let _root = reader.next_event();
let event = reader.next_event()?;
#write_unused
}
XmlEvent::Characters(ref text_content) => {
#set_text
let _root = reader.next_event();
let event = reader.next_event()?;
#write_unused
}
event => {
return Err(format!("unknown event {:?}", event))
@ -471,6 +512,8 @@ pub fn parse(
}
}
#visit_unused
Ok(#name{#struct_builder})
}
}
@ -613,3 +656,31 @@ fn build_visitor_ident(label: &str, span: Span, struct_id: Option<&str>) -> Iden
span,
)
}
fn build_code_for_unused_xml_events(
call_flatten_visitors: &TokenStream,
) -> (
Option<TokenStream>,
Option<TokenStream>,
Option<TokenStream>,
) {
(
Some(quote! {
let mut buf = Vec::new();
let mut writer = Some(EventWriter::new(&mut buf));
}),
Some(quote! {
if let Some(ref mut w) = writer {
if w.write(event.as_writer_event().unwrap()).is_err() {
writer = None;
}
}
}),
Some(quote! {
if writer.is_some() {
let unused_xml_elements = String::from_utf8(buf).unwrap();
#call_flatten_visitors
}
}),
)
}

View File

@ -327,19 +327,37 @@ pub fn serialize(
})
}
}
FieldType::FieldTypeStruct { .. } => Some(quote! {
if let Some(ref item) = &self.#label {
writer.set_start_event_name(Some(#label_name.to_string()));
writer.set_skip_start_end(false);
item.serialize(writer)?;
FieldType::FieldTypeStruct { .. } => Some(if field_attrs.flatten {
quote! {
if let Some(ref item) = &self.#label {
writer.set_start_event_name(None);
writer.set_skip_start_end(true);
item.serialize(writer)?;
}
}
} else {
quote! {
if let Some(ref item) = &self.#label {
writer.set_start_event_name(Some(#label_name.to_string()));
writer.set_skip_start_end(false);
item.serialize(writer)?;
}
}
}),
_ => unimplemented!(),
},
FieldType::FieldTypeStruct { .. } => Some(quote! {
writer.set_start_event_name(Some(#label_name.to_string()));
writer.set_skip_start_end(false);
self.#label.serialize(writer)?;
FieldType::FieldTypeStruct { .. } => Some(if field_attrs.flatten {
quote! {
writer.set_start_event_name(None);
writer.set_skip_start_end(true);
self.#label.serialize(writer)?;
}
} else {
quote! {
writer.set_start_event_name(Some(#label_name.to_string()));
writer.set_skip_start_end(false);
self.#label.serialize(writer)?;
}
}),
FieldType::FieldTypeVec { data_type } => match *data_type {
FieldType::FieldTypeString => {