Merge pull request #47 from media-io/default_namespace
add default_namespace attribute
This commit is contained in:
commit
1a67c4907d
@ -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};
|
||||||
|
|||||||
@ -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;
|
||||||
|
|||||||
@ -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;
|
||||||
|
|||||||
@ -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)]
|
||||||
|
|||||||
@ -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
|
||||||
);
|
);
|
||||||
|
|||||||
@ -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)
|
||||||
))
|
))
|
||||||
|
|||||||
@ -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)
|
||||||
))
|
))
|
||||||
|
|||||||
@ -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!(),
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user