// Copyright (c) 2017 Emmanuel Gil Peyrot // // 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/. use std::convert::TryFrom; use std::str::FromStr; use minidom::{Element, IntoAttributeValue}; use base64; use error::Error; use ns; generate_attribute!(Stanza, "stanza", { Iq => "iq", Message => "message", }, Default = Iq); #[derive(Debug, Clone)] pub enum IBB { Open { block_size: u16, sid: String, stanza: Stanza, }, Data { seq: u16, sid: String, data: Vec, }, Close { sid: String, }, } impl TryFrom for IBB { type Error = Error; fn try_from(elem: Element) -> Result { if elem.is("open", ns::IBB) { for _ in elem.children() { return Err(Error::ParseError("Unknown child in open element.")); } let block_size = get_attr!(elem, "block-size", required); let sid = get_attr!(elem, "sid", required); let stanza = get_attr!(elem, "stanza", default); Ok(IBB::Open { block_size: block_size, sid: sid, stanza: stanza }) } else if elem.is("data", ns::IBB) { for _ in elem.children() { return Err(Error::ParseError("Unknown child in data element.")); } let seq = get_attr!(elem, "seq", required); let sid = get_attr!(elem, "sid", required); let data = base64::decode(&elem.text())?; Ok(IBB::Data { seq: seq, sid: sid, data: data }) } else if elem.is("close", ns::IBB) { for _ in elem.children() { return Err(Error::ParseError("Unknown child in close element.")); } let sid = get_attr!(elem, "sid", required); Ok(IBB::Close { sid: sid, }) } else { Err(Error::ParseError("This is not an ibb element.")) } } } impl Into for IBB { fn into(self) -> Element { match self { IBB::Open { block_size, sid, stanza } => { Element::builder("open") .ns(ns::IBB) .attr("block-size", block_size) .attr("sid", sid) .attr("stanza", stanza) .build() }, IBB::Data { seq, sid, data } => { Element::builder("data") .ns(ns::IBB) .attr("seq", seq) .attr("sid", sid) .append(base64::encode(&data)) .build() }, IBB::Close { sid } => { Element::builder("close") .ns(ns::IBB) .attr("sid", sid) .build() }, } } } #[cfg(test)] mod tests { use super::*; use std::error::Error as StdError; #[test] fn test_simple() { let elem: Element = "".parse().unwrap(); let open = IBB::try_from(elem).unwrap(); match open { IBB::Open { block_size, sid, stanza } => { assert_eq!(block_size, 3); assert_eq!(sid, "coucou"); assert_eq!(stanza, Stanza::Iq); }, _ => panic!(), } let elem: Element = "AAAA".parse().unwrap(); let data = IBB::try_from(elem).unwrap(); match data { IBB::Data { seq, sid, data } => { assert_eq!(seq, 0); assert_eq!(sid, "coucou"); assert_eq!(data, vec!(0, 0, 0)); }, _ => panic!(), } let elem: Element = "".parse().unwrap(); let close = IBB::try_from(elem).unwrap(); match close { IBB::Close { sid } => { assert_eq!(sid, "coucou"); }, _ => panic!(), } } #[test] fn test_invalid() { let elem: Element = "".parse().unwrap(); let error = IBB::try_from(elem).unwrap_err(); let message = match error { Error::ParseError(string) => string, _ => panic!(), }; assert_eq!(message, "Required attribute 'block-size' missing."); let elem: Element = "".parse().unwrap(); let error = IBB::try_from(elem).unwrap_err(); let message = match error { Error::ParseIntError(error) => error, _ => panic!(), }; assert_eq!(message.description(), "invalid digit found in string"); let elem: Element = "".parse().unwrap(); let error = IBB::try_from(elem).unwrap_err(); let message = match error { Error::ParseError(error) => error, _ => panic!(), }; assert_eq!(message, "Required attribute 'sid' missing."); } #[test] fn test_invalid_stanza() { let elem: Element = "".parse().unwrap(); let error = IBB::try_from(elem).unwrap_err(); let message = match error { Error::ParseError(string) => string, _ => panic!(), }; assert_eq!(message, "Unknown value for 'stanza' attribute."); } }