2018-03-01 09:01:35 +00:00
|
|
|
// Copyright (c) 2017-2018 Emmanuel Gil Peyrot <linkmauve@linkmauve.fr>
|
2017-10-31 21:17:24 +00:00
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
|
|
|
macro_rules! get_attr {
|
2018-12-18 14:32:05 +00:00
|
|
|
($elem:ident, $attr:tt, $type:tt) => {
|
2017-10-31 21:17:24 +00:00
|
|
|
get_attr!($elem, $attr, $type, value, value.parse()?)
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2019-02-24 19:26:40 +00:00
|
|
|
($elem:ident, $attr:tt, OptionEmpty, $value:ident, $func:expr) => {
|
2017-11-24 05:09:25 +00:00
|
|
|
match $elem.attr($attr) {
|
|
|
|
Some("") => None,
|
|
|
|
Some($value) => Some($func),
|
|
|
|
None => None,
|
|
|
|
}
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2019-02-24 19:26:40 +00:00
|
|
|
($elem:ident, $attr:tt, Option, $value:ident, $func:expr) => {
|
2017-10-31 21:17:24 +00:00
|
|
|
match $elem.attr($attr) {
|
|
|
|
Some($value) => Some($func),
|
|
|
|
None => None,
|
|
|
|
}
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2019-02-24 19:26:40 +00:00
|
|
|
($elem:ident, $attr:tt, Required, $value:ident, $func:expr) => {
|
2017-10-31 21:17:24 +00:00
|
|
|
match $elem.attr($attr) {
|
|
|
|
Some($value) => $func,
|
2018-12-18 14:32:05 +00:00
|
|
|
None => {
|
2019-01-13 11:39:51 +00:00
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
2018-12-18 14:32:05 +00:00
|
|
|
"Required attribute '",
|
|
|
|
$attr,
|
|
|
|
"' missing."
|
2018-12-29 17:29:11 +00:00
|
|
|
)));
|
2018-12-18 14:32:05 +00:00
|
|
|
}
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2019-02-28 01:26:10 +00:00
|
|
|
($elem:ident, $attr:tt, RequiredNonEmpty, $value:ident, $func:expr) => {
|
|
|
|
match $elem.attr($attr) {
|
|
|
|
Some("") => {
|
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
|
|
|
"Required attribute '",
|
|
|
|
$attr,
|
|
|
|
"' must not be empty."
|
|
|
|
)));
|
2019-10-22 23:32:41 +00:00
|
|
|
}
|
2019-02-28 01:26:10 +00:00
|
|
|
Some($value) => $func,
|
|
|
|
None => {
|
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
|
|
|
"Required attribute '",
|
|
|
|
$attr,
|
|
|
|
"' missing."
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2019-02-24 19:26:40 +00:00
|
|
|
($elem:ident, $attr:tt, Default, $value:ident, $func:expr) => {
|
2017-10-31 21:17:24 +00:00
|
|
|
match $elem.attr($attr) {
|
|
|
|
Some($value) => $func,
|
2018-05-14 19:03:24 +00:00
|
|
|
None => ::std::default::Default::default(),
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! generate_attribute {
|
2018-05-14 22:18:15 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, {$($(#[$a_meta:meta])* $a:ident => $b:tt),+,}) => (
|
|
|
|
generate_attribute!($(#[$meta])* $elem, $name, {$($(#[$a_meta])* $a => $b),+});
|
2017-10-31 21:17:24 +00:00
|
|
|
);
|
2018-05-14 22:18:15 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, {$($(#[$a_meta:meta])* $a:ident => $b:tt),+,}, Default = $default:ident) => (
|
|
|
|
generate_attribute!($(#[$meta])* $elem, $name, {$($(#[$a_meta])* $a => $b),+}, Default = $default);
|
2017-10-31 21:17:24 +00:00
|
|
|
);
|
2018-05-14 22:18:15 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, {$($(#[$a_meta:meta])* $a:ident => $b:tt),+}) => (
|
|
|
|
$(#[$meta])*
|
2017-10-31 21:17:24 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum $elem {
|
|
|
|
$(
|
2018-05-14 22:18:15 +00:00
|
|
|
$(#[$a_meta])*
|
2017-10-31 21:17:24 +00:00
|
|
|
$a
|
|
|
|
),+
|
|
|
|
}
|
2018-05-14 14:11:22 +00:00
|
|
|
impl ::std::str::FromStr for $elem {
|
2019-01-13 11:39:51 +00:00
|
|
|
type Err = crate::util::error::Error;
|
|
|
|
fn from_str(s: &str) -> Result<$elem, crate::util::error::Error> {
|
2017-10-31 21:17:24 +00:00
|
|
|
Ok(match s {
|
|
|
|
$($b => $elem::$a),+,
|
2019-01-13 11:39:51 +00:00
|
|
|
_ => return Err(crate::util::error::Error::ParseError(concat!("Unknown value for '", $name, "' attribute."))),
|
2017-10-31 21:17:24 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2019-10-18 11:05:33 +00:00
|
|
|
impl std::fmt::Display for $elem {
|
|
|
|
fn fmt(&self, fmt: &mut std::fmt::Formatter) -> Result<(), std::fmt::Error> {
|
|
|
|
write!(fmt, "{}", match self {
|
|
|
|
$($elem::$a => $b),+
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2018-05-14 14:12:56 +00:00
|
|
|
impl ::minidom::IntoAttributeValue for $elem {
|
2017-10-31 21:17:24 +00:00
|
|
|
fn into_attribute_value(self) -> Option<String> {
|
|
|
|
Some(String::from(match self {
|
|
|
|
$($elem::$a => $b),+
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2018-05-14 22:18:15 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, {$($(#[$a_meta:meta])* $a:ident => $b:tt),+}, Default = $default:ident) => (
|
|
|
|
$(#[$meta])*
|
2017-10-31 21:17:24 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum $elem {
|
|
|
|
$(
|
2018-05-14 22:18:15 +00:00
|
|
|
$(#[$a_meta])*
|
2017-10-31 21:17:24 +00:00
|
|
|
$a
|
|
|
|
),+
|
|
|
|
}
|
2018-05-14 14:11:22 +00:00
|
|
|
impl ::std::str::FromStr for $elem {
|
2019-01-13 11:39:51 +00:00
|
|
|
type Err = crate::util::error::Error;
|
|
|
|
fn from_str(s: &str) -> Result<$elem, crate::util::error::Error> {
|
2017-10-31 21:17:24 +00:00
|
|
|
Ok(match s {
|
|
|
|
$($b => $elem::$a),+,
|
2019-01-13 11:39:51 +00:00
|
|
|
_ => return Err(crate::util::error::Error::ParseError(concat!("Unknown value for '", $name, "' attribute."))),
|
2017-10-31 21:17:24 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2018-05-14 14:12:56 +00:00
|
|
|
impl ::minidom::IntoAttributeValue for $elem {
|
2017-10-31 21:17:24 +00:00
|
|
|
#[allow(unreachable_patterns)]
|
|
|
|
fn into_attribute_value(self) -> Option<String> {
|
|
|
|
Some(String::from(match self {
|
|
|
|
$elem::$default => return None,
|
|
|
|
$($elem::$a => $b),+
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
}
|
2018-05-14 19:03:24 +00:00
|
|
|
impl ::std::default::Default for $elem {
|
2017-10-31 21:17:24 +00:00
|
|
|
fn default() -> $elem {
|
|
|
|
$elem::$default
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2019-01-12 21:50:22 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, ($(#[$meta_symbol:meta])* $symbol:ident => $value:tt)) => (
|
|
|
|
$(#[$meta])*
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum $elem {
|
|
|
|
$(#[$meta_symbol])*
|
|
|
|
$symbol,
|
|
|
|
/// Value when absent.
|
|
|
|
None,
|
|
|
|
}
|
|
|
|
impl ::std::str::FromStr for $elem {
|
2019-01-13 11:39:51 +00:00
|
|
|
type Err = crate::util::error::Error;
|
|
|
|
fn from_str(s: &str) -> Result<Self, crate::util::error::Error> {
|
2019-01-12 21:50:22 +00:00
|
|
|
Ok(match s {
|
|
|
|
$value => $elem::$symbol,
|
2019-01-13 11:39:51 +00:00
|
|
|
_ => return Err(crate::util::error::Error::ParseError(concat!("Unknown value for '", $name, "' attribute."))),
|
2019-01-12 21:50:22 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl ::minidom::IntoAttributeValue for $elem {
|
|
|
|
fn into_attribute_value(self) -> Option<String> {
|
|
|
|
match self {
|
|
|
|
$elem::$symbol => Some(String::from($value)),
|
|
|
|
$elem::None => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl ::std::default::Default for $elem {
|
|
|
|
fn default() -> $elem {
|
|
|
|
$elem::None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2018-05-14 22:18:15 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, bool) => (
|
|
|
|
$(#[$meta])*
|
2018-05-14 19:04:16 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum $elem {
|
|
|
|
/// True value, represented by either 'true' or '1'.
|
|
|
|
True,
|
|
|
|
/// False value, represented by either 'false' or '0'.
|
|
|
|
False,
|
|
|
|
}
|
|
|
|
impl ::std::str::FromStr for $elem {
|
2019-01-13 11:39:51 +00:00
|
|
|
type Err = crate::util::error::Error;
|
|
|
|
fn from_str(s: &str) -> Result<Self, crate::util::error::Error> {
|
2018-05-14 19:04:16 +00:00
|
|
|
Ok(match s {
|
|
|
|
"true" | "1" => $elem::True,
|
|
|
|
"false" | "0" => $elem::False,
|
2019-01-13 11:39:51 +00:00
|
|
|
_ => return Err(crate::util::error::Error::ParseError(concat!("Unknown value for '", $name, "' attribute."))),
|
2018-05-14 19:04:16 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl ::minidom::IntoAttributeValue for $elem {
|
|
|
|
fn into_attribute_value(self) -> Option<String> {
|
|
|
|
match self {
|
|
|
|
$elem::True => Some(String::from("true")),
|
|
|
|
$elem::False => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl ::std::default::Default for $elem {
|
|
|
|
fn default() -> $elem {
|
|
|
|
$elem::False
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2019-02-28 02:10:21 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $type:tt, Default = $default:expr) => (
|
|
|
|
$(#[$meta])*
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub struct $elem(pub $type);
|
|
|
|
impl ::std::str::FromStr for $elem {
|
|
|
|
type Err = crate::util::error::Error;
|
|
|
|
fn from_str(s: &str) -> Result<Self, crate::util::error::Error> {
|
|
|
|
Ok($elem($type::from_str(s)?))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl ::minidom::IntoAttributeValue for $elem {
|
|
|
|
fn into_attribute_value(self) -> Option<String> {
|
|
|
|
match self {
|
|
|
|
$elem($default) => None,
|
|
|
|
$elem(value) => Some(format!("{}", value)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl ::std::default::Default for $elem {
|
|
|
|
fn default() -> $elem {
|
|
|
|
$elem($default)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
|
|
|
|
2017-11-24 04:27:35 +00:00
|
|
|
macro_rules! generate_element_enum {
|
2018-05-14 14:30:28 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, {$($(#[$enum_meta:meta])* $enum:ident => $enum_name:tt),+,}) => (
|
2017-11-24 04:27:35 +00:00
|
|
|
generate_element_enum!($(#[$meta])* $elem, $name, $ns, {$($(#[$enum_meta])* $enum => $enum_name),+});
|
|
|
|
);
|
2018-05-14 14:30:28 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, {$($(#[$enum_meta:meta])* $enum:ident => $enum_name:tt),+}) => (
|
2017-11-24 04:27:35 +00:00
|
|
|
$(#[$meta])*
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum $elem {
|
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
$(#[$enum_meta])*
|
|
|
|
$enum
|
2017-11-24 04:27:35 +00:00
|
|
|
),+
|
|
|
|
}
|
2019-09-25 08:28:44 +00:00
|
|
|
impl ::std::convert::TryFrom<crate::Element> for $elem {
|
2019-04-12 08:58:42 +00:00
|
|
|
type Error = crate::util::error::Error;
|
2019-09-25 08:28:44 +00:00
|
|
|
fn try_from(elem: crate::Element) -> Result<$elem, crate::util::error::Error> {
|
2017-11-24 04:27:35 +00:00
|
|
|
check_ns_only!(elem, $name, $ns);
|
|
|
|
check_no_children!(elem, $name);
|
|
|
|
check_no_attributes!(elem, $name);
|
|
|
|
Ok(match elem.name() {
|
|
|
|
$($enum_name => $elem::$enum,)+
|
2019-01-13 11:39:51 +00:00
|
|
|
_ => return Err(crate::util::error::Error::ParseError(concat!("This is not a ", $name, " element."))),
|
2017-11-24 04:27:35 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2019-09-25 08:28:44 +00:00
|
|
|
impl From<$elem> for crate::Element {
|
|
|
|
fn from(elem: $elem) -> crate::Element {
|
|
|
|
crate::Element::builder(
|
2018-12-29 17:29:11 +00:00
|
|
|
match elem {
|
|
|
|
$($elem::$enum => $enum_name,)+
|
2020-03-28 12:07:26 +00:00
|
|
|
},
|
|
|
|
crate::ns::$ns,
|
2018-12-29 17:29:11 +00:00
|
|
|
)
|
|
|
|
.build()
|
2017-11-24 04:27:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-11-24 05:44:58 +00:00
|
|
|
macro_rules! generate_attribute_enum {
|
2018-05-14 14:30:28 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, $attr:tt, {$($(#[$enum_meta:meta])* $enum:ident => $enum_name:tt),+,}) => (
|
2017-11-24 05:44:58 +00:00
|
|
|
generate_attribute_enum!($(#[$meta])* $elem, $name, $ns, $attr, {$($(#[$enum_meta])* $enum => $enum_name),+});
|
|
|
|
);
|
2018-05-14 14:30:28 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, $attr:tt, {$($(#[$enum_meta:meta])* $enum:ident => $enum_name:tt),+}) => (
|
2017-11-24 05:44:58 +00:00
|
|
|
$(#[$meta])*
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum $elem {
|
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
$(#[$enum_meta])*
|
|
|
|
$enum
|
2017-11-24 05:44:58 +00:00
|
|
|
),+
|
|
|
|
}
|
2019-09-25 08:28:44 +00:00
|
|
|
impl ::std::convert::TryFrom<crate::Element> for $elem {
|
2019-04-12 08:58:42 +00:00
|
|
|
type Error = crate::util::error::Error;
|
2019-09-25 08:28:44 +00:00
|
|
|
fn try_from(elem: crate::Element) -> Result<$elem, crate::util::error::Error> {
|
2017-11-24 05:44:58 +00:00
|
|
|
check_ns_only!(elem, $name, $ns);
|
|
|
|
check_no_children!(elem, $name);
|
|
|
|
check_no_unknown_attributes!(elem, $name, [$attr]);
|
2019-02-24 19:48:19 +00:00
|
|
|
Ok(match get_attr!(elem, $attr, Required) {
|
2017-11-24 05:44:58 +00:00
|
|
|
$($enum_name => $elem::$enum,)+
|
2019-01-13 11:39:51 +00:00
|
|
|
_ => return Err(crate::util::error::Error::ParseError(concat!("Invalid ", $name, " ", $attr, " value."))),
|
2017-11-24 05:44:58 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2019-09-25 08:28:44 +00:00
|
|
|
impl From<$elem> for crate::Element {
|
|
|
|
fn from(elem: $elem) -> crate::Element {
|
2020-03-28 12:07:26 +00:00
|
|
|
crate::Element::builder($name, crate::ns::$ns)
|
2018-12-29 17:29:11 +00:00
|
|
|
.attr($attr, match elem {
|
|
|
|
$($elem::$enum => $enum_name,)+
|
|
|
|
})
|
|
|
|
.build()
|
2017-11-24 05:44:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-10-31 21:17:24 +00:00
|
|
|
macro_rules! check_self {
|
2018-12-18 14:32:05 +00:00
|
|
|
($elem:ident, $name:tt, $ns:ident) => {
|
2017-10-31 21:17:24 +00:00
|
|
|
check_self!($elem, $name, $ns, $name);
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
|
|
|
($elem:ident, $name:tt, $ns:ident, $pretty_name:tt) => {
|
2018-12-18 14:27:30 +00:00
|
|
|
if !$elem.is($name, crate::ns::$ns) {
|
2019-01-13 11:39:51 +00:00
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
2018-12-18 14:32:05 +00:00
|
|
|
"This is not a ",
|
|
|
|
$pretty_name,
|
|
|
|
" element."
|
|
|
|
)));
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! check_ns_only {
|
2018-12-18 14:32:05 +00:00
|
|
|
($elem:ident, $name:tt, $ns:ident) => {
|
2018-12-18 14:27:30 +00:00
|
|
|
if !$elem.has_ns(crate::ns::$ns) {
|
2019-01-13 11:39:51 +00:00
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
2018-12-18 14:32:05 +00:00
|
|
|
"This is not a ",
|
|
|
|
$name,
|
|
|
|
" element."
|
|
|
|
)));
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! check_no_children {
|
2018-12-18 14:32:05 +00:00
|
|
|
($elem:ident, $name:tt) => {
|
2019-01-12 21:00:46 +00:00
|
|
|
#[cfg(not(feature = "disable-validation"))]
|
2017-10-31 21:17:24 +00:00
|
|
|
for _ in $elem.children() {
|
2019-01-13 11:39:51 +00:00
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
2018-12-18 14:32:05 +00:00
|
|
|
"Unknown child in ",
|
|
|
|
$name,
|
|
|
|
" element."
|
|
|
|
)));
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! check_no_attributes {
|
2018-12-18 14:32:05 +00:00
|
|
|
($elem:ident, $name:tt) => {
|
2019-01-12 21:00:46 +00:00
|
|
|
#[cfg(not(feature = "disable-validation"))]
|
2018-05-12 18:25:59 +00:00
|
|
|
for _ in $elem.attrs() {
|
2019-01-13 11:39:51 +00:00
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
2018-12-18 14:32:05 +00:00
|
|
|
"Unknown attribute in ",
|
|
|
|
$name,
|
|
|
|
" element."
|
|
|
|
)));
|
2018-05-12 18:25:59 +00:00
|
|
|
}
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! check_no_unknown_attributes {
|
|
|
|
($elem:ident, $name:tt, [$($attr:tt),*]) => (
|
2019-01-12 21:00:46 +00:00
|
|
|
#[cfg(not(feature = "disable-validation"))]
|
2017-10-31 21:17:24 +00:00
|
|
|
for (_attr, _) in $elem.attrs() {
|
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
if _attr == $attr {
|
|
|
|
continue;
|
|
|
|
}
|
2017-10-31 21:17:24 +00:00
|
|
|
)*
|
2019-01-13 11:39:51 +00:00
|
|
|
return Err(crate::util::error::Error::ParseError(concat!("Unknown attribute in ", $name, " element.")));
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! generate_empty_element {
|
2018-05-14 14:30:28 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident) => (
|
2017-10-31 21:17:24 +00:00
|
|
|
$(#[$meta])*
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct $elem;
|
|
|
|
|
2019-09-25 08:28:44 +00:00
|
|
|
impl ::std::convert::TryFrom<crate::Element> for $elem {
|
2019-04-12 08:58:42 +00:00
|
|
|
type Error = crate::util::error::Error;
|
2017-10-31 21:17:24 +00:00
|
|
|
|
2019-09-25 08:28:44 +00:00
|
|
|
fn try_from(elem: crate::Element) -> Result<$elem, crate::util::error::Error> {
|
2017-10-31 21:17:24 +00:00
|
|
|
check_self!(elem, $name, $ns);
|
|
|
|
check_no_children!(elem, $name);
|
|
|
|
check_no_attributes!(elem, $name);
|
|
|
|
Ok($elem)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-25 08:28:44 +00:00
|
|
|
impl From<$elem> for crate::Element {
|
|
|
|
fn from(_: $elem) -> crate::Element {
|
2020-03-28 12:07:26 +00:00
|
|
|
crate::Element::builder($name, crate::ns::$ns)
|
2018-12-29 17:29:11 +00:00
|
|
|
.build()
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! generate_id {
|
2018-07-02 11:47:32 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident) => (
|
|
|
|
$(#[$meta])*
|
2017-10-31 21:17:24 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
|
|
|
pub struct $elem(pub String);
|
2018-05-14 14:11:22 +00:00
|
|
|
impl ::std::str::FromStr for $elem {
|
2019-01-13 11:39:51 +00:00
|
|
|
type Err = crate::util::error::Error;
|
|
|
|
fn from_str(s: &str) -> Result<$elem, crate::util::error::Error> {
|
2017-10-31 21:17:24 +00:00
|
|
|
// TODO: add a way to parse that differently when needed.
|
|
|
|
Ok($elem(String::from(s)))
|
|
|
|
}
|
|
|
|
}
|
2018-05-14 14:12:56 +00:00
|
|
|
impl ::minidom::IntoAttributeValue for $elem {
|
2017-10-31 21:17:24 +00:00
|
|
|
fn into_attribute_value(self) -> Option<String> {
|
|
|
|
Some(self.0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! generate_elem_id {
|
2018-05-14 15:32:15 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident) => (
|
2019-09-05 13:34:21 +00:00
|
|
|
generate_elem_id!($(#[$meta])* $elem, $name, $ns, String);
|
2018-05-14 14:11:22 +00:00
|
|
|
impl ::std::str::FromStr for $elem {
|
2019-01-13 11:39:51 +00:00
|
|
|
type Err = crate::util::error::Error;
|
|
|
|
fn from_str(s: &str) -> Result<$elem, crate::util::error::Error> {
|
2017-10-31 21:17:24 +00:00
|
|
|
// TODO: add a way to parse that differently when needed.
|
|
|
|
Ok($elem(String::from(s)))
|
|
|
|
}
|
|
|
|
}
|
2019-09-05 13:34:21 +00:00
|
|
|
);
|
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, $type:ty) => (
|
|
|
|
$(#[$meta])*
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
|
|
|
pub struct $elem(pub $type);
|
2019-09-25 08:28:44 +00:00
|
|
|
impl ::std::convert::TryFrom<crate::Element> for $elem {
|
2019-04-12 08:58:42 +00:00
|
|
|
type Error = crate::util::error::Error;
|
2019-09-25 08:28:44 +00:00
|
|
|
fn try_from(elem: crate::Element) -> Result<$elem, crate::util::error::Error> {
|
2017-11-24 05:09:25 +00:00
|
|
|
check_self!(elem, $name, $ns);
|
|
|
|
check_no_children!(elem, $name);
|
|
|
|
check_no_attributes!(elem, $name);
|
|
|
|
// TODO: add a way to parse that differently when needed.
|
2019-09-05 13:34:21 +00:00
|
|
|
Ok($elem(elem.text().parse()?))
|
2017-11-24 05:09:25 +00:00
|
|
|
}
|
|
|
|
}
|
2019-09-25 08:28:44 +00:00
|
|
|
impl From<$elem> for crate::Element {
|
|
|
|
fn from(elem: $elem) -> crate::Element {
|
2020-03-28 12:07:26 +00:00
|
|
|
crate::Element::builder($name, crate::ns::$ns)
|
2019-09-05 13:34:21 +00:00
|
|
|
.append(elem.0.to_string())
|
2018-12-29 17:29:11 +00:00
|
|
|
.build()
|
2017-10-31 21:17:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2017-11-16 20:17:11 +00:00
|
|
|
|
2019-02-24 19:26:40 +00:00
|
|
|
macro_rules! decl_attr {
|
|
|
|
(OptionEmpty, $type:ty) => (
|
|
|
|
Option<$type>
|
|
|
|
);
|
|
|
|
(Option, $type:ty) => (
|
|
|
|
Option<$type>
|
|
|
|
);
|
|
|
|
(Required, $type:ty) => (
|
|
|
|
$type
|
|
|
|
);
|
2019-02-28 01:26:10 +00:00
|
|
|
(RequiredNonEmpty, $type:ty) => (
|
|
|
|
$type
|
|
|
|
);
|
2019-02-24 19:26:40 +00:00
|
|
|
(Default, $type:ty) => (
|
|
|
|
$type
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-05-28 14:29:51 +00:00
|
|
|
macro_rules! start_decl {
|
|
|
|
(Vec, $type:ty) => (
|
|
|
|
Vec<$type>
|
2018-05-14 23:55:16 +00:00
|
|
|
);
|
2018-05-28 14:29:51 +00:00
|
|
|
(Option, $type:ty) => (
|
|
|
|
Option<$type>
|
2018-05-14 19:04:16 +00:00
|
|
|
);
|
2018-05-28 14:29:51 +00:00
|
|
|
(Required, $type:ty) => (
|
|
|
|
$type
|
|
|
|
);
|
2019-09-08 13:53:55 +00:00
|
|
|
(Present, $type:ty) => (
|
|
|
|
bool
|
|
|
|
);
|
2018-05-28 14:29:51 +00:00
|
|
|
}
|
2017-11-16 21:00:01 +00:00
|
|
|
|
2018-05-28 14:29:51 +00:00
|
|
|
macro_rules! start_parse_elem {
|
2018-12-18 14:32:05 +00:00
|
|
|
($temp:ident: Vec) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
let mut $temp = Vec::new();
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
|
|
|
($temp:ident: Option) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
let mut $temp = None;
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
|
|
|
($temp:ident: Required) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
let mut $temp = None;
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2019-09-08 13:53:55 +00:00
|
|
|
($temp:ident: Present) => {
|
|
|
|
let mut $temp = false;
|
|
|
|
};
|
2018-05-28 14:29:51 +00:00
|
|
|
}
|
2018-05-14 19:04:16 +00:00
|
|
|
|
2018-05-28 14:29:51 +00:00
|
|
|
macro_rules! do_parse {
|
2018-12-18 14:32:05 +00:00
|
|
|
($elem:ident, Element) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
$elem.clone()
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
|
|
|
($elem:ident, String) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
$elem.text()
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
|
|
|
($elem:ident, $constructor:ident) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
$constructor::try_from($elem.clone())?
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2018-05-28 14:29:51 +00:00
|
|
|
}
|
2018-05-14 19:04:16 +00:00
|
|
|
|
2018-05-28 14:29:51 +00:00
|
|
|
macro_rules! do_parse_elem {
|
2018-12-18 14:32:05 +00:00
|
|
|
($temp:ident: Vec = $constructor:ident => $elem:ident, $name:tt, $parent_name:tt) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
$temp.push(do_parse!($elem, $constructor));
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
|
|
|
($temp:ident: Option = $constructor:ident => $elem:ident, $name:tt, $parent_name:tt) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
if $temp.is_some() {
|
2019-01-13 11:39:51 +00:00
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
2018-12-18 14:32:05 +00:00
|
|
|
"Element ",
|
|
|
|
$parent_name,
|
|
|
|
" must not have more than one ",
|
|
|
|
$name,
|
|
|
|
" child."
|
|
|
|
)));
|
2018-05-14 19:04:16 +00:00
|
|
|
}
|
2018-05-28 14:29:51 +00:00
|
|
|
$temp = Some(do_parse!($elem, $constructor));
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
|
|
|
($temp:ident: Required = $constructor:ident => $elem:ident, $name:tt, $parent_name:tt) => {
|
2018-05-28 15:04:40 +00:00
|
|
|
if $temp.is_some() {
|
2019-01-13 11:39:51 +00:00
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
2018-12-18 14:32:05 +00:00
|
|
|
"Element ",
|
|
|
|
$parent_name,
|
|
|
|
" must not have more than one ",
|
|
|
|
$name,
|
|
|
|
" child."
|
|
|
|
)));
|
2018-05-28 15:04:40 +00:00
|
|
|
}
|
2018-05-28 14:29:51 +00:00
|
|
|
$temp = Some(do_parse!($elem, $constructor));
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2019-09-08 13:53:55 +00:00
|
|
|
($temp:ident: Present = $constructor:ident => $elem:ident, $name:tt, $parent_name:tt) => {
|
|
|
|
if $temp {
|
|
|
|
return Err(crate::util::error::Error::ParseError(concat!(
|
|
|
|
"Element ",
|
|
|
|
$parent_name,
|
|
|
|
" must not have more than one ",
|
|
|
|
$name,
|
|
|
|
" child."
|
|
|
|
)));
|
|
|
|
}
|
|
|
|
$temp = true;
|
|
|
|
};
|
2018-05-28 14:29:51 +00:00
|
|
|
}
|
2018-05-14 19:04:16 +00:00
|
|
|
|
2018-05-28 14:29:51 +00:00
|
|
|
macro_rules! finish_parse_elem {
|
2018-12-18 14:32:05 +00:00
|
|
|
($temp:ident: Vec = $name:tt, $parent_name:tt) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
$temp
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
|
|
|
($temp:ident: Option = $name:tt, $parent_name:tt) => {
|
2018-05-28 14:29:51 +00:00
|
|
|
$temp
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
|
|
|
($temp:ident: Required = $name:tt, $parent_name:tt) => {
|
2019-01-13 11:39:51 +00:00
|
|
|
$temp.ok_or(crate::util::error::Error::ParseError(concat!(
|
2018-12-18 14:32:05 +00:00
|
|
|
"Missing child ",
|
|
|
|
$name,
|
|
|
|
" in ",
|
|
|
|
$parent_name,
|
|
|
|
" element."
|
|
|
|
)))?
|
|
|
|
};
|
2019-09-08 13:53:55 +00:00
|
|
|
($temp:ident: Present = $name:tt, $parent_name:tt) => {
|
|
|
|
$temp
|
|
|
|
};
|
2018-05-28 14:29:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! generate_serialiser {
|
2019-07-24 22:20:38 +00:00
|
|
|
($builder:ident, $parent:ident, $elem:ident, Required, String, ($name:tt, $ns:ident)) => {
|
2019-09-06 14:03:58 +00:00
|
|
|
$builder.append(
|
2020-03-28 12:07:26 +00:00
|
|
|
crate::Element::builder($name, crate::ns::$ns)
|
2019-10-22 23:32:41 +00:00
|
|
|
.append(::minidom::Node::Text($parent.$elem)),
|
2019-07-24 22:20:38 +00:00
|
|
|
)
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2019-07-24 22:20:38 +00:00
|
|
|
($builder:ident, $parent:ident, $elem:ident, Option, String, ($name:tt, $ns:ident)) => {
|
|
|
|
$builder.append_all($parent.$elem.map(|elem| {
|
2020-04-02 20:45:20 +00:00
|
|
|
crate::Element::builder($name, crate::ns::$ns).append(::minidom::Node::Text(elem))
|
2019-10-22 23:32:41 +00:00
|
|
|
}))
|
2019-07-24 22:20:38 +00:00
|
|
|
};
|
2019-10-12 15:10:36 +00:00
|
|
|
($builder:ident, $parent:ident, $elem:ident, Option, $constructor:ident, ($name:tt, *)) => {
|
2019-11-27 17:17:10 +00:00
|
|
|
$builder.append_all(
|
|
|
|
$parent
|
|
|
|
.$elem
|
|
|
|
.map(|elem| ::minidom::Node::Element(crate::Element::from(elem))),
|
|
|
|
)
|
2019-10-12 15:10:36 +00:00
|
|
|
};
|
2019-07-24 22:20:38 +00:00
|
|
|
($builder:ident, $parent:ident, $elem:ident, Option, $constructor:ident, ($name:tt, $ns:ident)) => {
|
2019-11-27 17:17:10 +00:00
|
|
|
$builder.append_all(
|
|
|
|
$parent
|
|
|
|
.$elem
|
|
|
|
.map(|elem| ::minidom::Node::Element(crate::Element::from(elem))),
|
|
|
|
)
|
2019-07-24 22:20:38 +00:00
|
|
|
};
|
|
|
|
($builder:ident, $parent:ident, $elem:ident, Vec, $constructor:ident, ($name:tt, $ns:ident)) => {
|
|
|
|
$builder.append_all($parent.$elem.into_iter())
|
|
|
|
};
|
2019-09-08 13:53:55 +00:00
|
|
|
($builder:ident, $parent:ident, $elem:ident, Present, $constructor:ident, ($name:tt, $ns:ident)) => {
|
2019-10-22 23:32:41 +00:00
|
|
|
$builder.append(::minidom::Node::Element(
|
2020-03-28 12:07:26 +00:00
|
|
|
crate::Element::builder($name, crate::ns::$ns).build(),
|
2019-10-22 23:32:41 +00:00
|
|
|
))
|
2019-09-08 13:53:55 +00:00
|
|
|
};
|
2019-07-24 22:20:38 +00:00
|
|
|
($builder:ident, $parent:ident, $elem:ident, $_:ident, $constructor:ident, ($name:tt, $ns:ident)) => {
|
2019-10-22 23:32:41 +00:00
|
|
|
$builder.append(::minidom::Node::Element(crate::Element::from(
|
|
|
|
$parent.$elem,
|
|
|
|
)))
|
2018-12-18 14:32:05 +00:00
|
|
|
};
|
2018-05-28 14:29:51 +00:00
|
|
|
}
|
|
|
|
|
2019-10-12 15:10:36 +00:00
|
|
|
macro_rules! generate_child_test {
|
|
|
|
($child:ident, $name:tt, *) => {
|
2019-11-09 00:31:36 +00:00
|
|
|
$child.is($name, ::minidom::NSChoice::Any)
|
2019-10-12 15:10:36 +00:00
|
|
|
};
|
|
|
|
($child:ident, $name:tt, $ns:tt) => {
|
|
|
|
$child.is($name, crate::ns::$ns)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-05-28 14:42:35 +00:00
|
|
|
macro_rules! generate_element {
|
2019-02-24 19:26:40 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, attributes: [$($(#[$attr_meta:meta])* $attr:ident: $attr_action:tt<$attr_type:ty> = $attr_name:tt),+,]) => (
|
|
|
|
generate_element!($(#[$meta])* $elem, $name, $ns, attributes: [$($(#[$attr_meta])* $attr: $attr_action<$attr_type> = $attr_name),*], children: []);
|
2018-05-28 14:42:35 +00:00
|
|
|
);
|
2019-02-24 19:26:40 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, attributes: [$($(#[$attr_meta:meta])* $attr:ident: $attr_action:tt<$attr_type:ty> = $attr_name:tt),+]) => (
|
|
|
|
generate_element!($(#[$meta])* $elem, $name, $ns, attributes: [$($(#[$attr_meta])* $attr: $attr_action<$attr_type> = $attr_name),*], children: []);
|
2018-05-28 14:42:35 +00:00
|
|
|
);
|
2019-10-12 15:10:36 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, children: [$($(#[$child_meta:meta])* $child_ident:ident: $coucou:tt<$child_type:ty> = ($child_name:tt, $child_ns:tt) => $child_constructor:ident),*]) => (
|
2018-05-28 14:45:13 +00:00
|
|
|
generate_element!($(#[$meta])* $elem, $name, $ns, attributes: [], children: [$($(#[$child_meta])* $child_ident: $coucou<$child_type> = ($child_name, $child_ns) => $child_constructor),*]);
|
2018-05-28 14:29:51 +00:00
|
|
|
);
|
2019-10-12 15:10:36 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, attributes: [$($(#[$attr_meta:meta])* $attr:ident: $attr_action:tt<$attr_type:ty> = $attr_name:tt),*,], children: [$($(#[$child_meta:meta])* $child_ident:ident: $coucou:tt<$child_type:ty> = ($child_name:tt, $child_ns:tt) => $child_constructor:ident),*]) => (
|
2019-02-24 19:26:40 +00:00
|
|
|
generate_element!($(#[$meta])* $elem, $name, $ns, attributes: [$($(#[$attr_meta])* $attr: $attr_action<$attr_type> = $attr_name),*], children: [$($(#[$child_meta])* $child_ident: $coucou<$child_type> = ($child_name, $child_ns) => $child_constructor),*]);
|
2018-05-28 14:29:51 +00:00
|
|
|
);
|
2018-08-02 17:07:07 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, text: ($(#[$text_meta:meta])* $text_ident:ident: $codec:ident < $text_type:ty >)) => (
|
|
|
|
generate_element!($(#[$meta])* $elem, $name, $ns, attributes: [], children: [], text: ($(#[$text_meta])* $text_ident: $codec<$text_type>));
|
|
|
|
);
|
2019-02-24 19:26:40 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, attributes: [$($(#[$attr_meta:meta])* $attr:ident: $attr_action:tt<$attr_type:ty> = $attr_name:tt),+], text: ($(#[$text_meta:meta])* $text_ident:ident: $codec:ident < $text_type:ty >)) => (
|
|
|
|
generate_element!($(#[$meta])* $elem, $name, $ns, attributes: [$($(#[$attr_meta])* $attr: $attr_action<$attr_type> = $attr_name),*], children: [], text: ($(#[$text_meta])* $text_ident: $codec<$text_type>));
|
2018-08-02 17:07:07 +00:00
|
|
|
);
|
2019-10-12 15:10:36 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, attributes: [$($(#[$attr_meta:meta])* $attr:ident: $attr_action:tt<$attr_type:ty> = $attr_name:tt),*], children: [$($(#[$child_meta:meta])* $child_ident:ident: $coucou:tt<$child_type:ty> = ($child_name:tt, $child_ns:tt) => $child_constructor:ident),*] $(, text: ($(#[$text_meta:meta])* $text_ident:ident: $codec:ident < $text_type:ty >))*) => (
|
2018-05-14 19:04:16 +00:00
|
|
|
$(#[$meta])*
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct $elem {
|
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
$(#[$attr_meta])*
|
2019-02-24 19:26:40 +00:00
|
|
|
pub $attr: decl_attr!($attr_action, $attr_type),
|
2018-05-14 19:04:16 +00:00
|
|
|
)*
|
2018-05-28 14:29:51 +00:00
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
$(#[$child_meta])*
|
|
|
|
pub $child_ident: start_decl!($coucou, $child_type),
|
2018-05-28 14:29:51 +00:00
|
|
|
)*
|
2018-08-02 17:07:07 +00:00
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
$(#[$text_meta])*
|
|
|
|
pub $text_ident: $text_type,
|
2018-08-02 17:07:07 +00:00
|
|
|
)*
|
2018-05-14 19:04:16 +00:00
|
|
|
}
|
|
|
|
|
2019-09-25 08:28:44 +00:00
|
|
|
impl ::std::convert::TryFrom<crate::Element> for $elem {
|
2019-04-12 08:58:42 +00:00
|
|
|
type Error = crate::util::error::Error;
|
2018-05-14 19:04:16 +00:00
|
|
|
|
2019-09-25 08:28:44 +00:00
|
|
|
fn try_from(elem: crate::Element) -> Result<$elem, crate::util::error::Error> {
|
2018-05-14 19:04:16 +00:00
|
|
|
check_self!(elem, $name, $ns);
|
|
|
|
check_no_unknown_attributes!(elem, $name, [$($attr_name),*]);
|
2018-05-28 14:29:51 +00:00
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
start_parse_elem!($child_ident: $coucou);
|
2018-05-28 14:29:51 +00:00
|
|
|
)*
|
2018-05-28 14:37:22 +00:00
|
|
|
for _child in elem.children() {
|
2018-05-28 14:29:51 +00:00
|
|
|
$(
|
2019-10-12 15:10:36 +00:00
|
|
|
if generate_child_test!(_child, $child_name, $child_ns) {
|
2018-05-28 15:04:40 +00:00
|
|
|
do_parse_elem!($child_ident: $coucou = $child_constructor => _child, $child_name, $name);
|
2018-05-14 19:04:16 +00:00
|
|
|
continue;
|
|
|
|
}
|
2018-05-28 14:29:51 +00:00
|
|
|
)*
|
2019-01-13 11:39:51 +00:00
|
|
|
return Err(crate::util::error::Error::ParseError(concat!("Unknown child in ", $name, " element.")));
|
2018-05-14 19:04:16 +00:00
|
|
|
}
|
|
|
|
Ok($elem {
|
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
$attr: get_attr!(elem, $attr_name, $attr_action),
|
2018-05-14 19:04:16 +00:00
|
|
|
)*
|
2018-05-28 14:29:51 +00:00
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
$child_ident: finish_parse_elem!($child_ident: $coucou = $child_name, $name),
|
2018-05-28 14:29:51 +00:00
|
|
|
)*
|
2018-08-02 17:07:07 +00:00
|
|
|
$(
|
2018-12-29 17:29:11 +00:00
|
|
|
$text_ident: $codec::decode(&elem.text())?,
|
2018-08-02 17:07:07 +00:00
|
|
|
)*
|
2018-05-14 19:04:16 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-25 08:28:44 +00:00
|
|
|
impl From<$elem> for crate::Element {
|
|
|
|
fn from(elem: $elem) -> crate::Element {
|
2020-03-28 12:07:26 +00:00
|
|
|
let mut builder = crate::Element::builder($name, crate::ns::$ns);
|
2019-07-24 22:20:38 +00:00
|
|
|
$(
|
|
|
|
builder = builder.attr($attr_name, elem.$attr);
|
|
|
|
)*
|
|
|
|
$(
|
|
|
|
builder = generate_serialiser!(builder, elem, $child_ident, $coucou, $child_constructor, ($child_name, $child_ns));
|
|
|
|
)*
|
|
|
|
$(
|
|
|
|
builder = builder.append_all($codec::encode(&elem.$text_ident).map(::minidom::Node::Text).into_iter());
|
|
|
|
)*
|
|
|
|
|
|
|
|
builder.build()
|
|
|
|
}
|
|
|
|
}
|
2018-05-14 19:04:16 +00:00
|
|
|
);
|
2017-11-16 21:00:01 +00:00
|
|
|
}
|
2018-10-26 12:26:16 +00:00
|
|
|
|
2018-12-18 14:43:49 +00:00
|
|
|
#[cfg(test)]
|
2018-10-26 12:26:16 +00:00
|
|
|
macro_rules! assert_size (
|
|
|
|
($t:ty, $sz:expr) => (
|
|
|
|
assert_eq!(::std::mem::size_of::<$t>(), $sz);
|
|
|
|
);
|
|
|
|
);
|
2019-01-27 17:57:25 +00:00
|
|
|
|
|
|
|
// TODO: move that to src/pubsub/mod.rs, once we figure out how to use macros from there.
|
|
|
|
macro_rules! impl_pubsub_item {
|
|
|
|
($item:ident, $ns:ident) => {
|
2019-07-17 20:30:49 +00:00
|
|
|
impl ::std::convert::TryFrom<crate::Element> for $item {
|
2019-04-12 08:58:42 +00:00
|
|
|
type Error = Error;
|
2019-01-27 17:57:25 +00:00
|
|
|
|
|
|
|
fn try_from(elem: crate::Element) -> Result<$item, Error> {
|
|
|
|
check_self!(elem, "item", $ns);
|
|
|
|
check_no_unknown_attributes!(elem, "item", ["id", "publisher"]);
|
|
|
|
let mut payloads = elem.children().cloned().collect::<Vec<_>>();
|
|
|
|
let payload = payloads.pop();
|
|
|
|
if !payloads.is_empty() {
|
|
|
|
return Err(Error::ParseError(
|
|
|
|
"More than a single payload in item element.",
|
|
|
|
));
|
|
|
|
}
|
|
|
|
Ok($item(crate::pubsub::Item {
|
2019-02-24 19:48:19 +00:00
|
|
|
id: get_attr!(elem, "id", Option),
|
|
|
|
publisher: get_attr!(elem, "publisher", Option),
|
2019-01-27 17:57:25 +00:00
|
|
|
payload,
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<$item> for crate::Element {
|
|
|
|
fn from(item: $item) -> crate::Element {
|
2020-03-28 12:07:26 +00:00
|
|
|
crate::Element::builder("item", ns::$ns)
|
2019-01-27 17:57:25 +00:00
|
|
|
.attr("id", item.0.id)
|
|
|
|
.attr("publisher", item.0.publisher)
|
2019-09-06 14:03:58 +00:00
|
|
|
.append_all(item.0.payload)
|
2019-01-27 17:57:25 +00:00
|
|
|
.build()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ::std::ops::Deref for $item {
|
|
|
|
type Target = crate::pubsub::Item;
|
|
|
|
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ::std::ops::DerefMut for $item {
|
|
|
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
|
|
|
&mut self.0
|
|
|
|
}
|
|
|
|
}
|
2019-10-22 23:32:41 +00:00
|
|
|
};
|
2019-01-27 17:57:25 +00:00
|
|
|
}
|