xmpp-rs/parsers/src/util/macros.rs

835 lines
29 KiB
Rust
Raw Normal View History

// 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
};
($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
};
($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
};
($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 => {
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-02-28 01:26:10 +00:00
Some($value) => $func,
None => {
return Err(crate::util::error::Error::ParseError(concat!(
"Required attribute '",
$attr,
"' missing."
)));
}
}
};
($elem:ident, $attr:tt, Default, $value:ident, $func:expr) => {
2017-10-31 21:17:24 +00:00
match $elem.attr($attr) {
Some($value) => $func,
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 {
($(#[$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 {
$(
$(#[$a_meta])*
2017-10-31 21:17:24 +00:00
$a
),+
}
impl ::std::str::FromStr for $elem {
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),+,
_ => return Err(crate::util::error::Error::ParseError(concat!("Unknown value for '", $name, "' attribute."))),
2017-10-31 21:17:24 +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),+
})
}
}
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),+
}))
}
}
);
($(#[$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 {
$(
$(#[$a_meta])*
2017-10-31 21:17:24 +00:00
$a
),+
}
impl ::std::str::FromStr for $elem {
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),+,
_ => return Err(crate::util::error::Error::ParseError(concat!("Unknown value for '", $name, "' attribute."))),
2017-10-31 21:17:24 +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),+
}))
}
}
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 {
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,
_ => 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
}
}
);
($(#[$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 {
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,
_ => 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
}
}
);
($(#[$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
}
macro_rules! generate_element_enum {
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, {$($(#[$enum_meta:meta])* $enum:ident => $enum_name:tt),+$(,)?}) => (
$(#[$meta])*
#[derive(Debug, Clone, PartialEq)]
pub enum $elem {
$(
2018-12-29 17:29:11 +00:00
$(#[$enum_meta])*
$enum
),+
}
impl ::std::convert::TryFrom<crate::Element> for $elem {
type Error = crate::util::error::Error;
fn try_from(elem: crate::Element) -> Result<$elem, crate::util::error::Error> {
check_ns_only!(elem, $name, $ns);
check_no_children!(elem, $name);
check_no_attributes!(elem, $name);
Ok(match elem.name() {
$($enum_name => $elem::$enum,)+
_ => return Err(crate::util::error::Error::ParseError(concat!("This is not a ", $name, " element."))),
})
}
}
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,)+
},
crate::ns::$ns,
2018-12-29 17:29:11 +00:00
)
.build()
}
}
);
}
macro_rules! generate_attribute_enum {
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, $attr:tt, {$($(#[$enum_meta:meta])* $enum:ident => $enum_name:tt),+$(,)?}) => (
$(#[$meta])*
#[derive(Debug, Clone, PartialEq)]
pub enum $elem {
$(
2018-12-29 17:29:11 +00:00
$(#[$enum_meta])*
$enum
),+
}
impl ::std::convert::TryFrom<crate::Element> for $elem {
type Error = crate::util::error::Error;
fn try_from(elem: crate::Element) -> Result<$elem, crate::util::error::Error> {
check_ns_only!(elem, $name, $ns);
check_no_children!(elem, $name);
check_no_unknown_attributes!(elem, $name, [$attr]);
Ok(match get_attr!(elem, $attr, Required) {
$($enum_name => $elem::$enum,)+
_ => return Err(crate::util::error::Error::ParseError(concat!("Invalid ", $name, " ", $attr, " value."))),
})
}
}
impl From<$elem> for crate::Element {
fn from(elem: $elem) -> crate::Element {
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-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) => {
if !$elem.is($name, crate::ns::$ns) {
return Err(crate::util::error::Error::TypeMismatch(
$name,
crate::ns::$ns,
$elem,
));
}
};
}
macro_rules! check_child {
($elem:ident, $name:tt, $ns:ident) => {
check_child!($elem, $name, $ns, $name);
};
2018-12-18 14:32:05 +00:00
($elem:ident, $name:tt, $ns:ident, $pretty_name:tt) => {
if !$elem.is($name, crate::ns::$ns) {
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) => {
if !$elem.has_ns(crate::ns::$ns) {
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) => {
#[cfg(not(feature = "disable-validation"))]
2017-10-31 21:17:24 +00:00
for _ in $elem.children() {
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) => {
#[cfg(not(feature = "disable-validation"))]
for _ in $elem.attrs() {
return Err(crate::util::error::Error::ParseError(concat!(
2018-12-18 14:32:05 +00:00
"Unknown attribute in ",
$name,
" element."
)));
}
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),*]) => (
#[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
)*
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, PartialEq)]
2017-10-31 21:17:24 +00:00
pub struct $elem;
impl ::std::convert::TryFrom<crate::Element> for $elem {
type Error = crate::util::error::Error;
2017-10-31 21:17:24 +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)
}
}
impl From<$elem> for crate::Element {
fn from(_: $elem) -> crate::Element {
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);
impl ::std::str::FromStr for $elem {
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)))
}
}
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 {
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident) => (
generate_elem_id!($(#[$meta])* $elem, $name, $ns, String);
impl ::std::str::FromStr for $elem {
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)))
}
}
);
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, $type:ty) => (
$(#[$meta])*
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
pub struct $elem(pub $type);
impl ::std::convert::TryFrom<crate::Element> for $elem {
type Error = crate::util::error::Error;
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.
Ok($elem(elem.text().parse()?))
2017-11-24 05:09:25 +00:00
}
}
impl From<$elem> for crate::Element {
fn from(elem: $elem) -> crate::Element {
crate::Element::builder($name, crate::ns::$ns)
.append(elem.0.to_string())
2018-12-29 17:29:11 +00:00
.build()
2017-10-31 21:17:24 +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
);
(Default, $type:ty) => (
$type
);
}
macro_rules! start_decl {
(Vec, $type:ty) => (
Vec<$type>
);
(Option, $type:ty) => (
Option<$type>
2018-05-14 19:04:16 +00:00
);
(Required, $type:ty) => (
$type
);
(Present, $type:ty) => (
bool
);
}
macro_rules! start_parse_elem {
2018-12-18 14:32:05 +00:00
($temp:ident: Vec) => {
let mut $temp = Vec::new();
2018-12-18 14:32:05 +00:00
};
($temp:ident: Option) => {
let mut $temp = None;
2018-12-18 14:32:05 +00:00
};
($temp:ident: Required) => {
let mut $temp = None;
2018-12-18 14:32:05 +00:00
};
($temp:ident: Present) => {
let mut $temp = false;
};
}
2018-05-14 19:04:16 +00:00
macro_rules! do_parse {
2018-12-18 14:32:05 +00:00
($elem:ident, Element) => {
Ok($elem)
2018-12-18 14:32:05 +00:00
};
($elem:ident, String) => {
Ok($elem.text())
2018-12-18 14:32:05 +00:00
};
($elem:ident, $constructor:ident) => {
$constructor::try_from($elem)
2018-12-18 14:32:05 +00:00
};
}
2018-05-14 19:04:16 +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) => {
match do_parse!($elem, $constructor) {
Ok(v) => Ok($temp.push(v)),
Err(e) => Err(e),
}
2018-12-18 14:32:05 +00:00
};
($temp:ident: Option = $constructor:ident => $elem:ident, $name:tt, $parent_name:tt) => {
if $temp.is_some() {
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."
)))
} else {
match do_parse!($elem, $constructor) {
Ok(v) => {
$temp = Some(v);
Ok(())
}
Err(e) => Err(e),
}
2018-05-14 19:04:16 +00:00
}
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() {
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."
)))
} else {
match do_parse!($elem, $constructor) {
Ok(v) => {
$temp = Some(v);
Ok(())
}
Err(e) => Err(e),
}
2018-05-28 15:04:40 +00:00
}
2018-12-18 14:32:05 +00:00
};
($temp:ident: Present = $constructor:ident => $elem:ident, $name:tt, $parent_name:tt) => {
if $temp {
Err(crate::util::error::Error::ParseError(concat!(
"Element ",
$parent_name,
" must not have more than one ",
$name,
" child."
)))
} else {
$temp = true;
Ok(())
}
};
}
2018-05-14 19:04:16 +00:00
macro_rules! finish_parse_elem {
2018-12-18 14:32:05 +00:00
($temp:ident: Vec = $name:tt, $parent_name:tt) => {
$temp
2018-12-18 14:32:05 +00:00
};
($temp:ident: Option = $name:tt, $parent_name:tt) => {
$temp
2018-12-18 14:32:05 +00:00
};
($temp:ident: Required = $name:tt, $parent_name:tt) => {
$temp.ok_or(crate::util::error::Error::ParseError(concat!(
2018-12-18 14:32:05 +00:00
"Missing child ",
$name,
" in ",
$parent_name,
" element."
)))?
};
($temp:ident: Present = $name:tt, $parent_name:tt) => {
$temp
};
}
macro_rules! generate_serialiser {
($builder:ident, $parent:ident, $elem:ident, Required, String, ($name:tt, $ns:ident)) => {
$builder.append(
crate::Element::builder($name, crate::ns::$ns)
.append(::minidom::Node::Text($parent.$elem)),
)
2018-12-18 14:32:05 +00:00
};
($builder:ident, $parent:ident, $elem:ident, Option, String, ($name:tt, $ns:ident)) => {
$builder.append_all($parent.$elem.map(|elem| {
crate::Element::builder($name, crate::ns::$ns).append(::minidom::Node::Text(elem))
}))
};
($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))),
)
};
($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))),
)
};
($builder:ident, $parent:ident, $elem:ident, Vec, $constructor:ident, ($name:tt, $ns:ident)) => {
$builder.append_all($parent.$elem.into_iter())
};
($builder:ident, $parent:ident, $elem:ident, Present, $constructor:ident, ($name:tt, $ns:ident)) => {
$builder.append_all(
$parent
.$elem
.then(|| crate::Element::builder($name, crate::ns::$ns)),
)
};
($builder:ident, $parent:ident, $elem:ident, $_:ident, $constructor:ident, ($name:tt, $ns:ident)) => {
$builder.append(::minidom::Node::Element(crate::Element::from(
$parent.$elem,
)))
2018-12-18 14:32:05 +00:00
};
}
macro_rules! generate_child_test {
($child:ident, $name:tt, *) => {
$child.is($name, ::minidom::NSChoice::Any)
};
($child:ident, $name:tt, $ns:tt) => {
$child.is($name, crate::ns::$ns)
};
}
macro_rules! generate_element {
($(#[$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: []);
);
($(#[$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),+$(,)?]) => (
generate_element!($(#[$meta])* $elem, $name, $ns, attributes: [], children: [$($(#[$child_meta])* $child_ident: $coucou<$child_type> = ($child_name, $child_ns) => $child_constructor),*]);
);
($(#[$meta:meta])* $elem:ident, $name:tt, $ns:ident, text: ($(#[$text_meta:meta])* $text_ident:ident: $codec:ty )) => (
generate_element!($(#[$meta])* $elem, $name, $ns, attributes: [], children: [], text: ($(#[$text_meta])* $text_ident: $codec));
);
($(#[$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:ty )) => (
generate_element!($(#[$meta])* $elem, $name, $ns, attributes: [$($(#[$attr_meta])* $attr: $attr_action<$attr_type> = $attr_name),*], children: [], text: ($(#[$text_meta])* $text_ident: $codec));
);
($(#[$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:ty ))*) => (
2018-05-14 19:04:16 +00:00
$(#[$meta])*
#[derive(Debug, Clone, PartialEq)]
2018-05-14 19:04:16 +00:00
pub struct $elem {
$(
2018-12-29 17:29:11 +00:00
$(#[$attr_meta])*
pub $attr: decl_attr!($attr_action, $attr_type),
2018-05-14 19:04:16 +00:00
)*
$(
2018-12-29 17:29:11 +00:00
$(#[$child_meta])*
pub $child_ident: start_decl!($coucou, $child_type),
)*
$(
2018-12-29 17:29:11 +00:00
$(#[$text_meta])*
pub $text_ident: <$codec as Codec>::Decoded,
)*
2018-05-14 19:04:16 +00:00
}
impl ::xso::FromXml for $elem {
type Builder = ::xso::minidom_compat::FromEventsViaElement<$elem>;
fn from_events(
qname: ::xso::exports::rxml::QName,
attrs: ::xso::exports::rxml::AttrMap,
) -> Result<Self::Builder, ::xso::error::FromEventsError> {
if qname.0 != crate::ns::$ns || qname.1 != $name {
return Err(::xso::error::FromEventsError::Mismatch {
name: qname,
attrs,
})
}
Self::Builder::new(qname, attrs)
}
}
impl ::std::convert::TryFrom<crate::Element> for $elem {
type Error = crate::util::error::Error;
2018-05-14 19:04:16 +00:00
fn try_from(mut 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-12-29 17:29:11 +00:00
start_parse_elem!($child_ident: $coucou);
)*
// We must pull the texts out of the Element before we pull
// the children, because take_elements_as_children drains
// *all* child nodes, including texts.
// To deal with the genericity of this macro, we need a local
// struct decl to carry the identifier around.
struct Texts {
$(
$text_ident: <$codec as Codec>::Decoded,
)*
}
#[allow(unused_variables)]
let texts = Texts {
$(
$text_ident: <$codec>::decode(&elem.text())?,
)*
};
for _child in elem.take_contents_as_children() {
let residual = _child;
$(
// TODO: get rid of the explicit generate_child_test here
// once !295 has landed.
let residual = if generate_child_test!(residual, $child_name, $child_ns) {
match do_parse_elem!($child_ident: $coucou = $child_constructor => residual, $child_name, $name) {
Ok(()) => continue,
Err(other) => return Err(other),
}
} else {
residual
};
)*
let _ = residual;
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-12-29 17:29:11 +00:00
$child_ident: finish_parse_elem!($child_ident: $coucou = $child_name, $name),
)*
$(
$text_ident: texts.$text_ident,
)*
2018-05-14 19:04:16 +00:00
})
}
}
impl From<$elem> for crate::Element {
fn from(elem: $elem) -> crate::Element {
let mut builder = crate::Element::builder($name, crate::ns::$ns);
$(
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()
}
}
impl ::xso::IntoXml for $elem {
type EventIter = ::xso::minidom_compat::IntoEventsViaElement;
fn into_event_iter(self) -> Result<Self::EventIter, ::xso::error::Error> {
Self::EventIter::new(self)
}
}
2018-05-14 19:04:16 +00:00
);
}
#[cfg(test)]
macro_rules! assert_size (
($t:ty, $sz:expr) => (
assert_eq!(::std::mem::size_of::<$t>(), $sz);
);
);
// 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) => {
impl ::std::convert::TryFrom<crate::Element> for $item {
type Error = Error;
fn try_from(mut elem: crate::Element) -> Result<$item, Error> {
check_self!(elem, "item", $ns);
check_no_unknown_attributes!(elem, "item", ["id", "publisher"]);
let mut payloads = elem.take_contents_as_children().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 {
id: get_attr!(elem, "id", Option),
publisher: get_attr!(elem, "publisher", Option),
payload,
}))
}
}
impl From<$item> for crate::Element {
fn from(item: $item) -> crate::Element {
crate::Element::builder("item", ns::$ns)
.attr("id", item.0.id)
.attr("publisher", item.0.publisher)
.append_all(item.0.payload)
.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
}
}
};
}