use attribute::*; use field_type::*; use quote::TokenStreamExt; use std::collections::BTreeMap; use syn::Ident; use syn::DataStruct; use proc_macro2::{Span, TokenStream}; use std::string::ToString; pub fn serialize( data_struct: &DataStruct, name: &Ident, root: &str, namespaces: &BTreeMap, ) -> TokenStream { let build_attributes: TokenStream = 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.clone(), }; 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, &*{ use std::mem; unsafe { let content = format!("{}", self.#label); let ret : &'static str = mem::transmute(&content as &str); mem::forget(content); ret } }) }), 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(TokenStream::empty(), |mut tokens, token| { tokens.append_all(token); tokens }); let add_namespaces: TokenStream = namespaces .iter() .map(|(prefix, namespace)| { Some(quote!( .ns(#prefix, #namespace) )) }) .filter(|x| x.is_some()) .map(|x| x.unwrap()) .fold(TokenStream::empty(), |mut tokens, token| { tokens.append_all(token); tokens }); let struct_inspector: TokenStream = 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.clone(), }; 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(TokenStream::empty(), |mut tokens, token| { tokens.append_all(token); tokens }); quote! { use xml::writer::XmlEvent; impl YaSerialize for #name { #[allow(unused_variables)] fn serialize(&self, writer: &mut yaserde::ser::Serializer) -> 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(()) } } } }