xmpp-rs/xmpp-parsers/src/mam.rs

457 lines
14 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/.
use crate::data_forms::DataForm;
use crate::forwarding::Forwarded;
2018-12-18 14:32:05 +00:00
use crate::iq::{IqGetPayload, IqResultPayload, IqSetPayload};
use crate::message::MessagePayload;
use crate::ns;
2018-12-18 14:32:05 +00:00
use crate::pubsub::NodeName;
use crate::rsm::{SetQuery, SetResult};
use crate::util::error::Error;
2018-12-18 14:32:05 +00:00
use jid::Jid;
use minidom::{Element, Node};
use std::convert::TryFrom;
2017-04-29 05:07:00 +00:00
generate_id!(
/// An identifier matching a result message to the query requesting it.
QueryId
);
generate_element!(
2018-08-02 18:05:51 +00:00
/// Starts a query to the archive.
Query, "query", MAM,
attributes: [
2018-08-02 18:05:51 +00:00
/// An optional identifier for matching forwarded messages to this
/// query.
queryid: Option<QueryId> = "queryid",
2018-08-02 18:05:51 +00:00
/// Must be set to Some when querying a PubSub nodes archive.
node: Option<NodeName> = "node"
],
children: [
2018-08-02 18:05:51 +00:00
/// Used for filtering the results.
form: Option<DataForm> = ("x", DATA_FORMS) => DataForm,
2018-08-02 18:05:51 +00:00
/// Used for paging through results.
set: Option<SetQuery> = ("set", RSM) => SetQuery
]
);
2017-04-29 05:07:00 +00:00
impl IqGetPayload for Query {}
impl IqSetPayload for Query {}
impl IqResultPayload for Query {}
generate_element!(
2018-08-02 18:05:51 +00:00
/// The wrapper around forwarded stanzas.
2018-05-15 00:06:38 +00:00
Result_, "result", MAM,
attributes: [
2018-08-02 18:05:51 +00:00
/// The stanza-id under which the archive stored this stanza.
id: Required<String> = "id",
2018-08-02 18:05:51 +00:00
/// The same queryid as the one requested in the
/// [query](struct.Query.html).
queryid: Option<QueryId> = "queryid",
2018-05-15 00:06:38 +00:00
],
children: [
2018-08-02 18:05:51 +00:00
/// The actual stanza being forwarded.
forwarded: Required<Forwarded> = ("forwarded", FORWARD) => Forwarded
]
2018-05-15 00:06:38 +00:00
);
2017-04-29 05:07:00 +00:00
impl MessagePayload for Result_ {}
generate_attribute!(
2018-08-02 18:05:51 +00:00
/// True when the end of a MAM query has been reached.
2018-12-18 14:32:05 +00:00
Complete,
"complete",
bool
);
generate_element!(
2018-08-02 18:05:51 +00:00
/// Notes the end of a page in a query.
Fin, "fin", MAM,
attributes: [
2018-08-02 18:05:51 +00:00
/// True when the end of a MAM query has been reached.
complete: Default<Complete> = "complete",
],
children: [
2018-08-02 18:05:51 +00:00
/// Describes the current page, it should contain at least [first]
/// (with an [index]) and [last], and generally [count].
///
/// [first]: ../rsm/struct.SetResult.html#structfield.first
/// [index]: ../rsm/struct.SetResult.html#structfield.first_index
/// [last]: ../rsm/struct.SetResult.html#structfield.last
/// [count]: ../rsm/struct.SetResult.html#structfield.count
set: Required<SetResult> = ("set", RSM) => SetResult
]
);
2017-04-29 05:07:00 +00:00
impl IqResultPayload for Fin {}
2018-08-02 18:05:51 +00:00
generate_attribute!(
/// Notes the default archiving preference for the user.
DefaultPrefs, "default", {
/// The default is to always log messages in the archive.
Always => "always",
/// The default is to never log messages in the archive.
Never => "never",
/// The default is to log messages in the archive only for contacts
/// present in the users [roster](../roster/index.html).
Roster => "roster",
}
);
2018-08-02 18:05:51 +00:00
/// Controls the archiving preferences of the user.
2017-04-29 05:07:00 +00:00
#[derive(Debug, Clone)]
pub struct Prefs {
2018-08-02 18:05:51 +00:00
/// The default preference for JIDs in neither
/// [always](#structfield.always) or [never](#structfield.never) lists.
pub default_: DefaultPrefs,
2018-08-02 18:05:51 +00:00
/// The set of JIDs for which to always store messages in the archive.
2017-04-29 05:07:00 +00:00
pub always: Vec<Jid>,
2018-08-02 18:05:51 +00:00
/// The set of JIDs for which to never store messages in the archive.
2017-04-29 05:07:00 +00:00
pub never: Vec<Jid>,
}
impl IqGetPayload for Prefs {}
impl IqSetPayload for Prefs {}
impl IqResultPayload for Prefs {}
impl TryFrom<Element> for Prefs {
type Error = Error;
2017-05-06 20:08:44 +00:00
fn try_from(elem: Element) -> Result<Prefs, Error> {
2018-05-14 14:30:28 +00:00
check_self!(elem, "prefs", MAM);
2017-10-31 21:10:04 +00:00
check_no_unknown_attributes!(elem, "prefs", ["default"]);
2018-12-18 14:32:05 +00:00
let mut always = vec![];
let mut never = vec![];
2017-05-06 20:08:44 +00:00
for child in elem.children() {
if child.is("always", ns::MAM) {
for jid_elem in child.children() {
if !jid_elem.is("jid", ns::MAM) {
return Err(Error::ParseError("Invalid jid element in always."));
}
always.push(jid_elem.text().parse()?);
2017-04-29 05:07:00 +00:00
}
2017-05-06 20:08:44 +00:00
} else if child.is("never", ns::MAM) {
for jid_elem in child.children() {
if !jid_elem.is("jid", ns::MAM) {
return Err(Error::ParseError("Invalid jid element in never."));
}
never.push(jid_elem.text().parse()?);
2017-04-29 05:07:00 +00:00
}
2017-05-06 20:08:44 +00:00
} else {
return Err(Error::ParseError("Unknown child in prefs element."));
2017-04-29 05:07:00 +00:00
}
}
let default_ = get_attr!(elem, "default", Required);
2018-12-18 14:32:05 +00:00
Ok(Prefs {
default_,
always,
never,
})
2017-04-29 05:07:00 +00:00
}
}
fn serialise_jid_list(name: &str, jids: Vec<Jid>) -> ::std::option::IntoIter<Node> {
2017-10-31 21:10:04 +00:00
if jids.is_empty() {
None.into_iter()
2017-10-31 21:10:04 +00:00
} else {
2018-12-18 14:32:05 +00:00
Some(
Element::builder(name)
.ns(ns::MAM)
.append_all(jids.into_iter().map(|jid| {
Element::builder("jid")
.ns(ns::MAM)
.append(String::from(jid))
}))
.into(),
)
.into_iter()
2017-05-06 20:08:44 +00:00
}
2017-04-29 05:07:00 +00:00
}
impl From<Prefs> for Element {
fn from(prefs: Prefs) -> Element {
2017-10-31 21:10:04 +00:00
Element::builder("prefs")
2018-12-18 14:32:05 +00:00
.ns(ns::MAM)
.attr("default", prefs.default_)
.append_all(serialise_jid_list("always", prefs.always))
.append_all(serialise_jid_list("never", prefs.never))
2018-12-18 14:32:05 +00:00
.build()
2017-04-29 05:07:00 +00:00
}
}
#[cfg(test)]
mod tests {
2017-05-06 20:08:44 +00:00
use super::*;
use crate::data_forms::{DataFormType, Field, FieldType};
use std::str::FromStr;
2017-04-29 05:07:00 +00:00
2018-10-28 12:10:48 +00:00
#[cfg(target_pointer_width = "32")]
#[test]
fn test_size() {
assert_size!(QueryId, 12);
assert_size!(Query, 116);
assert_size!(Result_, 236);
2018-10-28 12:10:48 +00:00
assert_size!(Complete, 1);
assert_size!(Fin, 44);
assert_size!(DefaultPrefs, 1);
assert_size!(Prefs, 28);
}
#[cfg(target_pointer_width = "64")]
#[test]
fn test_size() {
assert_size!(QueryId, 24);
assert_size!(Query, 232);
assert_size!(Result_, 456);
assert_size!(Complete, 1);
assert_size!(Fin, 88);
assert_size!(DefaultPrefs, 1);
assert_size!(Prefs, 56);
}
2017-04-29 05:07:00 +00:00
#[test]
2017-04-29 05:41:55 +00:00
fn test_query() {
let elem: Element = "<query xmlns='urn:xmpp:mam:2'/>".parse().unwrap();
Query::try_from(elem).unwrap();
2017-04-29 05:07:00 +00:00
}
#[test]
2017-04-29 05:41:55 +00:00
fn test_result() {
#[cfg(not(feature = "component"))]
2017-04-29 05:41:55 +00:00
let elem: Element = r#"
<result xmlns='urn:xmpp:mam:2' queryid='f27' id='28482-98726-73623'>
<forwarded xmlns='urn:xmpp:forward:0'>
<delay xmlns='urn:xmpp:delay' stamp='2010-07-10T23:08:25Z'/>
<message xmlns='jabber:client' from="witch@shakespeare.lit" to="macbeth@shakespeare.lit">
<body>Hail to thee</body>
</message>
</forwarded>
</result>
2018-12-18 14:32:05 +00:00
"#
.parse()
.unwrap();
#[cfg(feature = "component")]
let elem: Element = r#"
<result xmlns='urn:xmpp:mam:2' queryid='f27' id='28482-98726-73623'>
<forwarded xmlns='urn:xmpp:forward:0'>
<delay xmlns='urn:xmpp:delay' stamp='2010-07-10T23:08:25Z'/>
<message xmlns='jabber:component:accept' from="witch@shakespeare.lit" to="macbeth@shakespeare.lit">
<body>Hail to thee</body>
</message>
</forwarded>
</result>
2017-04-29 05:41:55 +00:00
"#.parse().unwrap();
Result_::try_from(elem).unwrap();
2017-04-29 05:41:55 +00:00
}
#[test]
fn test_fin() {
let elem: Element = r#"
<fin xmlns='urn:xmpp:mam:2'>
<set xmlns='http://jabber.org/protocol/rsm'>
<first index='0'>28482-98726-73623</first>
<last>09af3-cc343-b409f</last>
</set>
</fin>
2018-12-18 14:32:05 +00:00
"#
.parse()
.unwrap();
Fin::try_from(elem).unwrap();
2017-04-29 05:41:55 +00:00
}
#[test]
fn test_query_x() {
let elem: Element = r#"
<query xmlns='urn:xmpp:mam:2'>
<x xmlns='jabber:x:data' type='submit'>
<field var='FORM_TYPE' type='hidden'>
<value>urn:xmpp:mam:2</value>
</field>
<field var='with'>
<value>juliet@capulet.lit</value>
</field>
</x>
</query>
2018-12-18 14:32:05 +00:00
"#
.parse()
.unwrap();
Query::try_from(elem).unwrap();
2017-04-29 05:07:00 +00:00
}
#[test]
2017-04-29 05:41:55 +00:00
fn test_query_x_set() {
let elem: Element = r#"
<query xmlns='urn:xmpp:mam:2'>
<x xmlns='jabber:x:data' type='submit'>
<field var='FORM_TYPE' type='hidden'>
<value>urn:xmpp:mam:2</value>
</field>
<field var='start'>
<value>2010-08-07T00:00:00Z</value>
</field>
</x>
<set xmlns='http://jabber.org/protocol/rsm'>
<max>10</max>
</set>
</query>
2018-12-18 14:32:05 +00:00
"#
.parse()
.unwrap();
Query::try_from(elem).unwrap();
2017-04-29 05:41:55 +00:00
}
#[test]
fn test_prefs_get() {
2018-12-18 14:32:05 +00:00
let elem: Element = "<prefs xmlns='urn:xmpp:mam:2' default='always'/>"
.parse()
.unwrap();
2017-10-31 21:10:04 +00:00
let prefs = Prefs::try_from(elem).unwrap();
assert_eq!(prefs.always, vec!());
assert_eq!(prefs.never, vec!());
2017-04-29 05:41:55 +00:00
let elem: Element = r#"
<prefs xmlns='urn:xmpp:mam:2' default='roster'>
<always/>
<never/>
</prefs>
2018-12-18 14:32:05 +00:00
"#
.parse()
.unwrap();
2017-10-31 21:10:04 +00:00
let prefs = Prefs::try_from(elem).unwrap();
assert_eq!(prefs.always, vec!());
assert_eq!(prefs.never, vec!());
2017-04-29 05:41:55 +00:00
}
#[test]
fn test_prefs_result() {
let elem: Element = r#"
<prefs xmlns='urn:xmpp:mam:2' default='roster'>
<always>
<jid>romeo@montague.lit</jid>
</always>
<never>
<jid>montague@montague.lit</jid>
</never>
</prefs>
2018-12-18 14:32:05 +00:00
"#
.parse()
.unwrap();
2017-10-31 21:10:04 +00:00
let prefs = Prefs::try_from(elem).unwrap();
2018-12-18 14:32:05 +00:00
assert_eq!(
prefs.always,
vec!(Jid::from_str("romeo@montague.lit").unwrap())
);
assert_eq!(
prefs.never,
vec!(Jid::from_str("montague@montague.lit").unwrap())
);
2017-10-31 21:10:04 +00:00
let elem2 = Element::from(prefs.clone());
println!("{:?}", elem2);
let prefs2 = Prefs::try_from(elem2).unwrap();
assert_eq!(prefs.default_, prefs2.default_);
assert_eq!(prefs.always, prefs2.always);
assert_eq!(prefs.never, prefs2.never);
2017-04-29 05:41:55 +00:00
}
#[test]
fn test_invalid_child() {
2018-12-18 14:32:05 +00:00
let elem: Element = "<query xmlns='urn:xmpp:mam:2'><coucou/></query>"
.parse()
.unwrap();
let error = Query::try_from(elem).unwrap_err();
2017-04-29 05:07:00 +00:00
let message = match error {
Error::ParseError(string) => string,
_ => panic!(),
};
2017-04-29 05:41:55 +00:00
assert_eq!(message, "Unknown child in query element.");
2017-04-29 05:07:00 +00:00
}
#[test]
fn test_serialise_empty() {
2017-04-29 05:41:55 +00:00
let elem: Element = "<query xmlns='urn:xmpp:mam:2'/>".parse().unwrap();
2018-12-18 14:32:05 +00:00
let replace = Query {
queryid: None,
node: None,
form: None,
set: None,
};
let elem2 = replace.into();
2017-04-29 05:07:00 +00:00
assert_eq!(elem, elem2);
}
#[test]
fn test_serialize_query_with_form() {
2019-11-27 17:16:51 +00:00
let reference: Element = "<query xmlns='urn:xmpp:mam:2'><x xmlns='jabber:x:data' type='submit'><field xmlns='jabber:x:data' var='FORM_TYPE' type='hidden'><value xmlns='jabber:x:data'>urn:xmpp:mam:2</value></field><field xmlns='jabber:x:data' var='with'><value xmlns='jabber:x:data'>juliet@capulet.lit</value></field></x></query>"
.parse()
.unwrap();
let elem: Element = "<x xmlns='jabber:x:data' type='submit'><field xmlns='jabber:x:data' var='FORM_TYPE' type='hidden'><value xmlns='jabber:x:data'>urn:xmpp:mam:2</value></field><field xmlns='jabber:x:data' var='with'><value xmlns='jabber:x:data'>juliet@capulet.lit</value></field></x>"
.parse()
.unwrap();
let form = DataForm::try_from(elem).unwrap();
let query = Query {
queryid: None,
node: None,
set: None,
form: Some(form),
};
let serialized: Element = query.into();
2019-11-27 17:16:51 +00:00
assert_eq!(serialized, reference);
}
#[test]
fn test_serialize_result() {
let reference: Element = "<result xmlns='urn:xmpp:mam:2' queryid='f27' id='28482-98726-73623'><forwarded xmlns='urn:xmpp:forward:0'><delay xmlns='urn:xmpp:delay' stamp='2002-09-10T23:08:25+00:00'/><message xmlns='jabber:client' to='juliet@capulet.example/balcony' from='romeo@montague.example/home'/></forwarded></result>"
.parse()
.unwrap();
let elem: Element = "<forwarded xmlns='urn:xmpp:forward:0'><delay xmlns='urn:xmpp:delay' stamp='2002-09-10T23:08:25+00:00'/><message xmlns='jabber:client' to='juliet@capulet.example/balcony' from='romeo@montague.example/home'/></forwarded>"
.parse()
.unwrap();
let forwarded = Forwarded::try_from(elem).unwrap();
let result = Result_ {
id: String::from("28482-98726-73623"),
queryid: Some(QueryId(String::from("f27"))),
forwarded: forwarded,
};
let serialized: Element = result.into();
assert_eq!(serialized, reference);
}
#[test]
fn test_serialize_fin() {
let reference: Element = "<fin xmlns='urn:xmpp:mam:2'><set xmlns='http://jabber.org/protocol/rsm'><first index='0'>28482-98726-73623</first><last>09af3-cc343-b409f</last></set></fin>"
.parse()
.unwrap();
let elem: Element = "<set xmlns='http://jabber.org/protocol/rsm'><first index='0'>28482-98726-73623</first><last>09af3-cc343-b409f</last></set>"
.parse()
.unwrap();
let set = SetResult::try_from(elem).unwrap();
let fin = Fin {
set: set,
complete: Complete::default(),
};
let serialized: Element = fin.into();
assert_eq!(serialized, reference);
}
2017-04-29 05:07:00 +00:00
}