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-04-19 20:52:14 +00:00
|
|
|
use std::str::FromStr;
|
|
|
|
|
2017-04-24 18:25:00 +00:00
|
|
|
use minidom::{Element, IntoAttributeValue};
|
2017-04-21 02:07:21 +00:00
|
|
|
use base64;
|
2017-04-19 20:52:14 +00:00
|
|
|
|
|
|
|
use error::Error;
|
|
|
|
|
2017-04-20 22:16:12 +00:00
|
|
|
use ns;
|
2017-04-19 20:52:14 +00:00
|
|
|
|
2017-06-13 23:50:57 +00:00
|
|
|
generate_attribute!(Stanza, "stanza", {
|
|
|
|
Iq => "iq",
|
|
|
|
Message => "message",
|
|
|
|
}, Default = Iq);
|
2017-04-19 20:52:14 +00:00
|
|
|
|
2017-04-20 23:41:15 +00:00
|
|
|
#[derive(Debug, Clone)]
|
2017-07-29 02:44:35 +00:00
|
|
|
pub struct Open {
|
|
|
|
pub block_size: u16,
|
|
|
|
pub sid: String,
|
|
|
|
pub stanza: Stanza,
|
2017-04-19 20:52:14 +00:00
|
|
|
}
|
|
|
|
|
2017-07-29 02:44:35 +00:00
|
|
|
impl TryFrom<Element> for Open {
|
2017-07-20 19:03:15 +00:00
|
|
|
type Err = Error;
|
2017-05-04 22:11:10 +00:00
|
|
|
|
2017-07-29 02:44:35 +00:00
|
|
|
fn try_from(elem: Element) -> Result<Open, Error> {
|
|
|
|
if !elem.is("open", ns::IBB) {
|
|
|
|
return Err(Error::ParseError("This is not an open element."));
|
2017-04-19 20:52:14 +00:00
|
|
|
}
|
2017-07-29 02:44:35 +00:00
|
|
|
for _ in elem.children() {
|
|
|
|
return Err(Error::ParseError("Unknown child in open element."));
|
|
|
|
}
|
|
|
|
Ok(Open {
|
|
|
|
block_size: get_attr!(elem, "block-size", required),
|
|
|
|
sid: get_attr!(elem, "sid", required),
|
|
|
|
stanza: get_attr!(elem, "stanza", default),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Open> for Element {
|
|
|
|
fn from(open: Open) -> Element {
|
|
|
|
Element::builder("open")
|
|
|
|
.ns(ns::IBB)
|
|
|
|
.attr("block-size", open.block_size)
|
|
|
|
.attr("sid", open.sid)
|
|
|
|
.attr("stanza", open.stanza)
|
|
|
|
.build()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct Data {
|
|
|
|
pub seq: u16,
|
|
|
|
pub sid: String,
|
|
|
|
pub data: Vec<u8>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Element> for Data {
|
|
|
|
type Err = Error;
|
|
|
|
|
|
|
|
fn try_from(elem: Element) -> Result<Data, Error> {
|
|
|
|
if !elem.is("data", ns::IBB) {
|
|
|
|
return Err(Error::ParseError("This is not a data element."));
|
|
|
|
}
|
|
|
|
for _ in elem.children() {
|
|
|
|
return Err(Error::ParseError("Unknown child in data element."));
|
|
|
|
}
|
|
|
|
Ok(Data {
|
|
|
|
seq: get_attr!(elem, "seq", required),
|
|
|
|
sid: get_attr!(elem, "sid", required),
|
|
|
|
data: base64::decode(&elem.text())?,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Data> for Element {
|
|
|
|
fn from(data: Data) -> Element {
|
|
|
|
Element::builder("data")
|
|
|
|
.ns(ns::IBB)
|
|
|
|
.attr("seq", data.seq)
|
|
|
|
.attr("sid", data.sid)
|
|
|
|
.append(base64::encode(&data.data))
|
|
|
|
.build()
|
2017-04-19 20:52:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-29 02:44:35 +00:00
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct Close {
|
|
|
|
pub sid: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Element> for Close {
|
|
|
|
type Err = Error;
|
|
|
|
|
|
|
|
fn try_from(elem: Element) -> Result<Close, Error> {
|
|
|
|
if !elem.is("close", ns::IBB) {
|
|
|
|
return Err(Error::ParseError("This is not a close element."));
|
2017-05-04 22:11:10 +00:00
|
|
|
}
|
2017-07-29 02:44:35 +00:00
|
|
|
for _ in elem.children() {
|
|
|
|
return Err(Error::ParseError("Unknown child in close element."));
|
|
|
|
}
|
|
|
|
Ok(Close {
|
|
|
|
sid: get_attr!(elem, "sid", required),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Close> for Element {
|
|
|
|
fn from(close: Close) -> Element {
|
|
|
|
Element::builder("close")
|
|
|
|
.ns(ns::IBB)
|
|
|
|
.attr("sid", close.sid)
|
|
|
|
.build()
|
2017-04-27 18:05:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-19 20:52:14 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2017-05-04 22:11:10 +00:00
|
|
|
use super::*;
|
2017-05-21 15:03:17 +00:00
|
|
|
use std::error::Error as StdError;
|
2017-04-19 20:52:14 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_simple() {
|
2017-04-21 02:07:21 +00:00
|
|
|
let elem: Element = "<open xmlns='http://jabber.org/protocol/ibb' block-size='3' sid='coucou'/>".parse().unwrap();
|
2017-07-29 02:44:35 +00:00
|
|
|
let open = Open::try_from(elem).unwrap();
|
|
|
|
assert_eq!(open.block_size, 3);
|
|
|
|
assert_eq!(open.sid, "coucou");
|
|
|
|
assert_eq!(open.stanza, Stanza::Iq);
|
2017-04-21 02:07:21 +00:00
|
|
|
|
|
|
|
let elem: Element = "<data xmlns='http://jabber.org/protocol/ibb' seq='0' sid='coucou'>AAAA</data>".parse().unwrap();
|
2017-07-29 02:44:35 +00:00
|
|
|
let data = Data::try_from(elem).unwrap();
|
|
|
|
assert_eq!(data.seq, 0);
|
|
|
|
assert_eq!(data.sid, "coucou");
|
|
|
|
assert_eq!(data.data, vec!(0, 0, 0));
|
2017-04-21 02:09:10 +00:00
|
|
|
|
|
|
|
let elem: Element = "<close xmlns='http://jabber.org/protocol/ibb' sid='coucou'/>".parse().unwrap();
|
2017-07-29 02:44:35 +00:00
|
|
|
let close = Close::try_from(elem).unwrap();
|
|
|
|
assert_eq!(close.sid, "coucou");
|
2017-04-19 20:52:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_invalid() {
|
|
|
|
let elem: Element = "<open xmlns='http://jabber.org/protocol/ibb'/>".parse().unwrap();
|
2017-07-29 02:44:35 +00:00
|
|
|
let error = Open::try_from(elem).unwrap_err();
|
2017-04-19 20:52:14 +00:00
|
|
|
let message = match error {
|
|
|
|
Error::ParseError(string) => string,
|
|
|
|
_ => panic!(),
|
|
|
|
};
|
2017-05-21 15:03:17 +00:00
|
|
|
assert_eq!(message, "Required attribute 'block-size' missing.");
|
2017-04-19 20:52:14 +00:00
|
|
|
|
|
|
|
let elem: Element = "<open xmlns='http://jabber.org/protocol/ibb' block-size='-5'/>".parse().unwrap();
|
2017-07-29 02:44:35 +00:00
|
|
|
let error = Open::try_from(elem).unwrap_err();
|
2017-04-19 20:52:14 +00:00
|
|
|
let message = match error {
|
2017-05-21 15:03:17 +00:00
|
|
|
Error::ParseIntError(error) => error,
|
2017-04-19 20:52:14 +00:00
|
|
|
_ => panic!(),
|
|
|
|
};
|
2017-05-21 15:03:17 +00:00
|
|
|
assert_eq!(message.description(), "invalid digit found in string");
|
2017-04-19 20:52:14 +00:00
|
|
|
|
|
|
|
let elem: Element = "<open xmlns='http://jabber.org/protocol/ibb' block-size='128'/>".parse().unwrap();
|
2017-07-29 02:44:35 +00:00
|
|
|
let error = Open::try_from(elem).unwrap_err();
|
2017-04-19 20:52:14 +00:00
|
|
|
let message = match error {
|
2017-05-21 15:03:17 +00:00
|
|
|
Error::ParseError(error) => error,
|
2017-04-19 20:52:14 +00:00
|
|
|
_ => panic!(),
|
|
|
|
};
|
2017-05-21 15:03:17 +00:00
|
|
|
assert_eq!(message, "Required attribute 'sid' missing.");
|
2017-04-19 20:52:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_invalid_stanza() {
|
|
|
|
let elem: Element = "<open xmlns='http://jabber.org/protocol/ibb' block-size='128' sid='coucou' stanza='fdsq'/>".parse().unwrap();
|
2017-07-29 02:44:35 +00:00
|
|
|
let error = Open::try_from(elem).unwrap_err();
|
2017-04-19 20:52:14 +00:00
|
|
|
let message = match error {
|
|
|
|
Error::ParseError(string) => string,
|
|
|
|
_ => panic!(),
|
|
|
|
};
|
2017-06-14 00:57:02 +00:00
|
|
|
assert_eq!(message, "Unknown value for 'stanza' attribute.");
|
2017-04-19 20:52:14 +00:00
|
|
|
}
|
|
|
|
}
|