285 lines
9.4 KiB
Rust
285 lines
9.4 KiB
Rust
use attribute::*;
|
|
use field_type::*;
|
|
use quote::TokenStreamExt;
|
|
use std::collections::BTreeMap;
|
|
use syn::Fields;
|
|
use syn::Ident;
|
|
use syn::DataEnum;
|
|
use proc_macro2::{Span, TokenStream};
|
|
|
|
pub fn parse(
|
|
data_enum: &DataEnum,
|
|
name: &Ident,
|
|
root: &str,
|
|
_namespaces: &BTreeMap<String, String>,
|
|
) -> TokenStream {
|
|
let variables: TokenStream = data_enum
|
|
.variants
|
|
.iter()
|
|
.map(|variant| match variant.fields {
|
|
Fields::Unit => None,
|
|
Fields::Named(ref fields) => {
|
|
let enum_fields = fields
|
|
.named
|
|
.iter()
|
|
.map(|field| {
|
|
let field_label = &field.ident;
|
|
|
|
match get_field_type(field) {
|
|
Some(FieldType::FieldTypeString) => {
|
|
build_default_value(field_label, "e!{String}, "e!{"".to_string()})
|
|
}
|
|
Some(FieldType::FieldTypeBool) => {
|
|
build_default_value(field_label, "e!{bool}, "e!{false})
|
|
}
|
|
Some(FieldType::FieldTypeI8) => {
|
|
build_default_value(field_label, "e!{i8}, "e!{0})
|
|
}
|
|
Some(FieldType::FieldTypeU8) => {
|
|
build_default_value(field_label, "e!{u8}, "e!{0})
|
|
}
|
|
Some(FieldType::FieldTypeI16) => {
|
|
build_default_value(field_label, "e!{i16}, "e!{0})
|
|
}
|
|
Some(FieldType::FieldTypeU16) => {
|
|
build_default_value(field_label, "e!{u16}, "e!{0})
|
|
}
|
|
Some(FieldType::FieldTypeI32) => {
|
|
build_default_value(field_label, "e!{i32}, "e!{0})
|
|
}
|
|
Some(FieldType::FieldTypeU32) => {
|
|
build_default_value(field_label, "e!{u32}, "e!{0})
|
|
}
|
|
Some(FieldType::FieldTypeI64) => {
|
|
build_default_value(field_label, "e!{i64}, "e!{0})
|
|
}
|
|
Some(FieldType::FieldTypeU64) => {
|
|
build_default_value(field_label, "e!{u64}, "e!{0})
|
|
}
|
|
Some(FieldType::FieldTypeStruct { struct_name }) => Some(quote!{
|
|
#[allow(unused_mut)]
|
|
let mut #field_label : #struct_name = #struct_name::default();
|
|
}),
|
|
Some(FieldType::FieldTypeVec { data_type }) => {
|
|
let dt = Box::into_raw(data_type);
|
|
match unsafe { dt.as_ref() } {
|
|
Some(&FieldType::FieldTypeString) => {
|
|
build_default_value(field_label, "e!{Vec<String>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeBool) => {
|
|
build_default_value(field_label, "e!{Vec<bool>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeI8) => {
|
|
build_default_value(field_label, "e!{Vec<i8>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeU8) => {
|
|
build_default_value(field_label, "e!{Vec<u8>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeI16) => {
|
|
build_default_value(field_label, "e!{Vec<i16>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeU16) => {
|
|
build_default_value(field_label, "e!{Vec<u16>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeI32) => {
|
|
build_default_value(field_label, "e!{Vec<i32>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeU32) => {
|
|
build_default_value(field_label, "e!{Vec<u32>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeI64) => {
|
|
build_default_value(field_label, "e!{Vec<i64>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeU64) => {
|
|
build_default_value(field_label, "e!{Vec<u64>}, "e!{vec![]})
|
|
}
|
|
Some(&FieldType::FieldTypeStruct { ref struct_name }) => Some(quote!{
|
|
#[allow(unused_mut)]
|
|
let mut #field_label : Vec<#struct_name> = vec![];
|
|
}),
|
|
Some(&FieldType::FieldTypeVec { .. }) => {
|
|
unimplemented!();
|
|
}
|
|
None => {
|
|
unimplemented!();
|
|
}
|
|
}
|
|
}
|
|
None => None,
|
|
}
|
|
})
|
|
.filter(|x| x.is_some())
|
|
.map(|x| x.unwrap())
|
|
.fold(TokenStream::empty(), |mut sum, val| {
|
|
sum.append_all(val);
|
|
sum
|
|
});
|
|
|
|
Some(enum_fields)
|
|
}
|
|
Fields::Unnamed(ref _fields) => {
|
|
unimplemented!();
|
|
}
|
|
})
|
|
.filter(|x| x.is_some())
|
|
.map(|x| x.unwrap())
|
|
.fold(TokenStream::empty(), |mut sum, val| {
|
|
sum.append_all(val);
|
|
sum
|
|
});
|
|
|
|
let enum_visitors: TokenStream = data_enum
|
|
.variants
|
|
.iter()
|
|
.map(|variant| {
|
|
match variant.fields {
|
|
Fields::Unit => None,
|
|
Fields::Named(ref fields) => {
|
|
let enum_fields = fields
|
|
.named
|
|
.iter()
|
|
.map(|field| {
|
|
// let label = field.ident;
|
|
// let label_name = label.unwrap().to_string();
|
|
// let visitor_label = Ident::new(&format!("__Visitor{}", label_name), Span::call_site());
|
|
|
|
match get_field_type(field) {
|
|
Some(FieldType::FieldTypeString) => {
|
|
Some(quote!{
|
|
// struct #visitor_label;
|
|
// impl<'de> Visitor<'de> for #visitor_label {
|
|
// type Value = String;
|
|
|
|
// fn visit_str(self, v: &str) -> Result<Self::Value, String> {
|
|
// match v {
|
|
// _ => Err("unable to match \"{}\" with enum {}", v, #label_name)
|
|
// }
|
|
// Ok(String::from(v))
|
|
// }
|
|
// }
|
|
})
|
|
}
|
|
_ => None,
|
|
}
|
|
})
|
|
.filter(|x| x.is_some())
|
|
.map(|x| x.unwrap())
|
|
.fold(TokenStream::empty(), |mut sum, val| {
|
|
sum.append_all(val);
|
|
sum
|
|
});
|
|
|
|
Some(enum_fields)
|
|
}
|
|
Fields::Unnamed(ref _fields) => {
|
|
unimplemented!();
|
|
}
|
|
}
|
|
})
|
|
.filter(|x| x.is_some())
|
|
.map(|x| x.unwrap())
|
|
.fold(TokenStream::empty(), |mut sum, val| {
|
|
sum.append_all(val);
|
|
sum
|
|
});
|
|
|
|
let match_to_enum: TokenStream = data_enum
|
|
.variants
|
|
.iter()
|
|
.map(|variant| {
|
|
let field_attrs = YaSerdeAttribute::parse(&variant.attrs);
|
|
let renamed_label = match field_attrs.rename {
|
|
Some(value) => Ident::new(&format!("{}", value), Span::call_site()),
|
|
None => variant.ident.clone(),
|
|
};
|
|
let label = &variant.ident;
|
|
let label_name = renamed_label.to_string();
|
|
|
|
match variant.fields {
|
|
Fields::Unit => Some(quote!{
|
|
#label_name => {
|
|
simple_enum_value = Some(#name::#label);
|
|
}
|
|
}),
|
|
_ => None,
|
|
}
|
|
})
|
|
.filter(|x| x.is_some())
|
|
.map(|x| x.unwrap())
|
|
.fold(TokenStream::empty(), |mut tokens, token| {
|
|
tokens.append_all(token);
|
|
tokens
|
|
});
|
|
|
|
quote!{
|
|
use xml::reader::XmlEvent;
|
|
|
|
impl YaDeserialize for #name {
|
|
#[allow(unused_variables)]
|
|
fn deserialize<R: Read>(reader: &mut yaserde::de::Deserializer<R>) -> Result<Self, String> {
|
|
let named_element =
|
|
if let XmlEvent::StartElement{name, ..} = reader.peek()?.to_owned() {
|
|
name.local_name.to_owned()
|
|
} else {
|
|
String::from(#root)
|
|
};
|
|
debug!("Enum: start to parse {:?}", named_element);
|
|
|
|
#[allow(unused_assignments, unused_mut)]
|
|
let mut simple_enum_value = None;
|
|
|
|
#variables
|
|
#enum_visitors
|
|
|
|
loop {
|
|
match reader.peek()?.to_owned() {
|
|
XmlEvent::StartElement{name, attributes, namespace: _namespace} => {
|
|
debug!("Enum: {}: {}", named_element, name.local_name.as_str());
|
|
if name.local_name == named_element {
|
|
let _next = reader.next_event();
|
|
|
|
if let XmlEvent::Characters(content) = reader.peek()?.to_owned() {
|
|
match content.as_str() {
|
|
#match_to_enum
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
},
|
|
XmlEvent::EndElement{name} => {
|
|
if name.local_name.as_str() == named_element {
|
|
break;
|
|
}
|
|
let _root = reader.next_event();
|
|
},
|
|
xml::reader::XmlEvent::Characters(characters_content) => {
|
|
let _root = reader.next_event();
|
|
},
|
|
event => {
|
|
return Err(format!("unknown event {:?}", event))
|
|
},
|
|
}
|
|
}
|
|
|
|
match simple_enum_value {
|
|
Some(value) => Ok(value),
|
|
None => {
|
|
Ok(#name::default())
|
|
},
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
fn build_default_value(
|
|
label: &Option<Ident>,
|
|
field_type: &TokenStream,
|
|
default: &TokenStream,
|
|
) -> Option<TokenStream> {
|
|
Some(quote!{
|
|
#[allow(unused_mut)]
|
|
let mut #label : #field_type = #default;
|
|
})
|
|
}
|