xmpp-rs/src/disco.rs

471 lines
17 KiB
Rust
Raw Normal View History

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/.
2018-12-18 14:32:05 +00:00
use crate::data_forms::{DataForm, DataFormType};
use crate::util::error::Error;
use crate::iq::{IqGetPayload, IqResultPayload};
2018-12-18 14:32:05 +00:00
use crate::ns;
use jid::Jid;
use minidom::Element;
use try_from::TryFrom;
2017-04-18 19:44:36 +00:00
generate_element!(
/// Structure representing a `<query xmlns='http://jabber.org/protocol/disco#info'/>` element.
///
/// It should only be used in an `<iq type='get'/>`, as it can only represent
/// the request, and not a result.
DiscoInfoQuery, "query", DISCO_INFO,
attributes: [
/// Node on which we are doing the discovery.
node: Option<String> = "node" => optional,
]);
2017-07-20 16:39:59 +00:00
impl IqGetPayload for DiscoInfoQuery {}
generate_element!(
/// Structure representing a `<feature xmlns='http://jabber.org/protocol/disco#info'/>` element.
#[derive(PartialEq)]
Feature, "feature", DISCO_INFO,
attributes: [
/// Namespace of the feature we want to represent.
var: String = "var" => required,
]);
impl Feature {
/// Create a new `<feature/>` with the according `@var`.
pub fn new(var: &str) -> Feature {
Feature {
var: String::from(var)
}
}
}
/// Structure representing an `<identity xmlns='http://jabber.org/protocol/disco#info'/>` element.
2017-04-20 23:41:15 +00:00
#[derive(Debug, Clone)]
2017-04-18 19:44:36 +00:00
pub struct Identity {
/// Category of this identity.
2017-04-18 19:44:36 +00:00
pub category: String, // TODO: use an enum here.
/// Type of this identity.
2017-04-18 19:44:36 +00:00
pub type_: String, // TODO: use an enum here.
/// Lang of the name of this identity.
pub lang: Option<String>,
/// Name of this identity.
2017-04-18 19:44:36 +00:00
pub name: Option<String>,
}
impl TryFrom<Element> for Identity {
type Err = Error;
fn try_from(elem: Element) -> Result<Identity, Error> {
2018-05-14 14:30:28 +00:00
check_self!(elem, "identity", DISCO_INFO, "disco#info identity");
check_no_children!(elem, "disco#info identity");
2018-12-18 14:32:05 +00:00
check_no_unknown_attributes!(
elem,
"disco#info identity",
["category", "type", "xml:lang", "name"]
);
let category = get_attr!(elem, "category", required);
if category == "" {
2018-12-18 14:32:05 +00:00
return Err(Error::ParseError(
"Identity must have a non-empty 'category' attribute.",
));
}
let type_ = get_attr!(elem, "type", required);
if type_ == "" {
2018-12-18 14:32:05 +00:00
return Err(Error::ParseError(
"Identity must have a non-empty 'type' attribute.",
));
}
Ok(Identity {
category: category,
type_: type_,
lang: get_attr!(elem, "xml:lang", optional),
name: get_attr!(elem, "name", optional),
})
}
}
impl From<Identity> for Element {
fn from(identity: Identity) -> Element {
Element::builder("identity")
2018-12-18 14:32:05 +00:00
.ns(ns::DISCO_INFO)
.attr("category", identity.category)
.attr("type", identity.type_)
.attr("xml:lang", identity.lang)
.attr("name", identity.name)
.build()
}
}
/// Structure representing a `<query xmlns='http://jabber.org/protocol/disco#info'/>` element.
///
/// It should only be used in an `<iq type='result'/>`, as it can only
/// represent the result, and not a request.
2017-04-20 23:41:15 +00:00
#[derive(Debug, Clone)]
2017-07-20 16:39:59 +00:00
pub struct DiscoInfoResult {
/// Node on which we have done this discovery.
2017-04-18 19:44:36 +00:00
pub node: Option<String>,
/// List of identities exposed by this entity.
2017-04-18 19:44:36 +00:00
pub identities: Vec<Identity>,
/// List of features supported by this entity.
2017-04-18 19:44:36 +00:00
pub features: Vec<Feature>,
/// List of extensions reported by this entity.
2017-04-18 19:44:36 +00:00
pub extensions: Vec<DataForm>,
}
impl IqResultPayload for DiscoInfoResult {}
2017-07-20 16:39:59 +00:00
impl TryFrom<Element> for DiscoInfoResult {
type Err = Error;
2017-04-18 19:44:36 +00:00
2017-07-20 16:39:59 +00:00
fn try_from(elem: Element) -> Result<DiscoInfoResult, Error> {
2018-05-14 14:30:28 +00:00
check_self!(elem, "query", DISCO_INFO, "disco#info result");
check_no_unknown_attributes!(elem, "disco#info result", ["node"]);
2017-04-18 19:44:36 +00:00
let mut result = DiscoInfoResult {
node: get_attr!(elem, "node", optional),
2018-12-18 14:32:05 +00:00
identities: vec![],
features: vec![],
extensions: vec![],
};
2017-05-06 20:01:15 +00:00
for child in elem.children() {
if child.is("identity", ns::DISCO_INFO) {
let identity = Identity::try_from(child.clone())?;
result.identities.push(identity);
} else if child.is("feature", ns::DISCO_INFO) {
let feature = Feature::try_from(child.clone())?;
result.features.push(feature);
2017-05-06 20:01:15 +00:00
} else if child.is("x", ns::DATA_FORMS) {
let data_form = DataForm::try_from(child.clone())?;
if data_form.type_ != DataFormType::Result_ {
2018-12-18 14:32:05 +00:00
return Err(Error::ParseError(
"Data form must have a 'result' type in disco#info.",
));
2017-05-06 20:01:15 +00:00
}
if data_form.form_type.is_none() {
return Err(Error::ParseError("Data form found without a FORM_TYPE."));
2017-05-06 20:01:15 +00:00
}
result.extensions.push(data_form);
2017-05-06 20:01:15 +00:00
} else {
return Err(Error::ParseError("Unknown element in disco#info."));
2017-04-18 19:44:36 +00:00
}
}
if result.identities.is_empty() {
2018-12-18 14:32:05 +00:00
return Err(Error::ParseError(
"There must be at least one identity in disco#info.",
));
2017-05-06 20:01:15 +00:00
}
if result.features.is_empty() {
2018-12-18 14:32:05 +00:00
return Err(Error::ParseError(
"There must be at least one feature in disco#info.",
));
2017-05-06 20:01:15 +00:00
}
2018-12-18 14:32:05 +00:00
if !result.features.contains(&Feature {
var: ns::DISCO_INFO.to_owned(),
}) {
return Err(Error::ParseError(
"disco#info feature not present in disco#info.",
));
2017-05-06 20:01:15 +00:00
}
Ok(result)
2017-04-18 19:44:36 +00:00
}
}
impl From<DiscoInfoResult> for Element {
fn from(disco: DiscoInfoResult) -> Element {
Element::builder("query")
2018-12-18 14:32:05 +00:00
.ns(ns::DISCO_INFO)
.attr("node", disco.node)
.append(disco.identities)
.append(disco.features)
.append(disco.extensions)
.build()
2017-04-20 20:03:02 +00:00
}
}
generate_element!(
/// Structure representing a `<query xmlns='http://jabber.org/protocol/disco#items'/>` element.
///
/// It should only be used in an `<iq type='get'/>`, as it can only represent
/// the request, and not a result.
DiscoItemsQuery, "query", DISCO_ITEMS,
attributes: [
/// Node on which we are doing the discovery.
node: Option<String> = "node" => optional,
]);
2017-07-21 16:33:58 +00:00
impl IqGetPayload for DiscoItemsQuery {}
generate_element!(
/// Structure representing an `<item xmlns='http://jabber.org/protocol/disco#items'/>` element.
Item, "item", DISCO_ITEMS,
attributes: [
/// JID of the entity pointed by this item.
jid: Jid = "jid" => required,
/// Node of the entity pointed by this item.
node: Option<String> = "node" => optional,
/// Name of the entity pointed by this item.
name: Option<String> = "name" => optional,
]);
2017-07-21 16:33:58 +00:00
generate_element!(
/// Structure representing a `<query
/// xmlns='http://jabber.org/protocol/disco#items'/>` element.
///
/// It should only be used in an `<iq type='result'/>`, as it can only
/// represent the result, and not a request.
2018-05-14 14:30:28 +00:00
DiscoItemsResult, "query", DISCO_ITEMS,
attributes: [
/// Node on which we have done this discovery.
node: Option<String> = "node" => optional
],
children: [
/// List of items pointed by this entity.
2018-05-14 14:30:28 +00:00
items: Vec<Item> = ("item", DISCO_ITEMS) => Item
]
);
2017-07-21 16:33:58 +00:00
impl IqResultPayload for DiscoItemsResult {}
2017-04-18 19:44:36 +00:00
#[cfg(test)]
mod tests {
2017-05-06 20:01:15 +00:00
use super::*;
use crate::util::compare_elements::NamespaceAwareCompare;
2017-07-21 16:33:58 +00:00
use std::str::FromStr;
2017-04-18 19:44:36 +00:00
2018-10-28 12:10:48 +00:00
#[cfg(target_pointer_width = "32")]
#[test]
fn test_size() {
assert_size!(Identity, 48);
assert_size!(Feature, 12);
assert_size!(DiscoInfoQuery, 12);
assert_size!(DiscoInfoResult, 48);
assert_size!(Item, 60);
assert_size!(DiscoItemsQuery, 12);
assert_size!(DiscoItemsResult, 24);
}
#[cfg(target_pointer_width = "64")]
#[test]
fn test_size() {
assert_size!(Identity, 96);
assert_size!(Feature, 24);
assert_size!(DiscoInfoQuery, 24);
assert_size!(DiscoInfoResult, 96);
assert_size!(Item, 120);
assert_size!(DiscoItemsQuery, 24);
assert_size!(DiscoItemsResult, 48);
}
2017-04-18 19:44:36 +00:00
#[test]
fn test_simple() {
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/><feature var='http://jabber.org/protocol/disco#info'/></query>".parse().unwrap();
2017-07-20 16:39:59 +00:00
let query = DiscoInfoResult::try_from(elem).unwrap();
2017-04-18 19:44:36 +00:00
assert!(query.node.is_none());
assert_eq!(query.identities.len(), 1);
assert_eq!(query.features.len(), 1);
assert!(query.extensions.is_empty());
}
#[test]
fn test_identity_after_feature() {
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><feature var='http://jabber.org/protocol/disco#info'/><identity category='client' type='pc'/></query>".parse().unwrap();
let query = DiscoInfoResult::try_from(elem).unwrap();
assert_eq!(query.identities.len(), 1);
assert_eq!(query.features.len(), 1);
assert!(query.extensions.is_empty());
}
#[test]
fn test_feature_after_dataform() {
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/><x xmlns='jabber:x:data' type='result'><field var='FORM_TYPE' type='hidden'><value>coucou</value></field></x><feature var='http://jabber.org/protocol/disco#info'/></query>".parse().unwrap();
let query = DiscoInfoResult::try_from(elem).unwrap();
assert_eq!(query.identities.len(), 1);
assert_eq!(query.features.len(), 1);
assert_eq!(query.extensions.len(), 1);
}
#[test]
fn test_extension() {
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/><feature var='http://jabber.org/protocol/disco#info'/><x xmlns='jabber:x:data' type='result'><field var='FORM_TYPE' type='hidden'><value>example</value></field></x></query>".parse().unwrap();
let elem1 = elem.clone();
let query = DiscoInfoResult::try_from(elem).unwrap();
assert!(query.node.is_none());
assert_eq!(query.identities.len(), 1);
assert_eq!(query.features.len(), 1);
assert_eq!(query.extensions.len(), 1);
assert_eq!(query.extensions[0].form_type, Some(String::from("example")));
let elem2 = query.into();
assert!(elem1.compare_to(&elem2));
}
2017-04-18 19:44:36 +00:00
#[test]
fn test_invalid() {
2018-12-18 14:32:05 +00:00
let elem: Element =
"<query xmlns='http://jabber.org/protocol/disco#info'><coucou/></query>"
.parse()
.unwrap();
2017-07-20 16:39:59 +00:00
let error = DiscoInfoResult::try_from(elem).unwrap_err();
2017-04-18 19:44:36 +00:00
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
assert_eq!(message, "Unknown element in disco#info.");
}
2017-04-18 19:44:36 +00:00
#[test]
fn test_invalid_identity() {
2018-12-18 14:32:05 +00:00
let elem: Element =
"<query xmlns='http://jabber.org/protocol/disco#info'><identity/></query>"
.parse()
.unwrap();
2017-07-20 16:39:59 +00:00
let error = DiscoInfoResult::try_from(elem).unwrap_err();
2017-04-18 19:44:36 +00:00
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
assert_eq!(message, "Required attribute 'category' missing.");
2017-04-18 19:44:36 +00:00
2018-12-18 14:32:05 +00:00
let elem: Element =
"<query xmlns='http://jabber.org/protocol/disco#info'><identity category=''/></query>"
.parse()
.unwrap();
2017-07-20 16:39:59 +00:00
let error = DiscoInfoResult::try_from(elem).unwrap_err();
2017-04-18 19:44:36 +00:00
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
2018-12-18 14:32:05 +00:00
assert_eq!(
message,
"Identity must have a non-empty 'category' attribute."
);
2017-04-18 19:44:36 +00:00
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='coucou'/></query>".parse().unwrap();
2017-07-20 16:39:59 +00:00
let error = DiscoInfoResult::try_from(elem).unwrap_err();
2017-04-18 19:44:36 +00:00
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
assert_eq!(message, "Required attribute 'type' missing.");
2017-04-18 19:44:36 +00:00
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='coucou' type=''/></query>".parse().unwrap();
2017-07-20 16:39:59 +00:00
let error = DiscoInfoResult::try_from(elem).unwrap_err();
2017-04-18 19:44:36 +00:00
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
assert_eq!(message, "Identity must have a non-empty 'type' attribute.");
}
2017-04-18 19:44:36 +00:00
#[test]
fn test_invalid_feature() {
2018-12-18 14:32:05 +00:00
let elem: Element =
"<query xmlns='http://jabber.org/protocol/disco#info'><feature/></query>"
.parse()
.unwrap();
2017-07-20 16:39:59 +00:00
let error = DiscoInfoResult::try_from(elem).unwrap_err();
2017-04-18 19:44:36 +00:00
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
assert_eq!(message, "Required attribute 'var' missing.");
}
#[test]
fn test_invalid_result() {
2018-12-18 14:32:05 +00:00
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'/>"
.parse()
.unwrap();
2017-07-20 16:39:59 +00:00
let error = DiscoInfoResult::try_from(elem).unwrap_err();
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
2018-12-18 14:32:05 +00:00
assert_eq!(
message,
"There must be at least one identity in disco#info."
);
2017-04-18 19:44:36 +00:00
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/></query>".parse().unwrap();
2017-07-20 16:39:59 +00:00
let error = DiscoInfoResult::try_from(elem).unwrap_err();
2017-04-18 19:44:36 +00:00
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
assert_eq!(message, "There must be at least one feature in disco#info.");
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/><feature var='http://jabber.org/protocol/disco#items'/></query>".parse().unwrap();
2017-07-20 16:39:59 +00:00
let error = DiscoInfoResult::try_from(elem).unwrap_err();
2017-04-18 19:44:36 +00:00
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
assert_eq!(message, "disco#info feature not present in disco#info.");
}
2017-07-21 16:33:58 +00:00
#[test]
fn test_simple_items() {
2018-12-18 14:32:05 +00:00
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#items'/>"
.parse()
.unwrap();
2017-07-21 16:33:58 +00:00
let query = DiscoItemsQuery::try_from(elem).unwrap();
assert!(query.node.is_none());
2018-12-18 14:32:05 +00:00
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#items' node='coucou'/>"
.parse()
.unwrap();
2017-07-21 16:33:58 +00:00
let query = DiscoItemsQuery::try_from(elem).unwrap();
assert_eq!(query.node, Some(String::from("coucou")));
}
#[test]
fn test_simple_items_result() {
2018-12-18 14:32:05 +00:00
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#items'/>"
.parse()
.unwrap();
2017-07-21 16:33:58 +00:00
let query = DiscoItemsResult::try_from(elem).unwrap();
assert!(query.node.is_none());
assert!(query.items.is_empty());
2018-12-18 14:32:05 +00:00
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#items' node='coucou'/>"
.parse()
.unwrap();
2017-07-21 16:33:58 +00:00
let query = DiscoItemsResult::try_from(elem).unwrap();
assert_eq!(query.node, Some(String::from("coucou")));
assert!(query.items.is_empty());
}
#[test]
fn test_answers_items_result() {
let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#items'><item jid='component'/><item jid='component2' node='test' name='A component'/></query>".parse().unwrap();
let query = DiscoItemsResult::try_from(elem).unwrap();
let elem2 = Element::from(query);
let query = DiscoItemsResult::try_from(elem2).unwrap();
2017-07-21 16:33:58 +00:00
assert_eq!(query.items.len(), 2);
assert_eq!(query.items[0].jid, Jid::from_str("component").unwrap());
assert_eq!(query.items[0].node, None);
assert_eq!(query.items[0].name, None);
assert_eq!(query.items[1].jid, Jid::from_str("component2").unwrap());
assert_eq!(query.items[1].node, Some(String::from("test")));
assert_eq!(query.items[1].name, Some(String::from("A component")));
}
2017-04-18 19:44:36 +00:00
}