1use xso::{AsXml, FromXml};
8
9use crate::iq::IqSetPayload;
10use crate::jingle_grouping::Group;
11use crate::jingle_ibb::Transport as IbbTransport;
12use crate::jingle_ice_udp::Transport as IceUdpTransport;
13use crate::jingle_rtp::Description as RtpDescription;
14use crate::jingle_s5b::Transport as Socks5Transport;
15use crate::ns;
16use alloc::{collections::BTreeMap, fmt};
17use jid::Jid;
18use minidom::Element;
19use xso::error::Error;
20
21generate_attribute!(
22 Action, "action", {
24 ContentAccept => "content-accept",
26
27 ContentAdd => "content-add",
29
30 ContentModify => "content-modify",
32
33 ContentReject => "content-reject",
35
36 ContentRemove => "content-remove",
38
39 DescriptionInfo => "description-info",
41
42 SecurityInfo => "security-info",
44
45 SessionAccept => "session-accept",
47
48 SessionInfo => "session-info",
50
51 SessionInitiate => "session-initiate",
53
54 SessionTerminate => "session-terminate",
56
57 TransportAccept => "transport-accept",
59
60 TransportInfo => "transport-info",
62
63 TransportReject => "transport-reject",
65
66 TransportReplace => "transport-replace",
68 }
69);
70
71generate_attribute!(
72 Creator, "creator", {
74 Initiator => "initiator",
76
77 Responder => "responder",
79 }
80);
81
82generate_attribute!(
83 Senders, "senders", {
85 Both => "both",
87
88 Initiator => "initiator",
90
91 None => "none",
93
94 Responder => "responder",
96 }, Default = Both
97);
98
99generate_attribute!(
100 Disposition, "disposition", {
107 Inline => "inline",
109
110 Attachment => "attachment",
112
113 FormData => "form-data",
115
116 Signal => "signal",
118
119 Alert => "alert",
121
122 Icon => "icon",
124
125 Render => "render",
127
128 RecipientListHistory => "recipient-list-history",
131
132 Session => "session",
135
136 Aib => "aib",
138
139 EarlySession => "early-session",
142
143 RecipientList => "recipient-list",
146
147 Notification => "notification",
151
152 ByReference => "by-reference",
155
156 InfoPackage => "info-package",
158
159 RecordingSession => "recording-session",
163 }, Default = Session
164);
165
166generate_id!(
167 ContentId
170);
171
172#[derive(AsXml, Debug, Clone, PartialEq)]
174#[xml()]
175pub enum Description {
176 #[xml(transparent)]
178 Rtp(RtpDescription),
179
180 #[xml(transparent)]
183 Unknown(Element),
184}
185
186impl TryFrom<Element> for Description {
187 type Error = Error;
188
189 fn try_from(elem: Element) -> Result<Description, Error> {
190 Ok(if elem.is("description", ns::JINGLE_RTP) {
191 Description::Rtp(RtpDescription::try_from(elem)?)
192 } else if elem.name() == "description" {
193 Description::Unknown(elem)
194 } else {
195 return Err(Error::Other("Invalid description."));
196 })
197 }
198}
199
200impl ::xso::FromXml for Description {
201 type Builder = ::xso::minidom_compat::FromEventsViaElement<Description>;
202
203 fn from_events(
204 qname: ::xso::exports::rxml::QName,
205 attrs: ::xso::exports::rxml::AttrMap,
206 ) -> Result<Self::Builder, ::xso::error::FromEventsError> {
207 if qname.1 != "description" {
208 return Err(::xso::error::FromEventsError::Mismatch { name: qname, attrs });
209 }
210 Self::Builder::new(qname, attrs)
211 }
212}
213
214impl From<RtpDescription> for Description {
215 fn from(desc: RtpDescription) -> Description {
216 Description::Rtp(desc)
217 }
218}
219
220#[derive(AsXml, Debug, Clone, PartialEq)]
222#[xml()]
223pub enum Transport {
224 #[xml(transparent)]
226 IceUdp(IceUdpTransport),
227
228 #[xml(transparent)]
230 Ibb(IbbTransport),
231
232 #[xml(transparent)]
234 Socks5(Socks5Transport),
235
236 #[xml(transparent)]
239 Unknown(Element),
240}
241
242impl TryFrom<Element> for Transport {
243 type Error = Error;
244
245 fn try_from(elem: Element) -> Result<Transport, Error> {
246 Ok(if elem.is("transport", ns::JINGLE_ICE_UDP) {
247 Transport::IceUdp(IceUdpTransport::try_from(elem)?)
248 } else if elem.is("transport", ns::JINGLE_IBB) {
249 Transport::Ibb(IbbTransport::try_from(elem)?)
250 } else if elem.is("transport", ns::JINGLE_S5B) {
251 Transport::Socks5(Socks5Transport::try_from(elem)?)
252 } else if elem.name() == "transport" {
253 Transport::Unknown(elem)
254 } else {
255 return Err(Error::Other("Invalid transport."));
256 })
257 }
258}
259
260impl ::xso::FromXml for Transport {
261 type Builder = ::xso::minidom_compat::FromEventsViaElement<Transport>;
262
263 fn from_events(
264 qname: ::xso::exports::rxml::QName,
265 attrs: ::xso::exports::rxml::AttrMap,
266 ) -> Result<Self::Builder, ::xso::error::FromEventsError> {
267 if qname.1 != "transport" {
268 return Err(::xso::error::FromEventsError::Mismatch { name: qname, attrs });
269 }
270 Self::Builder::new(qname, attrs)
271 }
272}
273
274impl From<IceUdpTransport> for Transport {
275 fn from(transport: IceUdpTransport) -> Transport {
276 Transport::IceUdp(transport)
277 }
278}
279
280impl From<IbbTransport> for Transport {
281 fn from(transport: IbbTransport) -> Transport {
282 Transport::Ibb(transport)
283 }
284}
285
286impl From<Socks5Transport> for Transport {
287 fn from(transport: Socks5Transport) -> Transport {
288 Transport::Socks5(transport)
289 }
290}
291
292#[derive(FromXml, AsXml, Debug, Clone, PartialEq)]
294#[xml(namespace = ns::JINGLE, name = "security")]
295pub struct Security;
296
297#[derive(FromXml, AsXml, Debug, Clone, PartialEq)]
300#[xml(namespace = ns::JINGLE, name = "content")]
301pub struct Content {
302 #[xml(attribute)]
304 pub creator: Creator,
305
306 #[xml(attribute(default))]
308 pub disposition: Disposition,
309
310 #[xml(attribute)]
312 pub name: ContentId,
313
314 #[xml(attribute(default))]
316 pub senders: Senders,
317
318 #[xml(child(default))]
320 pub description: Option<Description>,
321
322 #[xml(child(default))]
324 pub transport: Option<Transport>,
325
326 #[xml(child(default))]
328 pub security: Option<Security>,
329}
330
331impl Content {
332 pub fn new(creator: Creator, name: ContentId) -> Content {
334 Content {
335 creator,
336 name,
337 disposition: Disposition::Session,
338 senders: Senders::Both,
339 description: None,
340 transport: None,
341 security: None,
342 }
343 }
344
345 pub fn with_disposition(mut self, disposition: Disposition) -> Content {
347 self.disposition = disposition;
348 self
349 }
350
351 pub fn with_senders(mut self, senders: Senders) -> Content {
353 self.senders = senders;
354 self
355 }
356
357 pub fn with_description<D: Into<Description>>(mut self, description: D) -> Content {
359 self.description = Some(description.into());
360 self
361 }
362
363 pub fn with_transport<T: Into<Transport>>(mut self, transport: T) -> Content {
365 self.transport = Some(transport.into());
366 self
367 }
368
369 pub fn with_security(mut self, security: Security) -> Content {
371 self.security = Some(security);
372 self
373 }
374}
375
376#[derive(FromXml, AsXml, Debug, Clone, PartialEq)]
378#[xml(namespace = ns::JINGLE)]
379pub enum Reason {
380 #[xml(name = "alternative-session")]
385 AlternativeSession {
386 #[xml(extract(namespace = ns::JINGLE, name = "sid", default, fields(text(type_ = String))))]
388 sid: Option<String>,
389 },
390
391 #[xml(name = "busy")]
393 Busy,
394
395 #[xml(name = "cancel")]
397 Cancel,
398
399 #[xml(name = "connectivity-error")]
401 ConnectivityError,
402
403 #[xml(name = "decline")]
405 Decline,
406
407 #[xml(name = "expired")]
410 Expired,
411
412 #[xml(name = "failed-application")]
415 FailedApplication,
416
417 #[xml(name = "failed-transport")]
420 FailedTransport,
421
422 #[xml(name = "general-error")]
424 GeneralError,
425
426 #[xml(name = "gone")]
428 Gone,
429
430 #[xml(name = "incompatible-parameters")]
433 IncompatibleParameters,
434
435 #[xml(name = "media-error")]
437 MediaError,
438
439 #[xml(name = "security-error")]
441 SecurityError,
442
443 #[xml(name = "success")]
446 Success,
447
448 #[xml(name = "timeout")]
451 Timeout,
452
453 #[xml(name = "unsupported-applications")]
455 UnsupportedApplications,
456
457 #[xml(name = "unsupported-transports")]
459 UnsupportedTransports,
460}
461
462type Lang = String;
463
464#[derive(FromXml, AsXml, Debug, Clone, PartialEq)]
466#[xml(namespace = ns::JINGLE, name = "reason")]
467pub struct ReasonElement {
468 #[xml(child)]
470 pub reason: Reason,
471
472 #[xml(extract(n = .., namespace = ns::JINGLE, name = "text", fields(
474 attribute(type_ = String, name = "xml:lang", default),
475 text(type_ = String),
476 )))]
477 pub texts: BTreeMap<Lang, String>,
478}
479
480impl fmt::Display for ReasonElement {
481 fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
482 write!(fmt, "{}", Element::from(self.reason.clone()).name())?;
483 if let Some(text) = self.texts.get("en") {
484 write!(fmt, ": {}", text)?;
485 } else if let Some(text) = self.texts.get("") {
486 write!(fmt, ": {}", text)?;
487 }
488 Ok(())
489 }
490}
491
492generate_id!(
493 SessionId
495);
496
497#[derive(FromXml, AsXml, Debug, Clone, PartialEq)]
499#[xml(namespace = ns::JINGLE, name = "jingle")]
500pub struct Jingle {
501 #[xml(attribute)]
503 pub action: Action,
504
505 #[xml(attribute(default))]
507 pub initiator: Option<Jid>,
508
509 #[xml(attribute(default))]
511 pub responder: Option<Jid>,
512
513 #[xml(attribute)]
515 pub sid: SessionId,
516
517 #[xml(child(n = ..))]
519 pub contents: Vec<Content>,
520
521 #[xml(child(default))]
523 pub reason: Option<ReasonElement>,
524
525 #[xml(child(default))]
527 pub group: Option<Group>,
528
529 #[xml(child(n = ..))]
531 pub other: Vec<Element>,
532}
533
534impl IqSetPayload for Jingle {}
535
536impl Jingle {
537 pub fn new(action: Action, sid: SessionId) -> Jingle {
539 Jingle {
540 action,
541 sid,
542 initiator: None,
543 responder: None,
544 contents: Vec::new(),
545 reason: None,
546 group: None,
547 other: Vec::new(),
548 }
549 }
550
551 pub fn with_initiator(mut self, initiator: Jid) -> Jingle {
553 self.initiator = Some(initiator);
554 self
555 }
556
557 pub fn with_responder(mut self, responder: Jid) -> Jingle {
559 self.responder = Some(responder);
560 self
561 }
562
563 pub fn add_content(mut self, content: Content) -> Jingle {
565 self.contents.push(content);
566 self
567 }
568
569 pub fn set_reason(mut self, reason: ReasonElement) -> Jingle {
571 self.reason = Some(reason);
572 self
573 }
574
575 pub fn set_group(mut self, group: Group) -> Jingle {
577 self.group = Some(group);
578 self
579 }
580}
581
582#[cfg(test)]
583mod tests {
584 use super::*;
585 use xso::error::FromElementError;
586
587 #[cfg(target_pointer_width = "32")]
588 #[test]
589 fn test_size() {
590 assert_size!(Action, 1);
591 assert_size!(Creator, 1);
592 assert_size!(Senders, 1);
593 assert_size!(Disposition, 1);
594 assert_size!(ContentId, 12);
595 assert_size!(Content, 156);
596 assert_size!(Reason, 12);
597 assert_size!(ReasonElement, 24);
598 assert_size!(SessionId, 12);
599 assert_size!(Jingle, 112);
600 }
601
602 #[cfg(target_pointer_width = "64")]
603 #[test]
604 fn test_size() {
605 assert_size!(Action, 1);
606 assert_size!(Creator, 1);
607 assert_size!(Senders, 1);
608 assert_size!(Disposition, 1);
609 assert_size!(ContentId, 24);
610 assert_size!(Content, 312);
611 assert_size!(Reason, 24);
612 assert_size!(ReasonElement, 48);
613 assert_size!(SessionId, 24);
614 assert_size!(Jingle, 224);
615 }
616
617 #[test]
618 fn test_simple() {
619 let elem: Element =
620 "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'/>"
621 .parse()
622 .unwrap();
623 let jingle = Jingle::try_from(elem).unwrap();
624 assert_eq!(jingle.action, Action::SessionInitiate);
625 assert_eq!(jingle.sid, SessionId(String::from("coucou")));
626 }
627
628 #[test]
629 fn test_invalid_jingle() {
630 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1'/>".parse().unwrap();
631 let error = Jingle::try_from(elem).unwrap_err();
632 let message = match error {
633 FromElementError::Invalid(Error::Other(string)) => string,
634 _ => panic!(),
635 };
636 assert_eq!(
637 message,
638 "Required attribute field 'action' on Jingle element missing."
639 );
640
641 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-info'/>"
642 .parse()
643 .unwrap();
644 let error = Jingle::try_from(elem).unwrap_err();
645 let message = match error {
646 FromElementError::Invalid(Error::Other(string)) => string,
647 _ => panic!(),
648 };
649 assert_eq!(
650 message,
651 "Required attribute field 'sid' on Jingle element missing."
652 );
653
654 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='coucou' sid='coucou'/>"
655 .parse()
656 .unwrap();
657 let error = Jingle::try_from(elem).unwrap_err();
658 let message = match error {
659 FromElementError::Invalid(Error::TextParseError(string)) => string,
660 _ => panic!(),
661 };
662 assert_eq!(message.to_string(), "Unknown value for 'action' attribute.");
663 }
664
665 #[test]
666 fn test_content() {
667 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();
668 let jingle = Jingle::try_from(elem).unwrap();
669 assert_eq!(jingle.contents[0].creator, Creator::Initiator);
670 assert_eq!(jingle.contents[0].name, ContentId(String::from("coucou")));
671 assert_eq!(jingle.contents[0].senders, Senders::Both);
672 assert_eq!(jingle.contents[0].disposition, Disposition::Session);
673
674 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();
675 let jingle = Jingle::try_from(elem).unwrap();
676 assert_eq!(jingle.contents[0].senders, Senders::Both);
677
678 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();
679 let jingle = Jingle::try_from(elem).unwrap();
680 assert_eq!(jingle.contents[0].disposition, Disposition::EarlySession);
681 }
682
683 #[test]
684 fn test_invalid_content() {
685 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content/></jingle>".parse().unwrap();
686 let error = Jingle::try_from(elem).unwrap_err();
687 let message = match error {
688 FromElementError::Invalid(Error::Other(string)) => string,
689 _ => panic!(),
690 };
691 assert_eq!(
692 message,
693 "Required attribute field 'creator' on Content element missing."
694 );
695
696 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='initiator'/></jingle>".parse().unwrap();
697 let error = Jingle::try_from(elem).unwrap_err();
698 let message = match error {
699 FromElementError::Invalid(Error::Other(string)) => string,
700 _ => panic!(),
701 };
702 assert_eq!(
703 message,
704 "Required attribute field 'name' on Content element missing."
705 );
706
707 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='coucou' name='coucou'/></jingle>".parse().unwrap();
708 let error = Jingle::try_from(elem).unwrap_err();
709 let message = match error {
710 FromElementError::Invalid(Error::TextParseError(string)) => string,
711 other => panic!("unexpected result: {:?}", other),
712 };
713 assert_eq!(
714 message.to_string(),
715 "Unknown value for 'creator' attribute."
716 );
717
718 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='initiator' name='coucou' senders='coucou'/></jingle>".parse().unwrap();
719 let error = Jingle::try_from(elem).unwrap_err();
720 let message = match error {
721 FromElementError::Invalid(Error::TextParseError(string)) => string,
722 _ => panic!(),
723 };
724 assert_eq!(
725 message.to_string(),
726 "Unknown value for 'senders' attribute."
727 );
728
729 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><content creator='initiator' name='coucou' senders=''/></jingle>".parse().unwrap();
730 let error = Jingle::try_from(elem).unwrap_err();
731 let message = match error {
732 FromElementError::Invalid(Error::TextParseError(string)) => string,
733 _ => panic!(),
734 };
735 assert_eq!(
736 message.to_string(),
737 "Unknown value for 'senders' attribute."
738 );
739 }
740
741 #[test]
742 fn test_reason() {
743 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><success/></reason></jingle>".parse().unwrap();
744 let jingle = Jingle::try_from(elem).unwrap();
745 let reason = jingle.reason.unwrap();
746 assert_eq!(reason.reason, Reason::Success);
747 assert_eq!(reason.texts, BTreeMap::new());
748
749 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><success/><text>coucou</text></reason></jingle>".parse().unwrap();
750 let jingle = Jingle::try_from(elem).unwrap();
751 let reason = jingle.reason.unwrap();
752 assert_eq!(reason.reason, Reason::Success);
753 assert_eq!(reason.texts.get(""), Some(&String::from("coucou")));
754 }
755
756 #[test]
757 fn test_missing_reason_text() {
758 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason/></jingle>".parse().unwrap();
759 let error = Jingle::try_from(elem).unwrap_err();
760 let message = match error {
761 FromElementError::Invalid(Error::Other(string)) => string,
762 _ => panic!(),
763 };
764 assert_eq!(
765 message,
766 "Missing child field 'reason' in ReasonElement element."
767 );
768 }
769
770 #[test]
771 #[cfg_attr(feature = "disable-validation", should_panic = "Result::unwrap_err")]
772 fn test_invalid_child_in_reason() {
773 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><decline/><a/></reason></jingle>".parse().unwrap();
774 let error = Jingle::try_from(elem).unwrap_err();
775 let message = match error {
776 FromElementError::Invalid(Error::Other(string)) => string,
777 _ => panic!(),
778 };
779 assert_eq!(message, "Unknown child in ReasonElement element.");
780 }
781
782 #[test]
783 fn test_multiple_reason_children() {
784 let elem: Element = "<jingle xmlns='urn:xmpp:jingle:1' action='session-initiate' sid='coucou'><reason><decline/></reason><reason/></jingle>".parse().unwrap();
785 let error = Jingle::try_from(elem).unwrap_err();
786 let message = match error {
787 FromElementError::Invalid(Error::Other(string)) => string,
788 _ => panic!(),
789 };
790 assert_eq!(
791 message,
792 "Jingle element must not have more than one child in field 'reason'."
793 );
794
795 }
807
808 #[test]
809 fn test_serialize_jingle() {
810 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>"
811 .parse()
812 .unwrap();
813
814 let jingle = Jingle {
815 action: Action::SessionInitiate,
816 initiator: None,
817 responder: None,
818 sid: SessionId(String::from("a73sjjvkla37jfea")),
819 contents: vec![Content {
820 creator: Creator::Initiator,
821 disposition: Disposition::default(),
822 name: ContentId(String::from("this-is-a-stub")),
823 senders: Senders::default(),
824 description: Some(Description::Unknown(
825 Element::builder("description", "urn:xmpp:jingle:apps:stub:0").build(),
826 )),
827 transport: Some(Transport::Unknown(
828 Element::builder("transport", "urn:xmpp:jingle:transports:stub:0").build(),
829 )),
830 security: None,
831 }],
832 reason: None,
833 group: None,
834 other: vec![],
835 };
836 let serialized: Element = jingle.into();
837 assert_eq!(serialized, reference);
838 }
839}