244 lines
7.5 KiB
Rust
244 lines
7.5 KiB
Rust
use attribute::*;
|
|
use field_type::*;
|
|
use quote::Tokens;
|
|
use std::collections::BTreeMap;
|
|
use syn::Ident;
|
|
use syn::DataStruct;
|
|
use proc_macro2::Span;
|
|
use std::string::ToString;
|
|
|
|
pub fn serialize(
|
|
data_struct: &DataStruct,
|
|
name: &Ident,
|
|
root: &str,
|
|
namespaces: &BTreeMap<String, String>,
|
|
) -> Tokens {
|
|
let build_attributes: Tokens = data_struct
|
|
.fields
|
|
.iter()
|
|
.map(|field| {
|
|
let field_attrs = YaSerdeAttribute::parse(&field.attrs);
|
|
if !field_attrs.attribute {
|
|
return None;
|
|
}
|
|
|
|
let renamed_label = match field_attrs.rename {
|
|
Some(value) => Some(Ident::new(&format!("{}", value), Span::call_site())),
|
|
None => field.ident,
|
|
};
|
|
let label = field.ident;
|
|
let label_name = if let Some(prefix) = field_attrs.prefix {
|
|
prefix + ":" + renamed_label.unwrap().to_string().as_ref()
|
|
} else {
|
|
renamed_label.unwrap().to_string()
|
|
};
|
|
|
|
match get_field_type(field) {
|
|
Some(FieldType::FieldTypeString)
|
|
| Some(FieldType::FieldTypeBool)
|
|
| Some(FieldType::FieldTypeI8)
|
|
| Some(FieldType::FieldTypeU8)
|
|
| Some(FieldType::FieldTypeI16)
|
|
| Some(FieldType::FieldTypeU16)
|
|
| Some(FieldType::FieldTypeI32)
|
|
| Some(FieldType::FieldTypeU32)
|
|
| Some(FieldType::FieldTypeI64)
|
|
| Some(FieldType::FieldTypeU64) => Some(quote!{
|
|
.attr(#label_name, &self.#label)
|
|
}),
|
|
Some(FieldType::FieldTypeStruct { .. }) => Some(quote!{
|
|
.attr(#label_name, &*{
|
|
use std::mem;
|
|
match yaserde::ser::to_string_content(&self.#label) {
|
|
Ok(value) => {
|
|
unsafe {
|
|
let ret : &'static str = mem::transmute(&value as &str);
|
|
mem::forget(value);
|
|
ret
|
|
}
|
|
},
|
|
Err(msg) => return Err("Unable to serialize content".to_owned()),
|
|
}
|
|
})
|
|
}),
|
|
_ => None,
|
|
}
|
|
})
|
|
.filter(|x| x.is_some())
|
|
.map(|x| x.unwrap())
|
|
.fold(Tokens::new(), |mut tokens, token| {
|
|
tokens.append_all(token);
|
|
tokens
|
|
});
|
|
|
|
let add_namespaces: Tokens = namespaces
|
|
.iter()
|
|
.map(|(prefix, namespace)| {
|
|
Some(quote!(
|
|
.ns(#prefix, #namespace)
|
|
))
|
|
})
|
|
.filter(|x| x.is_some())
|
|
.map(|x| x.unwrap())
|
|
.fold(Tokens::new(), |mut tokens, token| {
|
|
tokens.append_all(token);
|
|
tokens
|
|
});
|
|
|
|
let struct_inspector: Tokens = data_struct
|
|
.fields
|
|
.iter()
|
|
.map(|field| {
|
|
let field_attrs = YaSerdeAttribute::parse(&field.attrs);
|
|
if field_attrs.attribute {
|
|
return None;
|
|
}
|
|
|
|
let label = field.ident;
|
|
if field_attrs.text {
|
|
return Some(quote!(
|
|
let data_event = XmlEvent::characters(&self.#label);
|
|
let _ret = writer.write(data_event);
|
|
));
|
|
}
|
|
|
|
let renamed_label = match field_attrs.rename {
|
|
Some(value) => Some(Ident::new(&format!("{}", value), Span::call_site())),
|
|
None => field.ident,
|
|
};
|
|
|
|
let label_name = if let Some(prefix) = field_attrs.prefix {
|
|
prefix + ":" + renamed_label.unwrap().to_string().as_ref()
|
|
} else {
|
|
renamed_label.unwrap().to_string()
|
|
};
|
|
|
|
match get_field_type(field) {
|
|
Some(FieldType::FieldTypeString) => Some(quote!{
|
|
let start_event = XmlEvent::start_element(#label_name);
|
|
let _ret = writer.write(start_event);
|
|
|
|
let data_event = XmlEvent::characters(&self.#label);
|
|
let _ret = writer.write(data_event);
|
|
|
|
let end_event = XmlEvent::end_element();
|
|
let _ret = writer.write(end_event);
|
|
}),
|
|
Some(FieldType::FieldTypeBool)
|
|
| Some(FieldType::FieldTypeI8)
|
|
| Some(FieldType::FieldTypeU8)
|
|
| Some(FieldType::FieldTypeI16)
|
|
| Some(FieldType::FieldTypeU16)
|
|
| Some(FieldType::FieldTypeI32)
|
|
| Some(FieldType::FieldTypeU32)
|
|
| Some(FieldType::FieldTypeI64)
|
|
| Some(FieldType::FieldTypeU64) => Some(quote!{
|
|
let start_event = XmlEvent::start_element(#label_name);
|
|
let _ret = writer.write(start_event);
|
|
|
|
let content = format!("{}", &self.#label);
|
|
let data_event = XmlEvent::characters(&content);
|
|
let _ret = writer.write(data_event);
|
|
|
|
let end_event = XmlEvent::end_element();
|
|
let _ret = writer.write(end_event);
|
|
}),
|
|
Some(FieldType::FieldTypeStruct { .. }) => Some(quote!{
|
|
writer.set_skip_start_end(false);
|
|
match self.#label.serialize(writer) {
|
|
Ok(()) => {},
|
|
Err(msg) => {
|
|
return Err(msg);
|
|
},
|
|
};
|
|
}),
|
|
Some(FieldType::FieldTypeVec { data_type }) => {
|
|
let dt = Box::into_raw(data_type);
|
|
match unsafe { dt.as_ref() } {
|
|
Some(&FieldType::FieldTypeString) => Some(quote!{
|
|
for item in &self.#label {
|
|
let start_event = XmlEvent::start_element(#label_name);
|
|
let _ret = writer.write(start_event);
|
|
|
|
let data_event = XmlEvent::characters(item);
|
|
let _ret = writer.write(data_event);
|
|
|
|
let end_event = XmlEvent::end_element();
|
|
let _ret = writer.write(end_event);
|
|
}
|
|
}),
|
|
Some(&FieldType::FieldTypeBool)
|
|
| Some(&FieldType::FieldTypeI8)
|
|
| Some(&FieldType::FieldTypeU8)
|
|
| Some(&FieldType::FieldTypeI16)
|
|
| Some(&FieldType::FieldTypeU16)
|
|
| Some(&FieldType::FieldTypeI32)
|
|
| Some(&FieldType::FieldTypeU32)
|
|
| Some(&FieldType::FieldTypeI64)
|
|
| Some(&FieldType::FieldTypeU64) => Some(quote!{
|
|
for item in &self.#label {
|
|
let start_event = XmlEvent::start_element(#label_name);
|
|
let _ret = writer.write(start_event);
|
|
|
|
let data_event = XmlEvent::characters(format!("{}", item));
|
|
let _ret = writer.write(data_event);
|
|
|
|
let end_event = XmlEvent::end_element();
|
|
let _ret = writer.write(end_event);
|
|
}
|
|
}),
|
|
Some(&FieldType::FieldTypeStruct { .. }) => Some(quote!{
|
|
for item in &self.#label {
|
|
writer.set_skip_start_end(false);
|
|
match item.serialize(writer) {
|
|
Ok(()) => {},
|
|
Err(msg) => {
|
|
return Err(msg);
|
|
},
|
|
};
|
|
}
|
|
}),
|
|
Some(&FieldType::FieldTypeVec { .. }) => {
|
|
unimplemented!();
|
|
}
|
|
None => {
|
|
unimplemented!();
|
|
}
|
|
}
|
|
}
|
|
None => None,
|
|
}
|
|
})
|
|
.filter(|x| x.is_some())
|
|
.map(|x| x.unwrap())
|
|
.fold(Tokens::new(), |mut tokens, token| {
|
|
tokens.append_all(token);
|
|
tokens
|
|
});
|
|
|
|
quote! {
|
|
use xml::writer::XmlEvent;
|
|
|
|
impl YaSerialize for #name {
|
|
#[allow(unused_variables)]
|
|
fn serialize<W: Write>(&self, writer: &mut yaserde::ser::Serializer<W>)
|
|
-> Result<(), String> {
|
|
error!("Struct: start to expand {:?}", #root);
|
|
let skip = writer.skip_start_end();
|
|
if !skip {
|
|
let struct_start_event = XmlEvent::start_element(#root)#build_attributes#add_namespaces;
|
|
let _ret = writer.write(struct_start_event);
|
|
}
|
|
|
|
#struct_inspector
|
|
|
|
if !skip {
|
|
let struct_end_event = XmlEvent::end_element();
|
|
let _ret = writer.write(struct_end_event);
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
}
|
|
}
|