2017-04-21 00:06:30 +00:00
|
|
|
//! A crate parsing common XMPP elements into Rust structures.
|
|
|
|
//!
|
2017-07-29 04:24:20 +00:00
|
|
|
//! Each module implements the [`TryFrom<Element>`] trait, which takes a
|
|
|
|
//! minidom [`Element`] and returns a `Result` whose value is `Ok` if the
|
2017-07-20 22:46:44 +00:00
|
|
|
//! element parsed correctly, `Err(error::Error)` otherwise.
|
2017-04-21 00:06:30 +00:00
|
|
|
//!
|
2017-07-20 22:46:44 +00:00
|
|
|
//! The returned structure can be manipuled as any Rust structure, with each
|
|
|
|
//! field being public. You can also create the same structure manually, with
|
|
|
|
//! some having `new()` and `with_*()` helper methods to create them.
|
|
|
|
//!
|
|
|
|
//! Once you are happy with your structure, you can serialise it back to an
|
|
|
|
//! [`Element`], using either `From` or `Into<Element>`, which give you what
|
|
|
|
//! you want to be sending on the wire.
|
|
|
|
//!
|
|
|
|
//! [`TryFrom<Element>`]: ../try_from/trait.TryFrom.html
|
|
|
|
//! [`Element`]: ../minidom/element/struct.Element.html
|
2017-04-21 00:06:30 +00:00
|
|
|
|
2017-04-29 21:14:34 +00:00
|
|
|
// Copyright (c) 2017 Emmanuel Gil Peyrot <linkmauve@linkmauve.fr>
|
2017-05-30 21:02:56 +00:00
|
|
|
// Copyright (c) 2017 Maxime “pep” Buquet <pep+code@bouah.net>
|
2017-04-29 21:14:34 +00:00
|
|
|
//
|
|
|
|
// 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/.
|
|
|
|
|
2017-04-18 19:44:36 +00:00
|
|
|
extern crate minidom;
|
2017-04-23 14:13:03 +00:00
|
|
|
extern crate jid;
|
2017-04-21 01:46:41 +00:00
|
|
|
extern crate base64;
|
2017-04-23 18:28:03 +00:00
|
|
|
extern crate digest;
|
2017-05-25 01:34:03 +00:00
|
|
|
extern crate sha_1;
|
2017-04-29 00:31:39 +00:00
|
|
|
extern crate sha2;
|
|
|
|
extern crate sha3;
|
|
|
|
extern crate blake2;
|
2017-05-27 11:20:19 +00:00
|
|
|
extern crate chrono;
|
2017-07-20 19:03:15 +00:00
|
|
|
extern crate try_from;
|
2017-04-18 19:44:36 +00:00
|
|
|
|
2017-05-21 15:03:17 +00:00
|
|
|
macro_rules! get_attr {
|
2017-05-21 15:08:25 +00:00
|
|
|
($elem:ident, $attr:tt, $type:tt) => (
|
|
|
|
get_attr!($elem, $attr, $type, value, value.parse()?)
|
|
|
|
);
|
2017-05-21 15:03:17 +00:00
|
|
|
($elem:ident, $attr:tt, optional, $value:ident, $func:expr) => (
|
|
|
|
match $elem.attr($attr) {
|
|
|
|
Some($value) => Some($func),
|
|
|
|
None => None,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
($elem:ident, $attr:tt, required, $value:ident, $func:expr) => (
|
|
|
|
match $elem.attr($attr) {
|
|
|
|
Some($value) => $func,
|
|
|
|
None => return Err(Error::ParseError(concat!("Required attribute '", $attr, "' missing."))),
|
|
|
|
}
|
|
|
|
);
|
|
|
|
($elem:ident, $attr:tt, default, $value:ident, $func:expr) => (
|
|
|
|
match $elem.attr($attr) {
|
|
|
|
Some($value) => $func,
|
|
|
|
None => Default::default(),
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-06-13 23:50:57 +00:00
|
|
|
macro_rules! generate_attribute {
|
|
|
|
($elem:ident, $name:tt, {$($a:ident => $b:tt),+,}) => (
|
|
|
|
generate_attribute!($elem, $name, {$($a => $b),+});
|
|
|
|
);
|
|
|
|
($elem:ident, $name:tt, {$($a:ident => $b:tt),+,}, Default = $default:ident) => (
|
|
|
|
generate_attribute!($elem, $name, {$($a => $b),+}, Default = $default);
|
|
|
|
);
|
|
|
|
($elem:ident, $name:tt, {$($a:ident => $b:tt),+}) => (
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum $elem {
|
2017-07-20 22:10:13 +00:00
|
|
|
$(
|
|
|
|
#[doc=$b]
|
|
|
|
#[doc="value for this attribute."]
|
|
|
|
$a
|
|
|
|
),+
|
2017-06-13 23:50:57 +00:00
|
|
|
}
|
|
|
|
impl FromStr for $elem {
|
|
|
|
type Err = Error;
|
|
|
|
fn from_str(s: &str) -> Result<$elem, Error> {
|
|
|
|
Ok(match s {
|
|
|
|
$($b => $elem::$a),+,
|
|
|
|
_ => return Err(Error::ParseError(concat!("Unknown value for '", $name, "' attribute."))),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl IntoAttributeValue for $elem {
|
|
|
|
fn into_attribute_value(self) -> Option<String> {
|
|
|
|
Some(String::from(match self {
|
|
|
|
$($elem::$a => $b),+
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
($elem:ident, $name:tt, {$($a:ident => $b:tt),+}, Default = $default:ident) => (
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
|
|
pub enum $elem {
|
2017-07-20 22:10:13 +00:00
|
|
|
$(
|
|
|
|
#[doc=$b]
|
|
|
|
#[doc="value for this attribute."]
|
|
|
|
$a
|
|
|
|
),+
|
2017-06-13 23:50:57 +00:00
|
|
|
}
|
|
|
|
impl FromStr for $elem {
|
|
|
|
type Err = Error;
|
|
|
|
fn from_str(s: &str) -> Result<$elem, Error> {
|
|
|
|
Ok(match s {
|
|
|
|
$($b => $elem::$a),+,
|
|
|
|
_ => return Err(Error::ParseError(concat!("Unknown value for '", $name, "' attribute."))),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl IntoAttributeValue for $elem {
|
|
|
|
#[allow(unreachable_patterns)]
|
|
|
|
fn into_attribute_value(self) -> Option<String> {
|
|
|
|
Some(String::from(match self {
|
|
|
|
$elem::$default => return None,
|
|
|
|
$($elem::$a => $b),+
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl Default for $elem {
|
|
|
|
fn default() -> $elem {
|
|
|
|
$elem::$default
|
|
|
|
}
|
2017-06-25 21:14:21 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-10-10 16:52:14 +00:00
|
|
|
macro_rules! check_self {
|
|
|
|
($elem:ident, $name:tt, $ns:expr) => (
|
|
|
|
if !$elem.is($name, $ns) {
|
|
|
|
return Err(Error::ParseError(concat!("This is not a ", $name, " element.")));
|
|
|
|
}
|
|
|
|
);
|
|
|
|
($elem:ident, $name:tt, $ns:expr, $pretty_name:tt) => (
|
|
|
|
if !$elem.is($name, $ns) {
|
|
|
|
return Err(Error::ParseError(concat!("This is not a ", $pretty_name, " element.")));
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-10-10 17:09:58 +00:00
|
|
|
macro_rules! check_ns_only {
|
|
|
|
($elem:ident, $name:tt, $ns:expr) => (
|
|
|
|
if !$elem.has_ns($ns) {
|
|
|
|
return Err(Error::ParseError(concat!("This is not a ", $name, " element.")));
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-10-10 16:52:14 +00:00
|
|
|
macro_rules! check_no_children {
|
|
|
|
($elem:ident, $name:tt) => (
|
|
|
|
for _ in $elem.children() {
|
|
|
|
return Err(Error::ParseError(concat!("Unknown child in ", $name, " element.")));
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! check_no_unknown_attributes {
|
|
|
|
($elem:ident, $name:tt, [$($attr:tt),*]) => (
|
|
|
|
for (_attr, _) in $elem.attrs() {
|
|
|
|
$(
|
|
|
|
if _attr == $attr {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
)*
|
|
|
|
return Err(Error::ParseError(concat!("Unknown attribute in ", $name, " element.")));
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-10-10 16:53:25 +00:00
|
|
|
macro_rules! generate_empty_element {
|
|
|
|
($elem:ident, $name:tt, $ns:expr) => (
|
|
|
|
// TODO: Find a better way to concatenate doc.
|
|
|
|
#[doc="Structure representing a "]
|
|
|
|
#[doc=$name]
|
|
|
|
#[doc=" element."]
|
|
|
|
#[derive(Debug, Clone)]
|
|
|
|
pub struct $elem;
|
|
|
|
|
|
|
|
impl TryFrom<Element> for $elem {
|
|
|
|
type Err = Error;
|
|
|
|
|
|
|
|
fn try_from(elem: Element) -> Result<$elem, Error> {
|
|
|
|
check_self!(elem, $name, $ns);
|
|
|
|
check_no_children!(elem, $name);
|
|
|
|
check_no_unknown_attributes!(elem, $name, []);
|
|
|
|
Ok($elem)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<$elem> for Element {
|
|
|
|
fn from(_: $elem) -> Element {
|
|
|
|
Element::builder("attention")
|
|
|
|
.ns($ns)
|
|
|
|
.build()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-06-25 21:14:21 +00:00
|
|
|
macro_rules! generate_id {
|
|
|
|
($elem:ident) => (
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
2017-07-15 10:38:44 +00:00
|
|
|
pub struct $elem(pub String);
|
2017-06-25 21:14:21 +00:00
|
|
|
impl FromStr for $elem {
|
|
|
|
type Err = Error;
|
|
|
|
fn from_str(s: &str) -> Result<$elem, Error> {
|
|
|
|
// TODO: add a way to parse that differently when needed.
|
|
|
|
Ok($elem(String::from(s)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl IntoAttributeValue for $elem {
|
|
|
|
fn into_attribute_value(self) -> Option<String> {
|
|
|
|
Some(self.0)
|
|
|
|
}
|
2017-06-13 23:50:57 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-07-29 04:36:59 +00:00
|
|
|
macro_rules! generate_elem_id {
|
|
|
|
($elem:ident, $name:tt, $ns:expr) => (
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
|
|
|
pub struct $elem(pub String);
|
|
|
|
impl FromStr for $elem {
|
|
|
|
type Err = Error;
|
|
|
|
fn from_str(s: &str) -> Result<$elem, Error> {
|
|
|
|
// TODO: add a way to parse that differently when needed.
|
|
|
|
Ok($elem(String::from(s)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl From<$elem> for Element {
|
|
|
|
fn from(elem: $elem) -> Element {
|
|
|
|
Element::builder($name)
|
|
|
|
.ns($ns)
|
|
|
|
.append(elem.0)
|
|
|
|
.build()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-04-21 00:06:30 +00:00
|
|
|
/// Error type returned by every parser on failure.
|
2017-04-18 19:44:36 +00:00
|
|
|
pub mod error;
|
2017-04-21 00:06:30 +00:00
|
|
|
/// XML namespace definitions used through XMPP.
|
2017-04-18 19:44:36 +00:00
|
|
|
pub mod ns;
|
|
|
|
|
2017-08-18 23:04:18 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
/// Namespace-aware comparison for tests
|
|
|
|
mod compare_elements;
|
|
|
|
|
2017-04-23 14:13:03 +00:00
|
|
|
/// RFC 6120: Extensible Messaging and Presence Protocol (XMPP): Core
|
|
|
|
pub mod message;
|
2017-04-21 00:06:30 +00:00
|
|
|
/// RFC 6120: Extensible Messaging and Presence Protocol (XMPP): Core
|
2017-04-23 16:30:23 +00:00
|
|
|
pub mod presence;
|
2017-04-23 19:38:13 +00:00
|
|
|
/// RFC 6120: Extensible Messaging and Presence Protocol (XMPP): Core
|
|
|
|
pub mod iq;
|
2017-05-01 00:24:45 +00:00
|
|
|
/// RFC 6120: Extensible Messaging and Presence Protocol (XMPP): Core
|
|
|
|
pub mod stanza_error;
|
2017-04-23 18:28:25 +00:00
|
|
|
|
2017-05-28 15:30:43 +00:00
|
|
|
/// RFC 6121: Extensible Messaging and Presence Protocol (XMPP): Instant Messaging and Presence
|
|
|
|
pub mod roster;
|
|
|
|
|
2017-04-21 00:06:30 +00:00
|
|
|
/// XEP-0004: Data Forms
|
2017-04-18 19:44:36 +00:00
|
|
|
pub mod data_forms;
|
2017-04-21 00:06:30 +00:00
|
|
|
|
|
|
|
/// XEP-0030: Service Discovery
|
|
|
|
pub mod disco;
|
|
|
|
|
2017-05-30 21:02:56 +00:00
|
|
|
/// XEP-0045: Multi-User Chat
|
|
|
|
pub mod muc;
|
|
|
|
|
2017-04-21 00:06:30 +00:00
|
|
|
/// XEP-0047: In-Band Bytestreams
|
2017-04-19 20:52:14 +00:00
|
|
|
pub mod ibb;
|
2017-04-21 00:06:30 +00:00
|
|
|
|
2017-04-29 03:37:18 +00:00
|
|
|
/// XEP-0059: Result Set Management
|
|
|
|
pub mod rsm;
|
|
|
|
|
2017-06-11 13:48:31 +00:00
|
|
|
/// XEP-0060: Publish-Subscribe
|
|
|
|
pub mod pubsub;
|
|
|
|
|
2017-06-25 20:03:48 +00:00
|
|
|
/// XEP-0077: In-Band Registration
|
|
|
|
pub mod ibr;
|
|
|
|
|
2017-04-21 00:06:30 +00:00
|
|
|
/// XEP-0085: Chat State Notifications
|
|
|
|
pub mod chatstates;
|
|
|
|
|
2017-07-29 10:45:45 +00:00
|
|
|
/// XEP-0092: Software Version
|
|
|
|
pub mod version;
|
|
|
|
|
2017-05-25 01:34:03 +00:00
|
|
|
/// XEP-0115: Entity Capabilities
|
|
|
|
pub mod caps;
|
|
|
|
|
2017-04-21 00:06:30 +00:00
|
|
|
/// XEP-0166: Jingle
|
|
|
|
pub mod jingle;
|
|
|
|
|
|
|
|
/// XEP-0184: Message Delivery Receipts
|
2017-04-19 23:43:33 +00:00
|
|
|
pub mod receipts;
|
2017-04-19 22:21:53 +00:00
|
|
|
|
2017-04-21 00:06:30 +00:00
|
|
|
/// XEP-0199: XMPP Ping
|
|
|
|
pub mod ping;
|
|
|
|
|
2017-04-21 02:57:34 +00:00
|
|
|
/// XEP-0203: Delayed Delivery
|
|
|
|
pub mod delay;
|
|
|
|
|
2017-04-21 00:06:30 +00:00
|
|
|
/// XEP-0221: Data Forms Media Element
|
|
|
|
pub mod media_element;
|
|
|
|
|
2017-04-21 00:53:47 +00:00
|
|
|
/// XEP-0224: Attention
|
|
|
|
pub mod attention;
|
|
|
|
|
2017-04-22 16:39:21 +00:00
|
|
|
/// XEP-0234: Jingle File Transfer
|
|
|
|
pub mod jingle_ft;
|
|
|
|
|
2017-05-06 11:49:30 +00:00
|
|
|
/// XEP-0260: Jingle SOCKS5 Bytestreams Transport Method
|
|
|
|
pub mod jingle_s5b;
|
|
|
|
|
2017-04-22 18:15:48 +00:00
|
|
|
/// XEP-0261: Jingle In-Band Bytestreams Transport Method
|
|
|
|
pub mod jingle_ibb;
|
|
|
|
|
2017-04-29 02:50:49 +00:00
|
|
|
/// XEP-0297: Stanza Forwarding
|
|
|
|
pub mod forwarding;
|
|
|
|
|
2017-04-21 03:21:16 +00:00
|
|
|
/// XEP-0300: Use of Cryptographic Hash Functions in XMPP
|
|
|
|
pub mod hashes;
|
|
|
|
|
2017-04-21 01:07:44 +00:00
|
|
|
/// XEP-0308: Last Message Correction
|
|
|
|
pub mod message_correct;
|
|
|
|
|
2017-04-29 05:07:00 +00:00
|
|
|
/// XEP-0313: Message Archive Management
|
|
|
|
pub mod mam;
|
|
|
|
|
2017-05-21 19:22:48 +00:00
|
|
|
/// XEP-0319: Last User Interaction in Presence
|
|
|
|
pub mod idle;
|
|
|
|
|
2017-07-15 10:38:44 +00:00
|
|
|
/// XEP-0353: Jingle Message Initiation
|
|
|
|
pub mod jingle_message;
|
|
|
|
|
2017-04-29 02:23:50 +00:00
|
|
|
/// XEP-0359: Unique and Stable Stanza IDs
|
|
|
|
pub mod stanza_id;
|
|
|
|
|
2017-04-21 02:45:05 +00:00
|
|
|
/// XEP-0380: Explicit Message Encryption
|
|
|
|
pub mod eme;
|
|
|
|
|
2017-04-21 00:06:30 +00:00
|
|
|
/// XEP-0390: Entity Capabilities 2.0
|
|
|
|
pub mod ecaps2;
|