2017-04-29 21:14:34 +00:00
|
|
|
|
// Copyright (c) 2017 Emmanuel Gil Peyrot <linkmauve@linkmauve.fr>
|
|
|
|
|
//
|
|
|
|
|
// 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/.
|
|
|
|
|
|
2017-05-06 19:14:45 +00:00
|
|
|
|
use std::convert::TryFrom;
|
|
|
|
|
|
2017-04-29 03:37:18 +00:00
|
|
|
|
use minidom::Element;
|
|
|
|
|
|
|
|
|
|
use error::Error;
|
|
|
|
|
|
|
|
|
|
use ns;
|
|
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
|
pub struct First {
|
|
|
|
|
pub index: Option<usize>,
|
2017-05-19 01:58:18 +00:00
|
|
|
|
pub id: String,
|
2017-04-29 03:37:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
|
pub struct Set {
|
|
|
|
|
pub after: Option<String>,
|
|
|
|
|
pub before: Option<String>,
|
|
|
|
|
pub count: Option<usize>,
|
|
|
|
|
pub first: Option<First>,
|
|
|
|
|
pub index: Option<usize>,
|
|
|
|
|
pub last: Option<String>,
|
|
|
|
|
pub max: Option<usize>,
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-06 19:14:45 +00:00
|
|
|
|
impl<'a> TryFrom<&'a Element> for Set {
|
|
|
|
|
type Error = Error;
|
|
|
|
|
|
|
|
|
|
fn try_from(elem: &'a Element) -> Result<Set, Error> {
|
|
|
|
|
if !elem.is("set", ns::RSM) {
|
|
|
|
|
return Err(Error::ParseError("This is not a RSM element."));
|
|
|
|
|
}
|
|
|
|
|
let mut after = None;
|
|
|
|
|
let mut before = None;
|
|
|
|
|
let mut count = None;
|
|
|
|
|
let mut first = None;
|
|
|
|
|
let mut index = None;
|
|
|
|
|
let mut last = None;
|
|
|
|
|
let mut max = None;
|
|
|
|
|
for child in elem.children() {
|
|
|
|
|
if child.is("after", ns::RSM) {
|
|
|
|
|
if after.is_some() {
|
|
|
|
|
return Err(Error::ParseError("Set can’t have more than one after."));
|
|
|
|
|
}
|
|
|
|
|
after = Some(child.text());
|
|
|
|
|
} else if child.is("before", ns::RSM) {
|
|
|
|
|
if before.is_some() {
|
|
|
|
|
return Err(Error::ParseError("Set can’t have more than one before."));
|
|
|
|
|
}
|
|
|
|
|
before = Some(child.text());
|
|
|
|
|
} else if child.is("count", ns::RSM) {
|
|
|
|
|
if count.is_some() {
|
|
|
|
|
return Err(Error::ParseError("Set can’t have more than one count."));
|
|
|
|
|
}
|
|
|
|
|
count = Some(child.text().parse()?);
|
|
|
|
|
} else if child.is("first", ns::RSM) {
|
|
|
|
|
if first.is_some() {
|
|
|
|
|
return Err(Error::ParseError("Set can’t have more than one first."));
|
|
|
|
|
}
|
|
|
|
|
first = Some(First {
|
|
|
|
|
index: match child.attr("index") {
|
|
|
|
|
Some(index) => Some(index.parse()?),
|
|
|
|
|
None => None,
|
|
|
|
|
},
|
2017-05-19 01:58:18 +00:00
|
|
|
|
id: child.text(),
|
2017-05-06 19:14:45 +00:00
|
|
|
|
});
|
|
|
|
|
} else if child.is("index", ns::RSM) {
|
|
|
|
|
if index.is_some() {
|
|
|
|
|
return Err(Error::ParseError("Set can’t have more than one index."));
|
|
|
|
|
}
|
|
|
|
|
index = Some(child.text().parse()?);
|
|
|
|
|
} else if child.is("last", ns::RSM) {
|
|
|
|
|
if last.is_some() {
|
|
|
|
|
return Err(Error::ParseError("Set can’t have more than one last."));
|
|
|
|
|
}
|
|
|
|
|
last = Some(child.text());
|
|
|
|
|
} else if child.is("max", ns::RSM) {
|
|
|
|
|
if max.is_some() {
|
|
|
|
|
return Err(Error::ParseError("Set can’t have more than one max."));
|
|
|
|
|
}
|
|
|
|
|
max = Some(child.text().parse()?);
|
|
|
|
|
} else {
|
|
|
|
|
return Err(Error::ParseError("Unknown child in set element."));
|
2017-04-29 03:37:18 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2017-05-06 19:14:45 +00:00
|
|
|
|
Ok(Set {
|
|
|
|
|
after: after,
|
|
|
|
|
before: before,
|
|
|
|
|
count: count,
|
|
|
|
|
first: first,
|
|
|
|
|
index: index,
|
|
|
|
|
last: last,
|
|
|
|
|
max: max,
|
|
|
|
|
})
|
2017-04-29 03:37:18 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-06 19:14:45 +00:00
|
|
|
|
impl<'a> Into<Element> for &'a Set {
|
|
|
|
|
fn into(self) -> Element {
|
|
|
|
|
let mut elem = Element::builder("set")
|
|
|
|
|
.ns(ns::RSM)
|
|
|
|
|
.build();
|
|
|
|
|
if self.after.is_some() {
|
|
|
|
|
elem.append_child(Element::builder("after").ns(ns::RSM).append(self.after.clone()).build());
|
|
|
|
|
}
|
|
|
|
|
if self.before.is_some() {
|
|
|
|
|
elem.append_child(Element::builder("before").ns(ns::RSM).append(self.before.clone()).build());
|
|
|
|
|
}
|
|
|
|
|
if self.count.is_some() {
|
|
|
|
|
elem.append_child(Element::builder("count").ns(ns::RSM).append(format!("{}", self.count.unwrap())).build());
|
|
|
|
|
}
|
|
|
|
|
if self.first.is_some() {
|
|
|
|
|
let first = self.first.clone().unwrap();
|
|
|
|
|
elem.append_child(Element::builder("first")
|
|
|
|
|
.ns(ns::RSM)
|
|
|
|
|
.attr("index", match first.index {
|
|
|
|
|
Some(index) => Some(format!("{}", index)),
|
|
|
|
|
None => None,
|
|
|
|
|
})
|
2017-05-19 01:58:18 +00:00
|
|
|
|
.append(first.id.clone()).build());
|
2017-05-06 19:14:45 +00:00
|
|
|
|
}
|
|
|
|
|
if self.index.is_some() {
|
|
|
|
|
elem.append_child(Element::builder("index").ns(ns::RSM).append(format!("{}", self.index.unwrap())).build());
|
|
|
|
|
}
|
|
|
|
|
if self.last.is_some() {
|
|
|
|
|
elem.append_child(Element::builder("last").ns(ns::RSM).append(self.last.clone()).build());
|
|
|
|
|
}
|
|
|
|
|
if self.max.is_some() {
|
|
|
|
|
elem.append_child(Element::builder("max").ns(ns::RSM).append(format!("{}", self.max.unwrap())).build());
|
|
|
|
|
}
|
|
|
|
|
elem
|
2017-04-29 03:37:18 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
2017-05-06 19:14:45 +00:00
|
|
|
|
use super::*;
|
2017-04-29 03:37:18 +00:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_simple() {
|
|
|
|
|
let elem: Element = "<set xmlns='http://jabber.org/protocol/rsm'/>".parse().unwrap();
|
2017-05-06 19:14:45 +00:00
|
|
|
|
let set = Set::try_from(&elem).unwrap();
|
2017-04-29 03:37:18 +00:00
|
|
|
|
assert_eq!(set.after, None);
|
|
|
|
|
assert_eq!(set.before, None);
|
|
|
|
|
assert_eq!(set.count, None);
|
|
|
|
|
match set.first {
|
|
|
|
|
Some(_) => panic!(),
|
|
|
|
|
None => (),
|
|
|
|
|
}
|
|
|
|
|
assert_eq!(set.index, None);
|
|
|
|
|
assert_eq!(set.last, None);
|
|
|
|
|
assert_eq!(set.max, None);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_unknown() {
|
|
|
|
|
let elem: Element = "<replace xmlns='urn:xmpp:message-correct:0'/>".parse().unwrap();
|
2017-05-06 19:14:45 +00:00
|
|
|
|
let error = Set::try_from(&elem).unwrap_err();
|
2017-04-29 03:37:18 +00:00
|
|
|
|
let message = match error {
|
|
|
|
|
Error::ParseError(string) => string,
|
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
|
|
|
|
assert_eq!(message, "This is not a RSM element.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_invalid_child() {
|
|
|
|
|
let elem: Element = "<set xmlns='http://jabber.org/protocol/rsm'><coucou/></set>".parse().unwrap();
|
2017-05-06 19:14:45 +00:00
|
|
|
|
let error = Set::try_from(&elem).unwrap_err();
|
2017-04-29 03:37:18 +00:00
|
|
|
|
let message = match error {
|
|
|
|
|
Error::ParseError(string) => string,
|
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
|
|
|
|
assert_eq!(message, "Unknown child in set element.");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_serialise() {
|
|
|
|
|
let elem: Element = "<set xmlns='http://jabber.org/protocol/rsm'/>".parse().unwrap();
|
2017-05-06 19:14:45 +00:00
|
|
|
|
let rsm = Set {
|
2017-04-29 03:37:18 +00:00
|
|
|
|
after: None,
|
|
|
|
|
before: None,
|
|
|
|
|
count: None,
|
|
|
|
|
first: None,
|
|
|
|
|
index: None,
|
|
|
|
|
last: None,
|
|
|
|
|
max: None,
|
|
|
|
|
};
|
2017-05-06 19:14:45 +00:00
|
|
|
|
let elem2 = (&rsm).into();
|
2017-04-29 03:37:18 +00:00
|
|
|
|
assert_eq!(elem, elem2);
|
|
|
|
|
}
|
|
|
|
|
}
|