2017-10-31 15:48:11 +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/.
|
|
|
|
|
2024-06-21 15:54:12 +00:00
|
|
|
use xso::{FromXml, IntoXml};
|
|
|
|
|
2018-12-18 14:32:05 +00:00
|
|
|
use crate::iq::{IqGetPayload, IqResultPayload, IqSetPayload};
|
2018-12-18 14:27:30 +00:00
|
|
|
use crate::ns;
|
2019-09-25 08:28:44 +00:00
|
|
|
use crate::Element;
|
2019-10-22 23:32:41 +00:00
|
|
|
use jid::Jid;
|
2024-06-21 14:27:43 +00:00
|
|
|
use xso::error::FromElementError;
|
2017-10-31 15:48:11 +00:00
|
|
|
|
2024-06-21 15:54:12 +00:00
|
|
|
/// The element requesting the blocklist, the result iq will contain a
|
|
|
|
/// [BlocklistResult].
|
|
|
|
#[derive(FromXml, IntoXml, PartialEq, Debug, Clone)]
|
|
|
|
#[xml(namespace = ns::BLOCKING, name = "blocklist")]
|
|
|
|
pub struct BlocklistRequest;
|
2017-10-31 15:48:11 +00:00
|
|
|
|
2018-05-16 13:08:17 +00:00
|
|
|
impl IqGetPayload for BlocklistRequest {}
|
|
|
|
|
2017-10-31 15:48:11 +00:00
|
|
|
macro_rules! generate_blocking_element {
|
2018-08-02 16:15:56 +00:00
|
|
|
($(#[$meta:meta])* $elem:ident, $name:tt) => (
|
|
|
|
$(#[$meta])*
|
2017-10-31 15:48:11 +00:00
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct $elem {
|
2018-08-02 16:15:56 +00:00
|
|
|
/// List of JIDs affected by this command.
|
2017-10-31 15:48:11 +00:00
|
|
|
pub items: Vec<Jid>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Element> for $elem {
|
2024-06-21 14:27:43 +00:00
|
|
|
type Error = FromElementError;
|
2017-10-31 15:48:11 +00:00
|
|
|
|
2024-06-21 14:27:43 +00:00
|
|
|
fn try_from(elem: Element) -> Result<$elem, FromElementError> {
|
2018-05-14 14:30:28 +00:00
|
|
|
check_self!(elem, $name, BLOCKING);
|
2017-10-31 15:48:11 +00:00
|
|
|
check_no_attributes!(elem, $name);
|
2017-10-31 18:53:51 +00:00
|
|
|
let mut items = vec!();
|
|
|
|
for child in elem.children() {
|
2024-03-02 08:19:49 +00:00
|
|
|
check_child!(child, "item", BLOCKING);
|
2017-10-31 18:53:51 +00:00
|
|
|
check_no_unknown_attributes!(child, "item", ["jid"]);
|
|
|
|
check_no_children!(child, "item");
|
2019-02-24 19:48:19 +00:00
|
|
|
items.push(get_attr!(child, "jid", Required));
|
2017-10-31 18:53:51 +00:00
|
|
|
}
|
|
|
|
Ok($elem { items })
|
2017-10-31 15:48:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<$elem> for Element {
|
|
|
|
fn from(elem: $elem) -> Element {
|
2020-03-28 12:07:26 +00:00
|
|
|
Element::builder($name, ns::BLOCKING)
|
2019-07-24 22:20:38 +00:00
|
|
|
.append_all(elem.items.into_iter().map(|jid| {
|
2020-03-28 12:07:26 +00:00
|
|
|
Element::builder("item", ns::BLOCKING)
|
2017-10-31 15:48:11 +00:00
|
|
|
.attr("jid", jid)
|
2019-07-24 22:20:38 +00:00
|
|
|
}))
|
2017-10-31 15:48:11 +00:00
|
|
|
.build()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-08-02 16:15:56 +00:00
|
|
|
generate_blocking_element!(
|
|
|
|
/// The element containing the current blocklist, as a reply from
|
|
|
|
/// [BlocklistRequest].
|
2018-12-18 14:32:05 +00:00
|
|
|
BlocklistResult,
|
|
|
|
"blocklist"
|
2018-08-02 16:15:56 +00:00
|
|
|
);
|
2017-10-31 15:48:11 +00:00
|
|
|
|
2018-05-16 13:08:17 +00:00
|
|
|
impl IqResultPayload for BlocklistResult {}
|
2018-08-02 16:15:56 +00:00
|
|
|
|
|
|
|
// TODO: Prevent zero elements from being allowed.
|
|
|
|
generate_blocking_element!(
|
|
|
|
/// A query to block one or more JIDs.
|
2018-12-18 14:32:05 +00:00
|
|
|
Block,
|
|
|
|
"block"
|
2018-08-02 16:15:56 +00:00
|
|
|
);
|
|
|
|
|
2018-05-16 13:08:17 +00:00
|
|
|
impl IqSetPayload for Block {}
|
2018-08-02 16:15:56 +00:00
|
|
|
|
|
|
|
generate_blocking_element!(
|
|
|
|
/// A query to unblock one or more JIDs, or all of them.
|
|
|
|
///
|
|
|
|
/// Warning: not putting any JID there means clearing out the blocklist.
|
2018-12-18 14:32:05 +00:00
|
|
|
Unblock,
|
|
|
|
"unblock"
|
2018-08-02 16:15:56 +00:00
|
|
|
);
|
|
|
|
|
2018-05-16 13:08:17 +00:00
|
|
|
impl IqSetPayload for Unblock {}
|
|
|
|
|
2024-06-21 15:54:12 +00:00
|
|
|
/// The application-specific error condition when a message is blocked.
|
|
|
|
#[derive(FromXml, IntoXml, PartialEq, Debug, Clone)]
|
|
|
|
#[xml(namespace = ns::BLOCKING_ERRORS, name = "blocked")]
|
|
|
|
pub struct Blocked;
|
2017-10-31 15:48:11 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2024-06-21 14:27:43 +00:00
|
|
|
use xso::error::Error;
|
|
|
|
|
2017-10-31 15:48:11 +00:00
|
|
|
use super::*;
|
|
|
|
|
2018-10-28 12:10:48 +00:00
|
|
|
#[cfg(target_pointer_width = "32")]
|
|
|
|
#[test]
|
|
|
|
fn test_size() {
|
|
|
|
assert_size!(BlocklistRequest, 0);
|
|
|
|
assert_size!(BlocklistResult, 12);
|
|
|
|
assert_size!(Block, 12);
|
|
|
|
assert_size!(Unblock, 12);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(target_pointer_width = "64")]
|
2018-10-26 12:26:16 +00:00
|
|
|
#[test]
|
|
|
|
fn test_size() {
|
|
|
|
assert_size!(BlocklistRequest, 0);
|
|
|
|
assert_size!(BlocklistResult, 24);
|
|
|
|
assert_size!(Block, 24);
|
|
|
|
assert_size!(Unblock, 24);
|
|
|
|
}
|
|
|
|
|
2017-10-31 15:48:11 +00:00
|
|
|
#[test]
|
|
|
|
fn test_simple() {
|
|
|
|
let elem: Element = "<blocklist xmlns='urn:xmpp:blocking'/>".parse().unwrap();
|
|
|
|
let request_elem = elem.clone();
|
|
|
|
BlocklistRequest::try_from(request_elem).unwrap();
|
|
|
|
|
|
|
|
let result_elem = elem.clone();
|
|
|
|
let result = BlocklistResult::try_from(result_elem).unwrap();
|
2020-12-10 19:36:54 +00:00
|
|
|
assert!(result.items.is_empty());
|
2017-10-31 15:48:11 +00:00
|
|
|
|
|
|
|
let elem: Element = "<block xmlns='urn:xmpp:blocking'/>".parse().unwrap();
|
|
|
|
let block = Block::try_from(elem).unwrap();
|
2020-12-10 19:36:54 +00:00
|
|
|
assert!(block.items.is_empty());
|
2017-10-31 15:48:11 +00:00
|
|
|
|
|
|
|
let elem: Element = "<unblock xmlns='urn:xmpp:blocking'/>".parse().unwrap();
|
|
|
|
let unblock = Unblock::try_from(elem).unwrap();
|
2020-12-10 19:36:54 +00:00
|
|
|
assert!(unblock.items.is_empty());
|
2017-10-31 15:48:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_items() {
|
|
|
|
let elem: Element = "<blocklist xmlns='urn:xmpp:blocking'><item jid='coucou@coucou'/><item jid='domain'/></blocklist>".parse().unwrap();
|
2018-12-18 14:32:05 +00:00
|
|
|
let two_items = vec![
|
2023-06-20 12:07:50 +00:00
|
|
|
Jid::new("coucou@coucou").unwrap(),
|
|
|
|
Jid::new("domain").unwrap(),
|
2018-12-18 14:32:05 +00:00
|
|
|
];
|
2017-10-31 15:48:11 +00:00
|
|
|
|
|
|
|
let result_elem = elem.clone();
|
|
|
|
let result = BlocklistResult::try_from(result_elem).unwrap();
|
|
|
|
assert_eq!(result.items, two_items);
|
|
|
|
|
|
|
|
let elem: Element = "<block xmlns='urn:xmpp:blocking'><item jid='coucou@coucou'/><item jid='domain'/></block>".parse().unwrap();
|
|
|
|
let block = Block::try_from(elem).unwrap();
|
|
|
|
assert_eq!(block.items, two_items);
|
|
|
|
|
|
|
|
let elem: Element = "<unblock xmlns='urn:xmpp:blocking'><item jid='coucou@coucou'/><item jid='domain'/></unblock>".parse().unwrap();
|
|
|
|
let unblock = Unblock::try_from(elem).unwrap();
|
|
|
|
assert_eq!(unblock.items, two_items);
|
|
|
|
}
|
|
|
|
|
2019-01-12 21:00:46 +00:00
|
|
|
#[cfg(not(feature = "disable-validation"))]
|
2017-10-31 15:48:11 +00:00
|
|
|
#[test]
|
|
|
|
fn test_invalid() {
|
2018-12-18 14:32:05 +00:00
|
|
|
let elem: Element = "<blocklist xmlns='urn:xmpp:blocking' coucou=''/>"
|
|
|
|
.parse()
|
|
|
|
.unwrap();
|
2017-10-31 15:48:11 +00:00
|
|
|
let request_elem = elem.clone();
|
|
|
|
let error = BlocklistRequest::try_from(request_elem).unwrap_err();
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-10-31 15:48:11 +00:00
|
|
|
_ => panic!(),
|
|
|
|
};
|
|
|
|
assert_eq!(message, "Unknown attribute in blocklist element.");
|
|
|
|
|
|
|
|
let result_elem = elem.clone();
|
|
|
|
let error = BlocklistResult::try_from(result_elem).unwrap_err();
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-10-31 15:48:11 +00:00
|
|
|
_ => panic!(),
|
|
|
|
};
|
|
|
|
assert_eq!(message, "Unknown attribute in blocklist element.");
|
|
|
|
|
2018-12-18 14:32:05 +00:00
|
|
|
let elem: Element = "<block xmlns='urn:xmpp:blocking' coucou=''/>"
|
|
|
|
.parse()
|
|
|
|
.unwrap();
|
2017-10-31 15:48:11 +00:00
|
|
|
let error = Block::try_from(elem).unwrap_err();
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-10-31 15:48:11 +00:00
|
|
|
_ => panic!(),
|
|
|
|
};
|
|
|
|
assert_eq!(message, "Unknown attribute in block element.");
|
|
|
|
|
2018-12-18 14:32:05 +00:00
|
|
|
let elem: Element = "<unblock xmlns='urn:xmpp:blocking' coucou=''/>"
|
|
|
|
.parse()
|
|
|
|
.unwrap();
|
2017-10-31 15:48:11 +00:00
|
|
|
let error = Unblock::try_from(elem).unwrap_err();
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-10-31 15:48:11 +00:00
|
|
|
_ => panic!(),
|
|
|
|
};
|
|
|
|
assert_eq!(message, "Unknown attribute in unblock element.");
|
|
|
|
}
|
2019-01-12 19:41:40 +00:00
|
|
|
|
2019-01-12 21:00:46 +00:00
|
|
|
#[cfg(not(feature = "disable-validation"))]
|
2019-01-12 19:41:40 +00:00
|
|
|
#[test]
|
|
|
|
fn test_non_empty_blocklist_request() {
|
|
|
|
let elem: Element = "<blocklist xmlns='urn:xmpp:blocking'><item jid='coucou@coucou'/><item jid='domain'/></blocklist>".parse().unwrap();
|
|
|
|
let error = BlocklistRequest::try_from(elem).unwrap_err();
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2019-01-12 19:41:40 +00:00
|
|
|
_ => panic!(),
|
|
|
|
};
|
|
|
|
assert_eq!(message, "Unknown child in blocklist element.");
|
|
|
|
}
|
2017-10-31 15:48:11 +00:00
|
|
|
}
|