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/.
|
|
|
|
|
|
2018-12-18 14:27:30 +00:00
|
|
|
|
use crate::iq::IqSetPayload;
|
2020-11-27 19:58:20 +00:00
|
|
|
|
use crate::jingle_grouping::Group;
|
2019-10-12 15:13:42 +00:00
|
|
|
|
use crate::jingle_ibb::Transport as IbbTransport;
|
2019-10-22 23:32:41 +00:00
|
|
|
|
use crate::jingle_ice_udp::Transport as IceUdpTransport;
|
|
|
|
|
use crate::jingle_rtp::Description as RtpDescription;
|
2019-10-12 15:13:42 +00:00
|
|
|
|
use crate::jingle_s5b::Transport as Socks5Transport;
|
2018-12-18 14:32:05 +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;
|
2019-02-28 01:40:04 +00:00
|
|
|
|
use std::collections::BTreeMap;
|
2020-11-27 19:57:01 +00:00
|
|
|
|
use std::fmt;
|
2019-10-22 23:32:41 +00:00
|
|
|
|
use std::str::FromStr;
|
2024-06-21 14:27:43 +00:00
|
|
|
|
use xso::error::{Error, FromElementError};
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
2018-09-20 18:28:50 +00:00
|
|
|
|
generate_attribute!(
|
|
|
|
|
/// The action attribute.
|
|
|
|
|
Action, "action", {
|
|
|
|
|
/// Accept a content-add action received from another party.
|
|
|
|
|
ContentAccept => "content-accept",
|
|
|
|
|
|
|
|
|
|
/// Add one or more new content definitions to the session.
|
|
|
|
|
ContentAdd => "content-add",
|
|
|
|
|
|
|
|
|
|
/// Change the directionality of media sending.
|
|
|
|
|
ContentModify => "content-modify",
|
|
|
|
|
|
|
|
|
|
/// Reject a content-add action received from another party.
|
|
|
|
|
ContentReject => "content-reject",
|
|
|
|
|
|
|
|
|
|
/// Remove one or more content definitions from the session.
|
|
|
|
|
ContentRemove => "content-remove",
|
|
|
|
|
|
|
|
|
|
/// Exchange information about parameters for an application type.
|
|
|
|
|
DescriptionInfo => "description-info",
|
|
|
|
|
|
|
|
|
|
/// Exchange information about security preconditions.
|
|
|
|
|
SecurityInfo => "security-info",
|
|
|
|
|
|
|
|
|
|
/// Definitively accept a session negotiation.
|
|
|
|
|
SessionAccept => "session-accept",
|
|
|
|
|
|
|
|
|
|
/// Send session-level information, such as a ping or a ringing message.
|
|
|
|
|
SessionInfo => "session-info",
|
|
|
|
|
|
|
|
|
|
/// Request negotiation of a new Jingle session.
|
|
|
|
|
SessionInitiate => "session-initiate",
|
|
|
|
|
|
|
|
|
|
/// End an existing session.
|
|
|
|
|
SessionTerminate => "session-terminate",
|
|
|
|
|
|
|
|
|
|
/// Accept a transport-replace action received from another party.
|
|
|
|
|
TransportAccept => "transport-accept",
|
|
|
|
|
|
|
|
|
|
/// Exchange transport candidates.
|
|
|
|
|
TransportInfo => "transport-info",
|
|
|
|
|
|
|
|
|
|
/// Reject a transport-replace action received from another party.
|
|
|
|
|
TransportReject => "transport-reject",
|
|
|
|
|
|
|
|
|
|
/// Redefine a transport method or replace it with a different method.
|
|
|
|
|
TransportReplace => "transport-replace",
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
generate_attribute!(
|
|
|
|
|
/// Which party originally generated the content type.
|
|
|
|
|
Creator, "creator", {
|
|
|
|
|
/// This content was created by the initiator of this session.
|
|
|
|
|
Initiator => "initiator",
|
|
|
|
|
|
|
|
|
|
/// This content was created by the responder of this session.
|
|
|
|
|
Responder => "responder",
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
generate_attribute!(
|
|
|
|
|
/// Which parties in the session will be generating content.
|
|
|
|
|
Senders, "senders", {
|
|
|
|
|
/// Both parties can send for this content.
|
|
|
|
|
Both => "both",
|
|
|
|
|
|
|
|
|
|
/// Only the initiator can send for this content.
|
|
|
|
|
Initiator => "initiator",
|
|
|
|
|
|
|
|
|
|
/// No one can send for this content.
|
|
|
|
|
None => "none",
|
|
|
|
|
|
|
|
|
|
/// Only the responder can send for this content.
|
|
|
|
|
Responder => "responder",
|
|
|
|
|
}, Default = Both
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
generate_attribute!(
|
|
|
|
|
/// How the content definition is to be interpreted by the recipient. The
|
|
|
|
|
/// meaning of this attribute matches the "Content-Disposition" header as
|
|
|
|
|
/// defined in RFC 2183 and applied to SIP by RFC 3261.
|
|
|
|
|
///
|
|
|
|
|
/// Possible values are defined here:
|
2022-12-27 15:31:58 +00:00
|
|
|
|
/// <https://www.iana.org/assignments/cont-disp/cont-disp.xhtml>
|
2018-09-20 18:28:50 +00:00
|
|
|
|
Disposition, "disposition", {
|
|
|
|
|
/// Displayed automatically.
|
|
|
|
|
Inline => "inline",
|
|
|
|
|
|
|
|
|
|
/// User controlled display.
|
|
|
|
|
Attachment => "attachment",
|
|
|
|
|
|
|
|
|
|
/// Process as form response.
|
|
|
|
|
FormData => "form-data",
|
|
|
|
|
|
|
|
|
|
/// Tunneled content to be processed silently.
|
|
|
|
|
Signal => "signal",
|
|
|
|
|
|
|
|
|
|
/// The body is a custom ring tone to alert the user.
|
|
|
|
|
Alert => "alert",
|
|
|
|
|
|
|
|
|
|
/// The body is displayed as an icon to the user.
|
|
|
|
|
Icon => "icon",
|
|
|
|
|
|
|
|
|
|
/// The body should be displayed to the user.
|
|
|
|
|
Render => "render",
|
|
|
|
|
|
|
|
|
|
/// The body contains a list of URIs that indicates the recipients of
|
|
|
|
|
/// the request.
|
|
|
|
|
RecipientListHistory => "recipient-list-history",
|
|
|
|
|
|
|
|
|
|
/// The body describes a communications session, for example, an
|
2022-12-27 15:33:51 +00:00
|
|
|
|
/// [RFC2327](https://www.rfc-editor.org/rfc/rfc2327) SDP body.
|
2018-09-20 18:28:50 +00:00
|
|
|
|
Session => "session",
|
|
|
|
|
|
|
|
|
|
/// Authenticated Identity Body.
|
|
|
|
|
Aib => "aib",
|
|
|
|
|
|
|
|
|
|
/// The body describes an early communications session, for example,
|
2022-12-27 15:33:51 +00:00
|
|
|
|
/// and [RFC2327](https://www.rfc-editor.org/rfc/rfc2327) SDP body.
|
2018-09-20 18:28:50 +00:00
|
|
|
|
EarlySession => "early-session",
|
|
|
|
|
|
|
|
|
|
/// The body includes a list of URIs to which URI-list services are to
|
|
|
|
|
/// be applied.
|
|
|
|
|
RecipientList => "recipient-list",
|
|
|
|
|
|
|
|
|
|
/// The payload of the message carrying this Content-Disposition header
|
|
|
|
|
/// field value is an Instant Message Disposition Notification as
|
|
|
|
|
/// requested in the corresponding Instant Message.
|
|
|
|
|
Notification => "notification",
|
|
|
|
|
|
|
|
|
|
/// The body needs to be handled according to a reference to the body
|
|
|
|
|
/// that is located in the same SIP message as the body.
|
|
|
|
|
ByReference => "by-reference",
|
|
|
|
|
|
|
|
|
|
/// The body contains information associated with an Info Package.
|
|
|
|
|
InfoPackage => "info-package",
|
|
|
|
|
|
|
|
|
|
/// The body describes either metadata about the RS or the reason for
|
|
|
|
|
/// the metadata snapshot request as determined by the MIME value
|
|
|
|
|
/// indicated in the Content-Type.
|
|
|
|
|
RecordingSession => "recording-session",
|
|
|
|
|
}, Default = Session
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
generate_id!(
|
|
|
|
|
/// An unique identifier in a session, referencing a
|
|
|
|
|
/// [struct.Content.html](Content element).
|
|
|
|
|
ContentId
|
|
|
|
|
);
|
2017-07-29 01:47:33 +00:00
|
|
|
|
|
2019-10-18 11:06:08 +00:00
|
|
|
|
/// Enum wrapping all of the various supported descriptions of a Content.
|
2020-11-29 20:17:51 +00:00
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
2019-10-18 11:06:08 +00:00
|
|
|
|
pub enum Description {
|
|
|
|
|
/// Jingle RTP Sessions (XEP-0167) description.
|
|
|
|
|
Rtp(RtpDescription),
|
|
|
|
|
|
|
|
|
|
/// To be used for any description that isn’t known at compile-time.
|
|
|
|
|
Unknown(Element),
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl TryFrom<Element> for Description {
|
|
|
|
|
type Error = Error;
|
|
|
|
|
|
|
|
|
|
fn try_from(elem: Element) -> Result<Description, Error> {
|
|
|
|
|
Ok(if elem.is("description", ns::JINGLE_RTP) {
|
|
|
|
|
Description::Rtp(RtpDescription::try_from(elem)?)
|
|
|
|
|
} else {
|
|
|
|
|
Description::Unknown(elem)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl From<RtpDescription> for Description {
|
|
|
|
|
fn from(desc: RtpDescription) -> Description {
|
|
|
|
|
Description::Rtp(desc)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl From<Description> for Element {
|
|
|
|
|
fn from(desc: Description) -> Element {
|
|
|
|
|
match desc {
|
|
|
|
|
Description::Rtp(desc) => desc.into(),
|
|
|
|
|
Description::Unknown(elem) => elem,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-10-12 15:13:42 +00:00
|
|
|
|
/// Enum wrapping all of the various supported transports of a Content.
|
2020-11-29 20:17:51 +00:00
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
2019-10-12 15:13:42 +00:00
|
|
|
|
pub enum Transport {
|
|
|
|
|
/// Jingle ICE-UDP Bytestreams (XEP-0176) transport.
|
|
|
|
|
IceUdp(IceUdpTransport),
|
|
|
|
|
|
|
|
|
|
/// Jingle In-Band Bytestreams (XEP-0261) transport.
|
|
|
|
|
Ibb(IbbTransport),
|
|
|
|
|
|
|
|
|
|
/// Jingle SOCKS5 Bytestreams (XEP-0260) transport.
|
|
|
|
|
Socks5(Socks5Transport),
|
|
|
|
|
|
|
|
|
|
/// To be used for any transport that isn’t known at compile-time.
|
|
|
|
|
Unknown(Element),
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl TryFrom<Element> for Transport {
|
|
|
|
|
type Error = Error;
|
|
|
|
|
|
|
|
|
|
fn try_from(elem: Element) -> Result<Transport, Error> {
|
|
|
|
|
Ok(if elem.is("transport", ns::JINGLE_ICE_UDP) {
|
|
|
|
|
Transport::IceUdp(IceUdpTransport::try_from(elem)?)
|
|
|
|
|
} else if elem.is("transport", ns::JINGLE_IBB) {
|
|
|
|
|
Transport::Ibb(IbbTransport::try_from(elem)?)
|
|
|
|
|
} else if elem.is("transport", ns::JINGLE_S5B) {
|
|
|
|
|
Transport::Socks5(Socks5Transport::try_from(elem)?)
|
|
|
|
|
} else {
|
|
|
|
|
Transport::Unknown(elem)
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl From<IceUdpTransport> for Transport {
|
|
|
|
|
fn from(transport: IceUdpTransport) -> Transport {
|
|
|
|
|
Transport::IceUdp(transport)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl From<IbbTransport> for Transport {
|
|
|
|
|
fn from(transport: IbbTransport) -> Transport {
|
|
|
|
|
Transport::Ibb(transport)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl From<Socks5Transport> for Transport {
|
|
|
|
|
fn from(transport: Socks5Transport) -> Transport {
|
|
|
|
|
Transport::Socks5(transport)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl From<Transport> for Element {
|
|
|
|
|
fn from(transport: Transport) -> Element {
|
|
|
|
|
match transport {
|
|
|
|
|
Transport::IceUdp(transport) => transport.into(),
|
|
|
|
|
Transport::Ibb(transport) => transport.into(),
|
|
|
|
|
Transport::Socks5(transport) => transport.into(),
|
|
|
|
|
Transport::Unknown(elem) => elem,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-05-28 14:45:13 +00:00
|
|
|
|
generate_element!(
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Describes a session’s content, there can be multiple content in one
|
|
|
|
|
/// session.
|
2018-05-28 14:29:51 +00:00
|
|
|
|
Content, "content", JINGLE,
|
|
|
|
|
attributes: [
|
2018-09-20 18:28:50 +00:00
|
|
|
|
/// Who created this content.
|
2019-02-24 19:26:40 +00:00
|
|
|
|
creator: Required<Creator> = "creator",
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// How the content definition is to be interpreted by the recipient.
|
2019-02-24 19:26:40 +00:00
|
|
|
|
disposition: Default<Disposition> = "disposition",
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// A per-session unique identifier for this content.
|
2019-02-24 19:26:40 +00:00
|
|
|
|
name: Required<ContentId> = "name",
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// Who can send data for this content.
|
2019-02-24 19:26:40 +00:00
|
|
|
|
senders: Default<Senders> = "senders",
|
2018-05-28 14:29:51 +00:00
|
|
|
|
],
|
|
|
|
|
children: [
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// What to send.
|
2019-10-18 11:06:08 +00:00
|
|
|
|
description: Option<Description> = ("description", *) => Description,
|
2018-09-20 19:15:50 +00:00
|
|
|
|
|
|
|
|
|
/// How to send it.
|
2019-10-12 15:13:42 +00:00
|
|
|
|
transport: Option<Transport> = ("transport", *) => Transport,
|
2018-09-20 19:15:50 +00:00
|
|
|
|
|
|
|
|
|
/// With which security.
|
2018-05-28 14:29:51 +00:00
|
|
|
|
security: Option<Element> = ("security", JINGLE) => Element
|
|
|
|
|
]
|
|
|
|
|
);
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
2018-05-04 17:10:29 +00:00
|
|
|
|
impl Content {
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Create a new content.
|
2018-05-04 17:10:29 +00:00
|
|
|
|
pub fn new(creator: Creator, name: ContentId) -> Content {
|
|
|
|
|
Content {
|
|
|
|
|
creator,
|
|
|
|
|
name,
|
|
|
|
|
disposition: Disposition::Session,
|
|
|
|
|
senders: Senders::Both,
|
|
|
|
|
description: None,
|
|
|
|
|
transport: None,
|
|
|
|
|
security: None,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Set how the content is to be interpreted by the recipient.
|
2018-05-04 17:10:29 +00:00
|
|
|
|
pub fn with_disposition(mut self, disposition: Disposition) -> Content {
|
|
|
|
|
self.disposition = disposition;
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Specify who can send data for this content.
|
2018-05-04 17:10:29 +00:00
|
|
|
|
pub fn with_senders(mut self, senders: Senders) -> Content {
|
|
|
|
|
self.senders = senders;
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Set the description of this content.
|
2019-10-18 11:06:08 +00:00
|
|
|
|
pub fn with_description<D: Into<Description>>(mut self, description: D) -> Content {
|
|
|
|
|
self.description = Some(description.into());
|
2018-05-04 17:10:29 +00:00
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Set the transport of this content.
|
2019-10-12 15:13:42 +00:00
|
|
|
|
pub fn with_transport<T: Into<Transport>>(mut self, transport: T) -> Content {
|
|
|
|
|
self.transport = Some(transport.into());
|
2018-05-04 17:10:29 +00:00
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Set the security of this content.
|
2018-05-04 17:10:29 +00:00
|
|
|
|
pub fn with_security(mut self, security: Element) -> Content {
|
|
|
|
|
self.security = Some(security);
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Lists the possible reasons to be included in a Jingle iq.
|
2017-04-20 23:41:15 +00:00
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
2017-04-19 01:27:42 +00:00
|
|
|
|
pub enum Reason {
|
2018-09-20 18:28:50 +00:00
|
|
|
|
/// The party prefers to use an existing session with the peer rather than
|
|
|
|
|
/// initiate a new session; the Jingle session ID of the alternative
|
2022-12-27 15:31:58 +00:00
|
|
|
|
/// session SHOULD be provided as the XML character data of the \<sid/\>
|
2018-09-20 18:28:50 +00:00
|
|
|
|
/// child.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
AlternativeSession, //(String),
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The party is busy and cannot accept a session.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
Busy,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The initiator wishes to formally cancel the session initiation request.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
Cancel,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The action is related to connectivity problems.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
ConnectivityError,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The party wishes to formally decline the session.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
Decline,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The session length has exceeded a pre-defined time limit (e.g., a
|
|
|
|
|
/// meeting hosted at a conference service).
|
2017-04-19 01:27:42 +00:00
|
|
|
|
Expired,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The party has been unable to initialize processing related to the
|
|
|
|
|
/// application type.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
FailedApplication,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The party has been unable to establish connectivity for the transport
|
|
|
|
|
/// method.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
FailedTransport,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The action is related to a non-specific application error.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
GeneralError,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The entity is going offline or is no longer available.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
Gone,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The party supports the offered application type but does not support
|
|
|
|
|
/// the offered or negotiated parameters.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
IncompatibleParameters,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The action is related to media processing problems.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
MediaError,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The action is related to a violation of local security policies.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
SecurityError,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The action is generated during the normal course of state management
|
|
|
|
|
/// and does not reflect any error.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
Success,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// A request has not been answered so the sender is timing out the
|
|
|
|
|
/// request.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
Timeout,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The party supports none of the offered application types.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
UnsupportedApplications,
|
2018-09-20 18:28:50 +00:00
|
|
|
|
|
|
|
|
|
/// The party supports none of the offered transport methods.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
UnsupportedTransports,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl FromStr for Reason {
|
|
|
|
|
type Err = Error;
|
|
|
|
|
|
|
|
|
|
fn from_str(s: &str) -> Result<Reason, Error> {
|
2017-04-23 02:41:26 +00:00
|
|
|
|
Ok(match s {
|
|
|
|
|
"alternative-session" => Reason::AlternativeSession,
|
|
|
|
|
"busy" => Reason::Busy,
|
|
|
|
|
"cancel" => Reason::Cancel,
|
|
|
|
|
"connectivity-error" => Reason::ConnectivityError,
|
|
|
|
|
"decline" => Reason::Decline,
|
|
|
|
|
"expired" => Reason::Expired,
|
|
|
|
|
"failed-application" => Reason::FailedApplication,
|
|
|
|
|
"failed-transport" => Reason::FailedTransport,
|
|
|
|
|
"general-error" => Reason::GeneralError,
|
|
|
|
|
"gone" => Reason::Gone,
|
|
|
|
|
"incompatible-parameters" => Reason::IncompatibleParameters,
|
|
|
|
|
"media-error" => Reason::MediaError,
|
|
|
|
|
"security-error" => Reason::SecurityError,
|
|
|
|
|
"success" => Reason::Success,
|
|
|
|
|
"timeout" => Reason::Timeout,
|
|
|
|
|
"unsupported-applications" => Reason::UnsupportedApplications,
|
|
|
|
|
"unsupported-transports" => Reason::UnsupportedTransports,
|
|
|
|
|
|
2024-06-21 14:27:43 +00:00
|
|
|
|
_ => return Err(Error::Other("Unknown reason.")),
|
2017-04-23 02:41:26 +00:00
|
|
|
|
})
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-20 19:36:13 +00:00
|
|
|
|
impl From<Reason> for Element {
|
|
|
|
|
fn from(reason: Reason) -> Element {
|
2020-04-02 20:45:20 +00:00
|
|
|
|
Element::builder(
|
|
|
|
|
match reason {
|
|
|
|
|
Reason::AlternativeSession => "alternative-session",
|
|
|
|
|
Reason::Busy => "busy",
|
|
|
|
|
Reason::Cancel => "cancel",
|
|
|
|
|
Reason::ConnectivityError => "connectivity-error",
|
|
|
|
|
Reason::Decline => "decline",
|
|
|
|
|
Reason::Expired => "expired",
|
|
|
|
|
Reason::FailedApplication => "failed-application",
|
|
|
|
|
Reason::FailedTransport => "failed-transport",
|
|
|
|
|
Reason::GeneralError => "general-error",
|
|
|
|
|
Reason::Gone => "gone",
|
|
|
|
|
Reason::IncompatibleParameters => "incompatible-parameters",
|
|
|
|
|
Reason::MediaError => "media-error",
|
|
|
|
|
Reason::SecurityError => "security-error",
|
|
|
|
|
Reason::Success => "success",
|
|
|
|
|
Reason::Timeout => "timeout",
|
|
|
|
|
Reason::UnsupportedApplications => "unsupported-applications",
|
|
|
|
|
Reason::UnsupportedTransports => "unsupported-transports",
|
|
|
|
|
},
|
|
|
|
|
ns::JINGLE,
|
|
|
|
|
)
|
2019-10-22 23:32:41 +00:00
|
|
|
|
.build()
|
2017-04-28 22:45:02 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-02-28 01:40:04 +00:00
|
|
|
|
type Lang = String;
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Informs the recipient of something.
|
2020-11-29 20:17:51 +00:00
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
2017-04-19 01:27:42 +00:00
|
|
|
|
pub struct ReasonElement {
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// The list of possible reasons to be included in a Jingle iq.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
pub reason: Reason,
|
2018-09-20 19:15:50 +00:00
|
|
|
|
|
|
|
|
|
/// A human-readable description of this reason.
|
2019-02-28 01:40:04 +00:00
|
|
|
|
pub texts: BTreeMap<Lang, String>,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-11-27 19:57:01 +00:00
|
|
|
|
impl fmt::Display for ReasonElement {
|
|
|
|
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
|
write!(fmt, "{}", Element::from(self.reason.clone()).name())?;
|
|
|
|
|
if let Some(text) = self.texts.get("en") {
|
|
|
|
|
write!(fmt, ": {}", text)?;
|
|
|
|
|
} else if let Some(text) = self.texts.get("") {
|
|
|
|
|
write!(fmt, ": {}", text)?;
|
|
|
|
|
}
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-24 21:28:54 +00:00
|
|
|
|
impl TryFrom<Element> for ReasonElement {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
type Error = FromElementError;
|
2017-05-24 21:28:54 +00:00
|
|
|
|
|
2024-06-21 14:27:43 +00:00
|
|
|
|
fn try_from(elem: Element) -> Result<ReasonElement, FromElementError> {
|
2018-05-14 14:30:28 +00:00
|
|
|
|
check_self!(elem, "reason", JINGLE);
|
2019-02-28 01:40:04 +00:00
|
|
|
|
check_no_attributes!(elem, "reason");
|
2017-05-24 21:28:54 +00:00
|
|
|
|
let mut reason = None;
|
2019-02-28 01:40:04 +00:00
|
|
|
|
let mut texts = BTreeMap::new();
|
2017-05-24 21:28:54 +00:00
|
|
|
|
for child in elem.children() {
|
2019-02-28 01:40:04 +00:00
|
|
|
|
if child.is("text", ns::JINGLE) {
|
|
|
|
|
check_no_children!(child, "text");
|
|
|
|
|
check_no_unknown_attributes!(child, "text", ["xml:lang"]);
|
|
|
|
|
let lang = get_attr!(elem, "xml:lang", Default);
|
|
|
|
|
if texts.insert(lang, child.text()).is_some() {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
return Err(
|
|
|
|
|
Error::Other("Text element present twice for the same xml:lang.").into(),
|
|
|
|
|
);
|
2018-12-18 14:32:05 +00:00
|
|
|
|
}
|
2019-02-28 01:40:04 +00:00
|
|
|
|
} else if child.has_ns(ns::JINGLE) {
|
|
|
|
|
if reason.is_some() {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
return Err(Error::Other("Reason must not have more than one reason.").into());
|
2018-12-18 14:32:05 +00:00
|
|
|
|
}
|
2019-02-28 01:40:04 +00:00
|
|
|
|
check_no_children!(child, "reason");
|
|
|
|
|
check_no_attributes!(child, "reason");
|
|
|
|
|
reason = Some(child.name().parse()?);
|
|
|
|
|
} else {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
return Err(Error::Other("Reason contains a foreign element.").into());
|
2017-05-24 21:28:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-06-21 14:27:43 +00:00
|
|
|
|
let reason = reason.ok_or(Error::Other("Reason doesn’t contain a valid reason."))?;
|
2019-10-22 23:32:41 +00:00
|
|
|
|
Ok(ReasonElement { reason, texts })
|
2017-05-24 21:28:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-20 19:36:13 +00:00
|
|
|
|
impl From<ReasonElement> for Element {
|
|
|
|
|
fn from(reason: ReasonElement) -> Element {
|
2020-03-28 12:07:26 +00:00
|
|
|
|
Element::builder("reason", ns::JINGLE)
|
2018-12-18 14:32:05 +00:00
|
|
|
|
.append(Element::from(reason.reason))
|
2019-10-22 23:32:41 +00:00
|
|
|
|
.append_all(reason.texts.into_iter().map(|(lang, text)| {
|
2020-03-28 12:07:26 +00:00
|
|
|
|
Element::builder("text", ns::JINGLE)
|
2019-10-22 23:32:41 +00:00
|
|
|
|
.attr("xml:lang", lang)
|
|
|
|
|
.append(text)
|
|
|
|
|
}))
|
2018-12-18 14:32:05 +00:00
|
|
|
|
.build()
|
2017-05-24 21:28:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 18:28:50 +00:00
|
|
|
|
generate_id!(
|
|
|
|
|
/// Unique identifier for a session between two JIDs.
|
|
|
|
|
SessionId
|
|
|
|
|
);
|
2017-06-14 01:28:42 +00:00
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// The main Jingle container, to be included in an iq stanza.
|
2020-11-29 20:17:51 +00:00
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
2017-04-19 01:27:42 +00:00
|
|
|
|
pub struct Jingle {
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// The action to execute on both ends.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
pub action: Action,
|
2018-09-20 19:15:50 +00:00
|
|
|
|
|
|
|
|
|
/// Who the initiator is.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
pub initiator: Option<Jid>,
|
2018-09-20 19:15:50 +00:00
|
|
|
|
|
|
|
|
|
/// Who the responder is.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
pub responder: Option<Jid>,
|
2018-09-20 19:15:50 +00:00
|
|
|
|
|
|
|
|
|
/// Unique session identifier between two entities.
|
2017-06-25 21:14:51 +00:00
|
|
|
|
pub sid: SessionId,
|
2018-09-20 19:15:50 +00:00
|
|
|
|
|
2019-09-08 16:24:45 +00:00
|
|
|
|
/// A list of contents to be negotiated in this session.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
pub contents: Vec<Content>,
|
2018-09-20 19:15:50 +00:00
|
|
|
|
|
|
|
|
|
/// An optional reason.
|
2017-04-19 01:27:42 +00:00
|
|
|
|
pub reason: Option<ReasonElement>,
|
2018-09-20 19:15:50 +00:00
|
|
|
|
|
2020-11-27 19:58:20 +00:00
|
|
|
|
/// An optional grouping.
|
|
|
|
|
pub group: Option<Group>,
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Payloads to be included.
|
2017-04-28 22:45:27 +00:00
|
|
|
|
pub other: Vec<Element>,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-05-16 13:08:17 +00:00
|
|
|
|
impl IqSetPayload for Jingle {}
|
|
|
|
|
|
2018-05-04 17:10:29 +00:00
|
|
|
|
impl Jingle {
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Create a new Jingle element.
|
2018-05-04 17:10:29 +00:00
|
|
|
|
pub fn new(action: Action, sid: SessionId) -> Jingle {
|
|
|
|
|
Jingle {
|
2019-02-21 20:00:58 +00:00
|
|
|
|
action,
|
|
|
|
|
sid,
|
2018-05-04 17:10:29 +00:00
|
|
|
|
initiator: None,
|
|
|
|
|
responder: None,
|
|
|
|
|
contents: Vec::new(),
|
|
|
|
|
reason: None,
|
2020-11-27 19:58:20 +00:00
|
|
|
|
group: None,
|
2018-05-04 17:10:29 +00:00
|
|
|
|
other: Vec::new(),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Set the initiator’s JID.
|
2018-05-04 17:10:29 +00:00
|
|
|
|
pub fn with_initiator(mut self, initiator: Jid) -> Jingle {
|
|
|
|
|
self.initiator = Some(initiator);
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Set the responder’s JID.
|
2018-05-04 17:10:29 +00:00
|
|
|
|
pub fn with_responder(mut self, responder: Jid) -> Jingle {
|
|
|
|
|
self.responder = Some(responder);
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Add a content to this Jingle container.
|
2018-05-04 17:10:29 +00:00
|
|
|
|
pub fn add_content(mut self, content: Content) -> Jingle {
|
|
|
|
|
self.contents.push(content);
|
|
|
|
|
self
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-20 19:15:50 +00:00
|
|
|
|
/// Set the reason in this Jingle container.
|
2020-11-27 19:55:41 +00:00
|
|
|
|
pub fn set_reason(mut self, reason: ReasonElement) -> Jingle {
|
|
|
|
|
self.reason = Some(reason);
|
2018-05-04 17:10:29 +00:00
|
|
|
|
self
|
|
|
|
|
}
|
2020-11-27 19:58:20 +00:00
|
|
|
|
|
|
|
|
|
/// Set the grouping in this Jingle container.
|
|
|
|
|
pub fn set_group(mut self, group: Group) -> Jingle {
|
|
|
|
|
self.group = Some(group);
|
|
|
|
|
self
|
|
|
|
|
}
|
2018-05-04 17:10:29 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-05-23 22:31:33 +00:00
|
|
|
|
impl TryFrom<Element> for Jingle {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
type Error = FromElementError;
|
2017-04-19 22:41:54 +00:00
|
|
|
|
|
2024-06-21 14:27:43 +00:00
|
|
|
|
fn try_from(root: Element) -> Result<Jingle, FromElementError> {
|
2018-05-14 14:30:28 +00:00
|
|
|
|
check_self!(root, "jingle", JINGLE, "Jingle");
|
2018-05-14 23:47:12 +00:00
|
|
|
|
check_no_unknown_attributes!(root, "Jingle", ["action", "initiator", "responder", "sid"]);
|
2017-05-04 00:20:28 +00:00
|
|
|
|
|
2017-05-24 21:28:54 +00:00
|
|
|
|
let mut jingle = Jingle {
|
2019-02-24 19:48:19 +00:00
|
|
|
|
action: get_attr!(root, "action", Required),
|
|
|
|
|
initiator: get_attr!(root, "initiator", Option),
|
|
|
|
|
responder: get_attr!(root, "responder", Option),
|
|
|
|
|
sid: get_attr!(root, "sid", Required),
|
2018-12-18 14:32:05 +00:00
|
|
|
|
contents: vec![],
|
2017-05-24 21:28:54 +00:00
|
|
|
|
reason: None,
|
2020-11-27 19:58:20 +00:00
|
|
|
|
group: None,
|
2018-12-18 14:32:05 +00:00
|
|
|
|
other: vec![],
|
2017-05-24 21:28:54 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
for child in root.children().cloned() {
|
2017-05-04 00:20:28 +00:00
|
|
|
|
if child.is("content", ns::JINGLE) {
|
2017-05-24 22:38:44 +00:00
|
|
|
|
let content = Content::try_from(child)?;
|
2017-05-24 21:28:54 +00:00
|
|
|
|
jingle.contents.push(content);
|
2017-05-04 00:20:28 +00:00
|
|
|
|
} else if child.is("reason", ns::JINGLE) {
|
2017-05-24 21:28:54 +00:00
|
|
|
|
if jingle.reason.is_some() {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
return Err(Error::Other("Jingle must not have more than one reason.").into());
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
2017-05-24 22:38:44 +00:00
|
|
|
|
let reason = ReasonElement::try_from(child)?;
|
2017-05-24 21:28:54 +00:00
|
|
|
|
jingle.reason = Some(reason);
|
2020-11-27 19:58:20 +00:00
|
|
|
|
} else if child.is("group", ns::JINGLE_GROUPING) {
|
|
|
|
|
if jingle.group.is_some() {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
return Err(Error::Other("Jingle must not have more than one grouping.").into());
|
2020-11-27 19:58:20 +00:00
|
|
|
|
}
|
|
|
|
|
let group = Group::try_from(child)?;
|
|
|
|
|
jingle.group = Some(group);
|
2017-05-04 00:20:28 +00:00
|
|
|
|
} else {
|
2017-05-24 22:38:44 +00:00
|
|
|
|
jingle.other.push(child);
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-05-24 21:28:54 +00:00
|
|
|
|
Ok(jingle)
|
2017-04-24 18:25:00 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-20 19:36:13 +00:00
|
|
|
|
impl From<Jingle> for Element {
|
|
|
|
|
fn from(jingle: Jingle) -> Element {
|
2020-03-28 12:07:26 +00:00
|
|
|
|
Element::builder("jingle", ns::JINGLE)
|
2018-12-18 14:32:05 +00:00
|
|
|
|
.attr("action", jingle.action)
|
|
|
|
|
.attr("initiator", jingle.initiator)
|
|
|
|
|
.attr("responder", jingle.responder)
|
|
|
|
|
.attr("sid", jingle.sid)
|
2019-09-06 14:03:58 +00:00
|
|
|
|
.append_all(jingle.contents)
|
|
|
|
|
.append_all(jingle.reason.map(Element::from))
|
2020-11-27 19:58:20 +00:00
|
|
|
|
.append_all(jingle.group.map(Element::from))
|
2018-12-18 14:32:05 +00:00
|
|
|
|
.build()
|
2017-04-24 18:25:00 +00:00
|
|
|
|
}
|
2017-05-04 00:20:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-04-19 01:27:42 +00:00
|
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
2017-05-04 00:20:28 +00:00
|
|
|
|
use super::*;
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
2018-10-28 12:10:48 +00:00
|
|
|
|
#[cfg(target_pointer_width = "32")]
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_size() {
|
|
|
|
|
assert_size!(Action, 1);
|
|
|
|
|
assert_size!(Creator, 1);
|
|
|
|
|
assert_size!(Senders, 1);
|
|
|
|
|
assert_size!(Disposition, 1);
|
|
|
|
|
assert_size!(ContentId, 12);
|
2024-02-27 11:27:31 +00:00
|
|
|
|
assert_size!(Content, 216);
|
2018-10-28 12:10:48 +00:00
|
|
|
|
assert_size!(Reason, 1);
|
|
|
|
|
assert_size!(ReasonElement, 16);
|
|
|
|
|
assert_size!(SessionId, 12);
|
2024-04-15 15:03:57 +00:00
|
|
|
|
assert_size!(Jingle, 104);
|
2018-10-28 12:10:48 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(target_pointer_width = "64")]
|
2018-10-26 12:26:16 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_size() {
|
|
|
|
|
assert_size!(Action, 1);
|
|
|
|
|
assert_size!(Creator, 1);
|
|
|
|
|
assert_size!(Senders, 1);
|
|
|
|
|
assert_size!(Disposition, 1);
|
|
|
|
|
assert_size!(ContentId, 24);
|
2024-06-16 08:45:09 +00:00
|
|
|
|
assert_size!(Content, 432);
|
2018-10-26 12:26:16 +00:00
|
|
|
|
assert_size!(Reason, 1);
|
|
|
|
|
assert_size!(ReasonElement, 32);
|
|
|
|
|
assert_size!(SessionId, 24);
|
2024-04-15 15:03:57 +00:00
|
|
|
|
assert_size!(Jingle, 208);
|
2018-10-26 12:26:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-04-19 01:27:42 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_simple() {
|
2018-12-18 14:32:05 +00:00
|
|
|
|
let elem: Element =
|
|
|
|
|
"<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'/>"
|
|
|
|
|
.parse()
|
|
|
|
|
.unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let jingle = Jingle::try_from(elem).unwrap();
|
2017-05-04 00:20:28 +00:00
|
|
|
|
assert_eq!(jingle.action, Action::SessionInitiate);
|
2017-06-25 21:14:51 +00:00
|
|
|
|
assert_eq!(jingle.sid, SessionId(String::from("coucou")));
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_invalid_jingle() {
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1'/>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
2017-05-24 21:28:54 +00:00
|
|
|
|
assert_eq!(message, "Required attribute 'action' missing.");
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
2018-12-18 14:32:05 +00:00
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-info'/>"
|
|
|
|
|
.parse()
|
|
|
|
|
.unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
2017-05-24 21:28:54 +00:00
|
|
|
|
assert_eq!(message, "Required attribute 'sid' missing.");
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
2018-12-18 14:32:05 +00:00
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='coucou' sid='coucou'/>"
|
|
|
|
|
.parse()
|
|
|
|
|
.unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::TextParseError(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
2024-06-21 14:27:43 +00:00
|
|
|
|
assert_eq!(message.to_string(), "Unknown value for 'action' attribute.");
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_content() {
|
2019-10-12 15:13:42 +00:00
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='initiator' name='coucou'><description/><transport xmlns='urn:xmpp:jingle:transports:stub:0'/></content></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let jingle = Jingle::try_from(elem).unwrap();
|
2017-05-04 00:20:28 +00:00
|
|
|
|
assert_eq!(jingle.contents[0].creator, Creator::Initiator);
|
2017-07-15 10:37:29 +00:00
|
|
|
|
assert_eq!(jingle.contents[0].name, ContentId(String::from("coucou")));
|
2017-05-04 00:20:28 +00:00
|
|
|
|
assert_eq!(jingle.contents[0].senders, Senders::Both);
|
2017-10-31 17:17:06 +00:00
|
|
|
|
assert_eq!(jingle.contents[0].disposition, Disposition::Session);
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
2019-10-12 15:13:42 +00:00
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='initiator' name='coucou' senders='both'><description/><transport xmlns='urn:xmpp:jingle:transports:stub:0'/></content></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let jingle = Jingle::try_from(elem).unwrap();
|
2017-05-04 00:20:28 +00:00
|
|
|
|
assert_eq!(jingle.contents[0].senders, Senders::Both);
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
2019-10-12 15:13:42 +00:00
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='initiator' name='coucou' disposition='early-session'><description/><transport xmlns='urn:xmpp:jingle:transports:stub:0'/></content></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let jingle = Jingle::try_from(elem).unwrap();
|
2017-10-31 17:17:06 +00:00
|
|
|
|
assert_eq!(jingle.contents[0].disposition, Disposition::EarlySession);
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_invalid_content() {
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content/></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
2017-05-24 21:28:54 +00:00
|
|
|
|
assert_eq!(message, "Required attribute 'creator' missing.");
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='initiator'/></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
2017-05-24 21:28:54 +00:00
|
|
|
|
assert_eq!(message, "Required attribute 'name' missing.");
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='coucou' name='coucou'/></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::TextParseError(string)) => string,
|
|
|
|
|
other => panic!("unexpected result: {:?}", other),
|
2017-04-19 01:27:42 +00:00
|
|
|
|
};
|
2024-06-21 14:27:43 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
message.to_string(),
|
|
|
|
|
"Unknown value for 'creator' attribute."
|
|
|
|
|
);
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='initiator' name='coucou' senders='coucou'/></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::TextParseError(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
2024-06-21 14:27:43 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
message.to_string(),
|
|
|
|
|
"Unknown value for 'senders' attribute."
|
|
|
|
|
);
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='initiator' name='coucou' senders=''/></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::TextParseError(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
2024-06-21 14:27:43 +00:00
|
|
|
|
assert_eq!(
|
|
|
|
|
message.to_string(),
|
|
|
|
|
"Unknown value for 'senders' attribute."
|
|
|
|
|
);
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_reason() {
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><success/></reason></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let jingle = Jingle::try_from(elem).unwrap();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let reason = jingle.reason.unwrap();
|
2017-05-04 00:20:28 +00:00
|
|
|
|
assert_eq!(reason.reason, Reason::Success);
|
2019-02-28 01:40:04 +00:00
|
|
|
|
assert_eq!(reason.texts, BTreeMap::new());
|
2017-04-19 01:27:42 +00:00
|
|
|
|
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><success/><text>coucou</text></reason></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let jingle = Jingle::try_from(elem).unwrap();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let reason = jingle.reason.unwrap();
|
2017-05-04 00:20:28 +00:00
|
|
|
|
assert_eq!(reason.reason, Reason::Success);
|
2019-02-28 01:40:04 +00:00
|
|
|
|
assert_eq!(reason.texts.get(""), Some(&String::from("coucou")));
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_invalid_reason() {
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason/></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
|
|
|
|
assert_eq!(message, "Reason doesn’t contain a valid reason.");
|
|
|
|
|
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><a/></reason></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
|
|
|
|
assert_eq!(message, "Unknown reason.");
|
|
|
|
|
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><a xmlns='http://www.w3.org/1999/xhtml'/></reason></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
|
|
|
|
assert_eq!(message, "Reason contains a foreign element.");
|
|
|
|
|
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><decline/></reason><reason/></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:27:42 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-04-19 01:27:42 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
|
|
|
|
assert_eq!(message, "Jingle must not have more than one reason.");
|
2017-04-19 01:38:10 +00:00
|
|
|
|
|
|
|
|
|
let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><decline/><text/><text/></reason></jingle>".parse().unwrap();
|
2017-05-23 22:31:33 +00:00
|
|
|
|
let error = Jingle::try_from(elem).unwrap_err();
|
2017-04-19 01:38:10 +00:00
|
|
|
|
let message = match error {
|
2024-06-21 14:27:43 +00:00
|
|
|
|
FromElementError::Invalid(Error::Other(string)) => string,
|
2017-04-19 01:38:10 +00:00
|
|
|
|
_ => panic!(),
|
|
|
|
|
};
|
2019-02-28 01:40:04 +00:00
|
|
|
|
assert_eq!(message, "Text element present twice for the same xml:lang.");
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|
2019-11-27 17:16:33 +00:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_serialize_jingle() {
|
|
|
|
|
let reference: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='a73sjjvkla37jfea'><content xmlns='urn:xmpp:jingle:1' creator='initiator' name='this-is-a-stub'><description xmlns='urn:xmpp:jingle:apps:stub:0'/><transport xmlns='urn:xmpp:jingle:transports:stub:0'/></content></jingle>"
|
|
|
|
|
.parse()
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
|
|
let jingle = Jingle {
|
|
|
|
|
action: Action::SessionInitiate,
|
|
|
|
|
initiator: None,
|
|
|
|
|
responder: None,
|
|
|
|
|
sid: SessionId(String::from("a73sjjvkla37jfea")),
|
|
|
|
|
contents: vec![Content {
|
|
|
|
|
creator: Creator::Initiator,
|
|
|
|
|
disposition: Disposition::default(),
|
|
|
|
|
name: ContentId(String::from("this-is-a-stub")),
|
|
|
|
|
senders: Senders::default(),
|
|
|
|
|
description: Some(Description::Unknown(
|
2020-04-02 20:45:20 +00:00
|
|
|
|
Element::builder("description", "urn:xmpp:jingle:apps:stub:0").build(),
|
2019-11-27 17:16:33 +00:00
|
|
|
|
)),
|
|
|
|
|
transport: Some(Transport::Unknown(
|
2020-04-02 20:45:20 +00:00
|
|
|
|
Element::builder("transport", "urn:xmpp:jingle:transports:stub:0").build(),
|
2019-11-27 17:16:33 +00:00
|
|
|
|
)),
|
|
|
|
|
security: None,
|
|
|
|
|
}],
|
|
|
|
|
reason: None,
|
2020-11-27 19:58:20 +00:00
|
|
|
|
group: None,
|
2019-11-27 17:16:33 +00:00
|
|
|
|
other: vec![],
|
|
|
|
|
};
|
|
|
|
|
let serialized: Element = jingle.into();
|
|
|
|
|
assert_eq!(serialized, reference);
|
|
|
|
|
}
|
2017-04-19 01:27:42 +00:00
|
|
|
|
}
|