Merge pull request #47 from media-io/default_namespace

add default_namespace attribute
This commit is contained in:
Marc-Antoine ARNAUD 2020-02-25 08:11:14 +01:00 committed by GitHub
commit 1a67c4907d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 151 additions and 31 deletions

View File

@ -1,3 +1,6 @@
//! Generic data structure deserialization framework.
//!
use std::io::Read; use std::io::Read;
use xml::name::OwnedName; use xml::name::OwnedName;
use xml::reader::{EventReader, ParserConfig, XmlEvent}; use xml::reader::{EventReader, ParserConfig, XmlEvent};

View File

@ -1,3 +1,8 @@
//! # YaSerDe
//!
//! YaSerDe is a framework for ***ser***ializing and ***de***serializing Rust data
//! structures efficiently and generically from and into XML.
#[macro_use] #[macro_use]
extern crate log; extern crate log;
extern crate xml; extern crate xml;
@ -12,14 +17,17 @@ use xml::writer::XmlEvent;
pub mod de; pub mod de;
pub mod ser; pub mod ser;
/// A **data structure** that can be deserialized from any data format supported by YaSerDe.
pub trait YaDeserialize: Sized { pub trait YaDeserialize: Sized {
fn deserialize<R: Read>(reader: &mut de::Deserializer<R>) -> Result<Self, String>; fn deserialize<R: Read>(reader: &mut de::Deserializer<R>) -> Result<Self, String>;
} }
/// A **data structure** that can be serialized into any data format supported by YaSerDe.
pub trait YaSerialize: Sized { pub trait YaSerialize: Sized {
fn serialize<W: Write>(&self, writer: &mut ser::Serializer<W>) -> Result<(), String>; fn serialize<W: Write>(&self, writer: &mut ser::Serializer<W>) -> Result<(), String>;
} }
/// A **visitor** that can be implemented to retrieve information from source file.
pub trait Visitor<'de>: Sized { pub trait Visitor<'de>: Sized {
/// The value produced by this visitor. /// The value produced by this visitor.
type Value; type Value;

View File

@ -1,3 +1,6 @@
//! Generic data structure serialization framework.
//!
use std::io::{Cursor, Write}; use std::io::{Cursor, Write};
use std::str; use std::str;
use xml; use xml;

View File

@ -161,6 +161,27 @@ fn ser_struct_default_namespace() {
convert_and_validate!(model, content); convert_and_validate!(model, content);
} }
#[test]
fn ser_struct_default_namespace_via_attribute() {
#[derive(YaSerialize, PartialEq, Debug)]
#[yaserde(
root = "tt",
default_namespace = "ttml",
namespace = "ttml: http://www.w3.org/ns/ttml",
namespace = "ttm: http://www.w3.org/ns/ttml#metadata"
)]
pub struct XmlStruct {
item: String,
}
let model = XmlStruct {
item: "something".to_string(),
};
let content = "<?xml version=\"1.0\" encoding=\"utf-8\"?><tt xmlns=\"http://www.w3.org/ns/ttml\" xmlns:ttm=\"http://www.w3.org/ns/ttml#metadata\"><item>something</item></tt>";
convert_and_validate!(model, content);
}
#[test] #[test]
fn de_struct_namespace_nested() { fn de_struct_namespace_nested() {
#[derive(YaSerialize, Default, PartialEq, Debug)] #[derive(YaSerialize, Default, PartialEq, Debug)]

View File

@ -6,14 +6,15 @@ use syn::Attribute;
#[derive(Debug, PartialEq, Clone)] #[derive(Debug, PartialEq, Clone)]
pub struct YaSerdeAttribute { pub struct YaSerdeAttribute {
pub attribute: bool,
pub default: Option<String>,
pub default_namespace: Option<String>,
pub flatten: bool,
pub namespaces: BTreeMap<String, String>,
pub prefix: Option<String>,
pub root: Option<String>, pub root: Option<String>,
pub rename: Option<String>, pub rename: Option<String>,
pub prefix: Option<String>,
pub default: Option<String>,
pub namespaces: BTreeMap<String, String>,
pub attribute: bool,
pub text: bool, pub text: bool,
pub flatten: bool,
} }
fn get_value(iter: &mut IntoIter) -> Option<String> { fn get_value(iter: &mut IntoIter) -> Option<String> {
@ -33,13 +34,14 @@ fn get_value(iter: &mut IntoIter) -> Option<String> {
impl YaSerdeAttribute { impl YaSerdeAttribute {
pub fn parse(attrs: &[Attribute]) -> YaSerdeAttribute { pub fn parse(attrs: &[Attribute]) -> YaSerdeAttribute {
let mut attribute = false; let mut attribute = false;
let mut flatten = false;
let mut default = None;
let mut default_namespace = None;
let mut namespaces = BTreeMap::new(); let mut namespaces = BTreeMap::new();
let mut prefix = None; let mut prefix = None;
let mut rename = None; let mut rename = None;
let mut root = None; let mut root = None;
let mut default = None;
let mut text = false; let mut text = false;
let mut flatten = false;
for attr in attrs.iter() { for attr in attrs.iter() {
let mut attr_iter = attr.clone().tokens.into_iter(); let mut attr_iter = attr.clone().tokens.into_iter();
@ -54,6 +56,15 @@ impl YaSerdeAttribute {
"attribute" => { "attribute" => {
attribute = true; attribute = true;
} }
"default" => {
default = get_value(&mut attr_iter);
}
"default_namespace" => {
default_namespace = get_value(&mut attr_iter);
}
"flatten" => {
flatten = true;
}
"namespace" => { "namespace" => {
if let Some(namespace) = get_value(&mut attr_iter) { if let Some(namespace) = get_value(&mut attr_iter) {
let splitted: Vec<&str> = namespace.split(": ").collect(); let splitted: Vec<&str> = namespace.split(": ").collect();
@ -74,15 +85,9 @@ impl YaSerdeAttribute {
"root" => { "root" => {
root = get_value(&mut attr_iter); root = get_value(&mut attr_iter);
} }
"default" => {
default = get_value(&mut attr_iter);
}
"text" => { "text" => {
text = true; text = true;
} }
"flatten" => {
flatten = true;
}
_ => {} _ => {}
} }
} }
@ -94,13 +99,14 @@ impl YaSerdeAttribute {
YaSerdeAttribute { YaSerdeAttribute {
attribute, attribute,
default,
default_namespace,
flatten,
namespaces, namespaces,
prefix, prefix,
rename, rename,
root, root,
default,
text, text,
flatten,
} }
} }
} }
@ -112,14 +118,15 @@ fn parse_empty_attributes() {
assert_eq!( assert_eq!(
YaSerdeAttribute { YaSerdeAttribute {
attribute: false,
default: None,
default_namespace: None,
flatten: false,
namespaces: BTreeMap::new(),
prefix: None,
root: None, root: None,
rename: None, rename: None,
prefix: None,
default: None,
namespaces: BTreeMap::new(),
attribute: false,
text: false, text: false,
flatten: false,
}, },
attrs attrs
); );
@ -160,14 +167,68 @@ fn parse_attributes() {
assert_eq!( assert_eq!(
YaSerdeAttribute { YaSerdeAttribute {
attribute: true,
default: None,
default_namespace: None,
flatten: false,
namespaces: BTreeMap::new(),
prefix: None,
root: None,
rename: None,
text: false,
},
attrs
);
}
#[test]
fn parse_attributes_with_values() {
use proc_macro2::{Span, TokenStream};
use std::str::FromStr;
use syn::punctuated::Punctuated;
use syn::token::Bracket;
use syn::token::Pound;
use syn::AttrStyle::Outer;
use syn::{Ident, Path, PathArguments, PathSegment};
let mut punctuated = Punctuated::new();
punctuated.push(PathSegment {
ident: Ident::new("yaserde", Span::call_site()),
arguments: PathArguments::None,
});
// #[()]
let attributes = vec![Attribute {
pound_token: Pound {
spans: [Span::call_site()],
},
style: Outer,
bracket_token: Bracket {
span: Span::call_site(),
},
path: Path {
leading_colon: None,
segments: punctuated,
},
tokens: TokenStream::from_str("(attribute, flatten, default_namespace=\"example\", namespace=\"example: http://example.org\")").unwrap(),
}];
let attrs = YaSerdeAttribute::parse(&attributes);
let mut namespaces = BTreeMap::new();
namespaces.insert("example".to_string(), "http://example.org".to_string());
assert_eq!(
YaSerdeAttribute {
attribute: true,
default: None,
default_namespace: Some("example".to_string()),
flatten: true,
namespaces,
prefix: None,
root: None, root: None,
rename: None, rename: None,
prefix: None,
default: None,
namespaces: BTreeMap::new(),
attribute: true,
text: false, text: false,
flatten: false,
}, },
attrs attrs
); );

View File

@ -12,6 +12,7 @@ pub fn serialize(
name: &Ident, name: &Ident,
root: &str, root: &str,
namespaces: &BTreeMap<String, String>, namespaces: &BTreeMap<String, String>,
default_namespace: &Option<String>,
) -> TokenStream { ) -> TokenStream {
let write_enum_content: TokenStream = data_enum let write_enum_content: TokenStream = data_enum
.variants .variants
@ -212,6 +213,13 @@ pub fn serialize(
let add_namespaces: TokenStream = namespaces let add_namespaces: TokenStream = namespaces
.iter() .iter()
.map(|(prefix, namespace)| { .map(|(prefix, namespace)| {
if let Some(dn) = default_namespace {
if dn == prefix {
return Some(quote!(
.default_ns(#namespace)
));
}
}
Some(quote!( Some(quote!(
.ns(#prefix, #namespace) .ns(#prefix, #namespace)
)) ))

View File

@ -14,6 +14,7 @@ pub fn serialize(
name: &Ident, name: &Ident,
root: &str, root: &str,
namespaces: &BTreeMap<String, String>, namespaces: &BTreeMap<String, String>,
default_namespace: &Option<String>,
) -> TokenStream { ) -> TokenStream {
let build_attributes: TokenStream = data_struct let build_attributes: TokenStream = data_struct
.fields .fields
@ -208,6 +209,13 @@ pub fn serialize(
let add_namespaces: TokenStream = namespaces let add_namespaces: TokenStream = namespaces
.iter() .iter()
.map(|(prefix, namespace)| { .map(|(prefix, namespace)| {
if let Some(dn) = default_namespace {
if dn == prefix {
return Some(quote!(
.default_ns(#namespace)
));
}
}
Some(quote!( Some(quote!(
.ns(#prefix, #namespace) .ns(#prefix, #namespace)
)) ))

View File

@ -22,12 +22,20 @@ pub fn expand_derive_serialize(ast: &syn::DeriveInput) -> Result<TokenStream, St
}; };
let impl_block = match *data { let impl_block = match *data {
syn::Data::Struct(ref data_struct) => { syn::Data::Struct(ref data_struct) => expand_struct::serialize(
expand_struct::serialize(data_struct, name, &root, &root_attrs.namespaces) data_struct,
} name,
syn::Data::Enum(ref data_enum) => { &root,
expand_enum::serialize(data_enum, name, &root, &root_attrs.namespaces) &root_attrs.namespaces,
} &root_attrs.default_namespace,
),
syn::Data::Enum(ref data_enum) => expand_enum::serialize(
data_enum,
name,
&root,
&root_attrs.namespaces,
&root_attrs.default_namespace,
),
syn::Data::Union(ref _data_union) => unimplemented!(), syn::Data::Union(ref _data_union) => unimplemented!(),
}; };