xmpp-rs/parsers/src/bind.rs

204 lines
6.3 KiB
Rust
Raw Normal View History

2018-02-20 16:43:19 +00:00
// Copyright (c) 2018 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::iq::{IqResultPayload, IqSetPayload};
use crate::ns;
use jid::{FullJid, Jid};
use minidom::Element;
use std::str::FromStr;
use xso::error::{Error, FromElementError};
2018-02-20 16:43:19 +00:00
2018-08-08 17:44:55 +00:00
/// The request for resource binding, which is the process by which a client
/// can obtain a full JID and start exchanging on the XMPP network.
///
/// See <https://xmpp.org/rfcs/rfc6120.html#bind>
#[derive(Debug, Clone, PartialEq)]
2019-07-31 11:52:08 +00:00
pub struct BindQuery {
2018-08-08 17:44:55 +00:00
/// Requests this resource, the server may associate another one though.
///
/// If this is None, we request no particular resource, and a random one
/// will be affected by the server.
resource: Option<String>,
2018-02-20 16:43:19 +00:00
}
2019-07-31 11:52:08 +00:00
impl BindQuery {
2018-08-08 17:44:55 +00:00
/// Creates a resource binding request.
2019-07-31 11:52:08 +00:00
pub fn new(resource: Option<String>) -> BindQuery {
BindQuery { resource }
2018-02-20 16:43:19 +00:00
}
}
2019-07-31 11:52:08 +00:00
impl IqSetPayload for BindQuery {}
2019-07-31 11:52:08 +00:00
impl TryFrom<Element> for BindQuery {
type Error = FromElementError;
2018-02-20 16:43:19 +00:00
fn try_from(elem: Element) -> Result<BindQuery, FromElementError> {
2018-05-14 14:30:28 +00:00
check_self!(elem, "bind", BIND);
2018-02-20 16:43:19 +00:00
check_no_attributes!(elem, "bind");
let mut resource = None;
2018-02-20 16:43:19 +00:00
for child in elem.children() {
if resource.is_some() {
return Err(Error::Other("Bind can only have one child.").into());
2018-02-20 16:43:19 +00:00
}
if child.is("resource", ns::BIND) {
check_no_attributes!(child, "resource");
2018-02-20 16:43:19 +00:00
check_no_children!(child, "resource");
resource = Some(child.text());
2018-02-20 16:43:19 +00:00
} else {
return Err(Error::Other("Unknown element in bind request.").into());
2018-02-20 16:43:19 +00:00
}
}
2019-07-31 11:52:08 +00:00
Ok(BindQuery { resource })
2018-02-20 16:43:19 +00:00
}
}
2019-07-31 11:52:08 +00:00
impl From<BindQuery> for Element {
fn from(bind: BindQuery) -> Element {
Element::builder("bind", ns::BIND)
.append_all(
bind.resource
.map(|resource| Element::builder("resource", ns::BIND).append(resource)),
)
2018-12-18 14:32:05 +00:00
.build()
2018-02-20 16:43:19 +00:00
}
}
/// The response for resource binding, containing the clients full JID.
///
/// See <https://xmpp.org/rfcs/rfc6120.html#bind>
#[derive(Debug, Clone, PartialEq)]
pub struct BindResponse {
/// The full JID returned by the server for this client.
jid: FullJid,
}
impl IqResultPayload for BindResponse {}
impl From<BindResponse> for FullJid {
fn from(bind: BindResponse) -> FullJid {
bind.jid
}
}
impl From<BindResponse> for Jid {
fn from(bind: BindResponse) -> Jid {
Jid::from(bind.jid)
}
}
impl TryFrom<Element> for BindResponse {
type Error = FromElementError;
fn try_from(elem: Element) -> Result<BindResponse, FromElementError> {
check_self!(elem, "bind", BIND);
check_no_attributes!(elem, "bind");
let mut jid = None;
for child in elem.children() {
if jid.is_some() {
return Err(Error::Other("Bind can only have one child.").into());
}
if child.is("jid", ns::BIND) {
check_no_attributes!(child, "jid");
check_no_children!(child, "jid");
jid = Some(FullJid::from_str(&child.text()).map_err(Error::text_parse_error)?);
} else {
return Err(Error::Other("Unknown element in bind response.").into());
}
}
Ok(BindResponse {
jid: match jid {
None => {
return Err(Error::Other("Bind response must contain a jid element.").into())
}
Some(jid) => jid,
},
})
}
}
impl From<BindResponse> for Element {
fn from(bind: BindResponse) -> Element {
Element::builder("bind", ns::BIND)
.append(Element::builder("jid", ns::BIND).append(bind.jid))
.build()
}
}
2018-02-20 16:43:19 +00:00
#[cfg(test)]
mod tests {
use super::*;
2018-10-28 12:10:48 +00:00
#[cfg(target_pointer_width = "32")]
#[test]
fn test_size() {
2019-07-31 11:52:08 +00:00
assert_size!(BindQuery, 12);
2023-06-20 12:14:15 +00:00
assert_size!(BindResponse, 16);
2018-10-28 12:10:48 +00:00
}
#[cfg(target_pointer_width = "64")]
#[test]
fn test_size() {
2019-07-31 11:52:08 +00:00
assert_size!(BindQuery, 24);
2023-06-20 12:08:58 +00:00
assert_size!(BindResponse, 32);
}
2018-02-20 16:43:19 +00:00
#[test]
fn test_simple() {
2018-12-18 14:32:05 +00:00
let elem: Element = "<bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'/>"
.parse()
.unwrap();
2019-07-31 11:52:08 +00:00
let bind = BindQuery::try_from(elem).unwrap();
assert_eq!(bind.resource, None);
let elem: Element =
"<bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'><resource>Hello™</resource></bind>"
.parse()
.unwrap();
2019-07-31 11:52:08 +00:00
let bind = BindQuery::try_from(elem).unwrap();
// FIXME: “™” should be resourceprepd into “TM” here…
//assert_eq!(bind.resource.unwrap(), "HelloTM");
assert_eq!(bind.resource.unwrap(), "Hello™");
let elem: Element = "<bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'><jid>coucou@linkmauve.fr/Hello™</jid></bind>"
.parse()
.unwrap();
let bind = BindResponse::try_from(elem).unwrap();
assert_eq!(
bind.jid,
FullJid::new("coucou@linkmauve.fr/HelloTM").unwrap()
);
2018-02-20 16:43:19 +00:00
}
#[cfg(not(feature = "disable-validation"))]
#[test]
fn test_invalid_resource() {
let elem: Element = "<bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'><resource attr='coucou'>resource</resource></bind>"
.parse()
.unwrap();
2019-07-31 11:52:08 +00:00
let error = BindQuery::try_from(elem).unwrap_err();
let message = match error {
FromElementError::Invalid(Error::Other(string)) => string,
_ => panic!(),
};
assert_eq!(message, "Unknown attribute in resource element.");
let elem: Element = "<bind xmlns='urn:ietf:params:xml:ns:xmpp-bind'><resource><hello-world/>resource</resource></bind>"
.parse()
.unwrap();
2019-07-31 11:52:08 +00:00
let error = BindQuery::try_from(elem).unwrap_err();
let message = match error {
FromElementError::Invalid(Error::Other(string)) => string,
_ => panic!(),
};
assert_eq!(message, "Unknown child in resource element.");
}
2018-02-20 16:43:19 +00:00
}