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-07-20 19:03:15 +00:00
|
|
|
use try_from::TryFrom;
|
2017-05-06 19:42:12 +00:00
|
|
|
|
|
|
|
use minidom::Element;
|
2017-06-08 20:46:27 +00:00
|
|
|
use chrono::{DateTime, FixedOffset};
|
2017-04-21 02:57:34 +00:00
|
|
|
|
|
|
|
use error::Error;
|
2017-05-01 00:50:18 +00:00
|
|
|
use jid::Jid;
|
2017-04-21 02:57:34 +00:00
|
|
|
|
|
|
|
use ns;
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct Delay {
|
2017-05-01 00:50:18 +00:00
|
|
|
pub from: Option<Jid>,
|
2017-05-27 11:29:21 +00:00
|
|
|
pub stamp: DateTime<FixedOffset>,
|
2017-04-21 02:57:34 +00:00
|
|
|
pub data: Option<String>,
|
|
|
|
}
|
|
|
|
|
2017-05-23 22:31:33 +00:00
|
|
|
impl TryFrom<Element> for Delay {
|
2017-07-20 19:03:15 +00:00
|
|
|
type Err = Error;
|
2017-04-21 02:57:34 +00:00
|
|
|
|
2017-05-23 22:31:33 +00:00
|
|
|
fn try_from(elem: Element) -> Result<Delay, Error> {
|
2017-05-06 19:42:12 +00:00
|
|
|
if !elem.is("delay", ns::DELAY) {
|
|
|
|
return Err(Error::ParseError("This is not a delay element."));
|
|
|
|
}
|
|
|
|
for _ in elem.children() {
|
|
|
|
return Err(Error::ParseError("Unknown child in delay element."));
|
|
|
|
}
|
2017-05-23 00:02:23 +00:00
|
|
|
let from = get_attr!(elem, "from", optional);
|
2017-05-27 11:29:21 +00:00
|
|
|
let stamp = get_attr!(elem, "stamp", required, stamp, DateTime::parse_from_rfc3339(stamp)?);
|
2017-05-06 19:42:12 +00:00
|
|
|
let data = match elem.text().as_ref() {
|
|
|
|
"" => None,
|
|
|
|
text => Some(text.to_owned()),
|
|
|
|
};
|
|
|
|
Ok(Delay {
|
|
|
|
from: from,
|
|
|
|
stamp: stamp,
|
|
|
|
data: data,
|
|
|
|
})
|
|
|
|
}
|
2017-04-23 02:21:53 +00:00
|
|
|
}
|
|
|
|
|
2017-07-20 19:36:13 +00:00
|
|
|
impl From<Delay> for Element {
|
|
|
|
fn from(delay: Delay) -> Element {
|
2017-05-06 19:42:12 +00:00
|
|
|
Element::builder("delay")
|
|
|
|
.ns(ns::DELAY)
|
2017-07-21 00:19:34 +00:00
|
|
|
.attr("from", delay.from.map(String::from))
|
2017-07-20 19:36:13 +00:00
|
|
|
.attr("stamp", delay.stamp.to_rfc3339())
|
|
|
|
.append(delay.data)
|
2017-05-06 19:42:12 +00:00
|
|
|
.build()
|
2017-04-29 02:50:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-21 02:57:34 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2017-05-06 19:42:12 +00:00
|
|
|
use super::*;
|
2017-06-08 20:46:27 +00:00
|
|
|
use std::str::FromStr;
|
|
|
|
use chrono::{Datelike, Timelike};
|
2017-04-21 02:57:34 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_simple() {
|
|
|
|
let elem: Element = "<delay xmlns='urn:xmpp:delay' from='capulet.com' stamp='2002-09-10T23:08:25Z'/>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
let delay = Delay::try_from(elem).unwrap();
|
2017-05-01 00:50:18 +00:00
|
|
|
assert_eq!(delay.from, Some(Jid::from_str("capulet.com").unwrap()));
|
2017-05-27 11:29:21 +00:00
|
|
|
assert_eq!(delay.stamp.year(), 2002);
|
|
|
|
assert_eq!(delay.stamp.month(), 9);
|
|
|
|
assert_eq!(delay.stamp.day(), 10);
|
|
|
|
assert_eq!(delay.stamp.hour(), 23);
|
|
|
|
assert_eq!(delay.stamp.minute(), 08);
|
|
|
|
assert_eq!(delay.stamp.second(), 25);
|
|
|
|
assert_eq!(delay.stamp.nanosecond(), 0);
|
|
|
|
assert_eq!(delay.stamp.timezone(), FixedOffset::east(0));
|
2017-04-21 02:57:34 +00:00
|
|
|
assert_eq!(delay.data, None);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unknown() {
|
|
|
|
let elem: Element = "<replace xmlns='urn:xmpp:message-correct:0'/>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
let error = Delay::try_from(elem).unwrap_err();
|
2017-04-21 02:57:34 +00:00
|
|
|
let message = match error {
|
|
|
|
Error::ParseError(string) => string,
|
|
|
|
_ => panic!(),
|
|
|
|
};
|
|
|
|
assert_eq!(message, "This is not a delay element.");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_invalid_child() {
|
|
|
|
let elem: Element = "<delay xmlns='urn:xmpp:delay'><coucou/></delay>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
let error = Delay::try_from(elem).unwrap_err();
|
2017-04-21 02:57:34 +00:00
|
|
|
let message = match error {
|
|
|
|
Error::ParseError(string) => string,
|
|
|
|
_ => panic!(),
|
|
|
|
};
|
|
|
|
assert_eq!(message, "Unknown child in delay element.");
|
|
|
|
}
|
2017-04-23 02:21:53 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_serialise() {
|
2017-05-27 11:29:21 +00:00
|
|
|
let elem: Element = "<delay xmlns='urn:xmpp:delay' stamp='2002-09-10T23:08:25+00:00'/>".parse().unwrap();
|
2017-05-06 19:42:12 +00:00
|
|
|
let delay = Delay {
|
2017-04-23 02:21:53 +00:00
|
|
|
from: None,
|
2017-05-27 11:29:21 +00:00
|
|
|
stamp: DateTime::parse_from_rfc3339("2002-09-10T23:08:25Z").unwrap(),
|
2017-04-23 02:21:53 +00:00
|
|
|
data: None,
|
|
|
|
};
|
2017-05-23 22:31:33 +00:00
|
|
|
let elem2 = delay.into();
|
2017-04-23 02:21:53 +00:00
|
|
|
assert_eq!(elem, elem2);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_serialise_data() {
|
2017-05-27 11:29:21 +00:00
|
|
|
let elem: Element = "<delay xmlns='urn:xmpp:delay' from='juliet@example.org' stamp='2002-09-10T23:08:25+00:00'>Reason</delay>".parse().unwrap();
|
2017-05-06 19:42:12 +00:00
|
|
|
let delay = Delay {
|
2017-05-01 00:50:18 +00:00
|
|
|
from: Some(Jid::from_str("juliet@example.org").unwrap()),
|
2017-05-27 11:29:21 +00:00
|
|
|
stamp: DateTime::parse_from_rfc3339("2002-09-10T23:08:25Z").unwrap(),
|
2017-04-23 02:21:53 +00:00
|
|
|
data: Some(String::from("Reason")),
|
|
|
|
};
|
2017-05-23 22:31:33 +00:00
|
|
|
let elem2 = delay.into();
|
2017-04-23 02:21:53 +00:00
|
|
|
assert_eq!(elem, elem2);
|
|
|
|
}
|
2017-04-21 02:57:34 +00:00
|
|
|
}
|