rsm, ibb: Write and use a macro to parse attributes.

This commit is contained in:
Emmanuel Gil Peyrot 2017-05-21 16:03:17 +01:00
parent 16899f8c23
commit 3c083709cb
3 changed files with 34 additions and 25 deletions

View file

@ -65,12 +65,6 @@ pub enum IBB {
}, },
} }
fn required_attr<T: FromStr>(elem: &Element, attr: &str, err: Error) -> Result<T, Error> {
elem.attr(attr)
.and_then(|value| value.parse().ok())
.ok_or(err)
}
impl<'a> TryFrom<&'a Element> for IBB { impl<'a> TryFrom<&'a Element> for IBB {
type Error = Error; type Error = Error;
@ -79,12 +73,9 @@ impl<'a> TryFrom<&'a Element> for IBB {
for _ in elem.children() { for _ in elem.children() {
return Err(Error::ParseError("Unknown child in open element.")); return Err(Error::ParseError("Unknown child in open element."));
} }
let block_size = required_attr(elem, "block-size", Error::ParseError("Required attribute 'block-size' missing in open element."))?; let block_size = get_attr!(elem, "block-size", required, block_size, block_size.parse()?);
let sid = required_attr(elem, "sid", Error::ParseError("Required attribute 'sid' missing in open element."))?; let sid = get_attr!(elem, "sid", required, sid, sid.parse()?);
let stanza = match elem.attr("stanza") { let stanza = get_attr!(elem, "stanza", default, stanza, stanza.parse()?);
Some(stanza) => stanza.parse()?,
None => Default::default(),
};
Ok(IBB::Open { Ok(IBB::Open {
block_size: block_size, block_size: block_size,
sid: sid, sid: sid,
@ -94,8 +85,8 @@ impl<'a> TryFrom<&'a Element> for IBB {
for _ in elem.children() { for _ in elem.children() {
return Err(Error::ParseError("Unknown child in data element.")); return Err(Error::ParseError("Unknown child in data element."));
} }
let seq = required_attr(elem, "seq", Error::ParseError("Required attribute 'seq' missing in data element."))?; let seq = get_attr!(elem, "seq", required, seq, seq.parse()?);
let sid = required_attr(elem, "sid", Error::ParseError("Required attribute 'sid' missing in data element."))?; let sid = get_attr!(elem, "sid", required, sid, sid.parse()?);
let data = base64::decode(&elem.text())?; let data = base64::decode(&elem.text())?;
Ok(IBB::Data { Ok(IBB::Data {
seq: seq, seq: seq,
@ -103,10 +94,10 @@ impl<'a> TryFrom<&'a Element> for IBB {
data: data data: data
}) })
} else if elem.is("close", ns::IBB) { } else if elem.is("close", ns::IBB) {
let sid = required_attr(elem, "sid", Error::ParseError("Required attribute 'sid' missing in data element."))?;
for _ in elem.children() { for _ in elem.children() {
return Err(Error::ParseError("Unknown child in close element.")); return Err(Error::ParseError("Unknown child in close element."));
} }
let sid = get_attr!(elem, "sid", required, sid, sid.parse()?);
Ok(IBB::Close { Ok(IBB::Close {
sid: sid, sid: sid,
}) })
@ -148,6 +139,7 @@ impl<'a> Into<Element> for &'a IBB {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
use std::error::Error as StdError;
#[test] #[test]
fn test_simple() { fn test_simple() {
@ -191,24 +183,23 @@ mod tests {
Error::ParseError(string) => string, Error::ParseError(string) => string,
_ => panic!(), _ => panic!(),
}; };
assert_eq!(message, "Required attribute 'block-size' missing in open element."); assert_eq!(message, "Required attribute 'block-size' missing.");
// TODO: maybe make a better error message here.
let elem: Element = "<open xmlns='http://jabber.org/protocol/ibb' block-size='-5'/>".parse().unwrap(); let elem: Element = "<open xmlns='http://jabber.org/protocol/ibb' block-size='-5'/>".parse().unwrap();
let error = IBB::try_from(&elem).unwrap_err(); let error = IBB::try_from(&elem).unwrap_err();
let message = match error { let message = match error {
Error::ParseError(string) => string, Error::ParseIntError(error) => error,
_ => panic!(), _ => panic!(),
}; };
assert_eq!(message, "Required attribute 'block-size' missing in open element."); assert_eq!(message.description(), "invalid digit found in string");
let elem: Element = "<open xmlns='http://jabber.org/protocol/ibb' block-size='128'/>".parse().unwrap(); let elem: Element = "<open xmlns='http://jabber.org/protocol/ibb' block-size='128'/>".parse().unwrap();
let error = IBB::try_from(&elem).unwrap_err(); let error = IBB::try_from(&elem).unwrap_err();
let message = match error { let message = match error {
Error::ParseError(string) => string, Error::ParseError(error) => error,
_ => panic!(), _ => panic!(),
}; };
assert_eq!(message, "Required attribute 'sid' missing in open element."); assert_eq!(message, "Required attribute 'sid' missing.");
} }
#[test] #[test]

View file

@ -22,6 +22,27 @@ extern crate sha2;
extern crate sha3; extern crate sha3;
extern crate blake2; extern crate blake2;
macro_rules! get_attr {
($elem:ident, $attr:tt, optional, $value:ident, $func:expr) => (
match $elem.attr($attr) {
Some($value) => Some($func),
None => None,
}
);
($elem:ident, $attr:tt, required, $value:ident, $func:expr) => (
match $elem.attr($attr) {
Some($value) => $func,
None => return Err(Error::ParseError(concat!("Required attribute '", $attr, "' missing."))),
}
);
($elem:ident, $attr:tt, default, $value:ident, $func:expr) => (
match $elem.attr($attr) {
Some($value) => $func,
None => Default::default(),
}
);
}
/// Error type returned by every parser on failure. /// Error type returned by every parser on failure.
pub mod error; pub mod error;
/// XML namespace definitions used through XMPP. /// XML namespace definitions used through XMPP.

View file

@ -61,10 +61,7 @@ impl<'a> TryFrom<&'a Element> for Set {
if set.first.is_some() { if set.first.is_some() {
return Err(Error::ParseError("Set cant have more than one first.")); return Err(Error::ParseError("Set cant have more than one first."));
} }
set.first_index = match child.attr("index") { set.first_index = get_attr!(child, "index", optional, index, index.parse()?);
Some(index) => Some(index.parse()?),
None => None,
};
set.first = Some(child.text()); set.first = Some(child.text());
} else if child.is("index", ns::RSM) { } else if child.is("index", ns::RSM) {
if set.index.is_some() { if set.index.is_some() {