xmpp_parsers/
bookmarks2.rs

1// Copyright (c) 2019 Emmanuel Gil Peyrot <linkmauve@linkmauve.fr>
2//
3// This Source Code Form is subject to the terms of the Mozilla Public
4// License, v. 2.0. If a copy of the MPL was not distributed with this
5// file, You can obtain one at http://mozilla.org/MPL/2.0/.
6
7//!
8//! Chatroom bookmarks from [XEP-0402](https://xmpp.org/extensions/xep-0402.html) for newer servers
9//! which advertise `urn:xmpp:bookmarks:1#compat` on the user's BareJID in a disco info request.
10//! On legacy non-compliant servers, use the [`private`][crate::private] module instead.
11//!
12//! See [ModernXMPP docs](https://docs.modernxmpp.org/client/groupchat/#bookmarks) on how to handle all historic
13//! and newer specifications for your clients.
14
15use xso::{AsXml, FromXml};
16
17use crate::jid::ResourcePart;
18use crate::ns;
19use minidom::Element;
20
21/// Potential extensions in a conference.
22#[derive(FromXml, AsXml, Debug, Clone, Default)]
23#[xml(namespace = ns::BOOKMARKS2, name = "extensions")]
24pub struct Extensions {
25    /// Extension elements.
26    #[xml(element(n = ..))]
27    pub payloads: Vec<Element>,
28}
29
30/// A conference bookmark.
31#[derive(FromXml, AsXml, Debug, Clone, Default)]
32#[xml(namespace = ns::BOOKMARKS2, name = "conference")]
33pub struct Conference {
34    /// Whether a conference bookmark should be joined automatically.
35    #[xml(attribute(default))]
36    pub autojoin: bool,
37
38    /// A user-defined name for this conference.
39    #[xml(attribute(default))]
40    pub name: Option<String>,
41
42    /// The nick the user will use to join this conference.
43    #[xml(extract(default, fields(text(type_ = ResourcePart))))]
44    pub nick: Option<ResourcePart>,
45
46    /// The password required to join this conference.
47    #[xml(extract(default, fields(text(type_ = String))))]
48    pub password: Option<String>,
49
50    /// Extension elements.
51    #[xml(child(default))]
52    pub extensions: Option<Extensions>,
53}
54
55impl Conference {
56    /// Create a new conference.
57    pub fn new() -> Conference {
58        Conference::default()
59    }
60}
61
62#[cfg(test)]
63mod tests {
64    use super::*;
65    use crate::pubsub::{self, pubsub::Item as PubSubItem};
66
67    #[cfg(target_pointer_width = "32")]
68    #[test]
69    fn test_size() {
70        assert_size!(Conference, 52);
71    }
72
73    #[cfg(target_pointer_width = "64")]
74    #[test]
75    fn test_size() {
76        assert_size!(Conference, 104);
77    }
78
79    #[test]
80    fn simple() {
81        let elem: Element = "<conference xmlns='urn:xmpp:bookmarks:1' autojoin='false'/>"
82            .parse()
83            .unwrap();
84        let elem1 = elem.clone();
85        let conference = Conference::try_from(elem).unwrap();
86        assert_eq!(conference.autojoin, false);
87        assert_eq!(conference.name, None);
88        assert_eq!(conference.nick, None);
89        assert_eq!(conference.password, None);
90
91        let elem2 = Element::from(Conference::new());
92        assert_eq!(elem1, elem2);
93    }
94
95    #[test]
96    fn wrong_resource() {
97        // This emoji is not valid according to Resource prep
98        let elem: Element = "<conference xmlns='urn:xmpp:bookmarks:1' autojoin='true'><nick>Whatever\u{1F469}\u{1F3FE}\u{200D}\u{2764}\u{FE0F}\u{200D}\u{1F469}\u{1F3FC}</nick></conference>".parse().unwrap();
99        let res = Conference::try_from(elem);
100        assert!(res.is_err());
101        assert_eq!(
102            res.unwrap_err().to_string().as_str(),
103            "text parse error: resource doesn’t pass resourceprep validation"
104        );
105    }
106
107    #[test]
108    fn complete() {
109        let elem: Element = "<conference xmlns='urn:xmpp:bookmarks:1' autojoin='true' name='Test MUC'><nick>Coucou</nick><password>secret</password><extensions><test xmlns='urn:xmpp:unknown' /></extensions></conference>".parse().unwrap();
110        let conference = Conference::try_from(elem).unwrap();
111        assert_eq!(conference.autojoin, true);
112        assert_eq!(conference.name, Some(String::from("Test MUC")));
113        assert_eq!(conference.clone().nick.unwrap().as_str(), "Coucou");
114        assert_eq!(conference.clone().password.unwrap(), "secret");
115        let payloads = conference.clone().extensions.unwrap().payloads;
116        assert_eq!(payloads.len(), 1);
117        assert!(payloads[0].is("test", "urn:xmpp:unknown"));
118    }
119
120    #[test]
121    fn wrapped() {
122        let elem: Element = "<item xmlns='http://jabber.org/protocol/pubsub' id='test-muc@muc.localhost'><conference xmlns='urn:xmpp:bookmarks:1' autojoin='true' name='Test MUC'><nick>Coucou</nick><password>secret</password></conference></item>".parse().unwrap();
123        let item = PubSubItem::try_from(elem).unwrap();
124        let payload = item.payload.clone().unwrap();
125        println!("FOO: payload: {:?}", payload);
126        // let conference = Conference::try_from(payload).unwrap();
127        let conference = Conference::try_from(payload).unwrap();
128        println!("FOO: conference: {:?}", conference);
129        assert_eq!(conference.autojoin, true);
130        assert_eq!(conference.name, Some(String::from("Test MUC")));
131        assert_eq!(conference.clone().nick.unwrap().as_str(), "Coucou");
132        assert_eq!(conference.clone().password.unwrap(), "secret");
133
134        let elem: Element = "<event xmlns='http://jabber.org/protocol/pubsub#event'><items node='urn:xmpp:bookmarks:1'><item xmlns='http://jabber.org/protocol/pubsub#event' id='test-muc@muc.localhost'><conference xmlns='urn:xmpp:bookmarks:1' autojoin='true' name='Test MUC'><nick>Coucou</nick><password>secret</password></conference></item></items></event>".parse().unwrap();
135        let event = pubsub::Event::try_from(elem).unwrap();
136        let mut items = match event.payload {
137            pubsub::event::Payload::Items {
138                node,
139                published,
140                retracted,
141            } => {
142                assert_eq!(&node.0, ns::BOOKMARKS2);
143                assert_eq!(retracted.len(), 0);
144                published
145            }
146            _ => panic!(),
147        };
148        assert_eq!(items.len(), 1);
149        let item = items.pop().unwrap();
150        let payload = item.payload.clone().unwrap();
151        let conference = Conference::try_from(payload).unwrap();
152        assert_eq!(conference.autojoin, true);
153        assert_eq!(conference.name, Some(String::from("Test MUC")));
154        assert_eq!(conference.clone().nick.unwrap().as_str(), "Coucou");
155        assert_eq!(conference.clone().password.unwrap(), "secret");
156    }
157}