1use xso::{text::EmptyAsNone, AsXml, FromXml};
8
9use crate::message::MessagePayload;
10use crate::ns;
11use crate::presence::PresencePayload;
12use alloc::collections::BTreeMap;
13use core::convert::TryFrom;
14use jid::Jid;
15use minidom::Element;
16use xso::error::{Error, FromElementError};
17
18generate_attribute!(
19 ErrorType, "type", {
21 Auth => "auth",
23
24 Cancel => "cancel",
26
27 Continue => "continue",
29
30 Modify => "modify",
32
33 Wait => "wait",
35 }
36);
37
38#[derive(FromXml, AsXml, PartialEq, Debug, Clone)]
40#[xml(namespace = ns::XMPP_STANZAS, exhaustive)]
41pub enum DefinedCondition {
42 #[xml(name = "bad-request")]
49 BadRequest,
50
51 #[xml(name = "conflict")]
55 Conflict,
56
57 #[xml(name = "feature-not-implemented")]
63 FeatureNotImplemented,
64
65 #[xml(name = "forbidden")]
71 Forbidden,
72
73 #[xml(name = "gone")]
83 Gone {
84 #[xml(text(codec = EmptyAsNone))]
87 new_address: Option<String>,
88 },
89
90 #[xml(name = "internal-server-error")]
94 InternalServerError,
95
96 #[xml(name = "item-not-found")]
99 ItemNotFound,
100
101 #[xml(name = "jid-malformed")]
106 JidMalformed,
107
108 #[xml(name = "not-acceptable")]
115 NotAcceptable,
116
117 #[xml(name = "not-allowed")]
121 NotAllowed,
122
123 #[xml(name = "not-authorized")]
131 NotAuthorized,
132
133 #[xml(name = "policy-violation")]
140 PolicyViolation,
141
142 #[xml(name = "recipient-unavailable")]
145 RecipientUnavailable,
146
147 #[xml(name = "redirect")]
156 Redirect {
157 #[xml(text(codec = EmptyAsNone))]
160 new_address: Option<String>,
161 },
162
163 #[xml(name = "registration-required")]
170 RegistrationRequired,
171
172 #[xml(name = "remote-server-not-found")]
179 RemoteServerNotFound,
180
181 #[xml(name = "remote-server-timeout")]
192 RemoteServerTimeout,
193
194 #[xml(name = "resource-constraint")]
198 ResourceConstraint,
199
200 #[xml(name = "service-unavailable")]
203 ServiceUnavailable,
204
205 #[xml(name = "subscription-required")]
212 SubscriptionRequired,
213
214 #[xml(name = "undefined-condition")]
219 UndefinedCondition,
220
221 #[xml(name = "unexpected-request")]
225 UnexpectedRequest,
226}
227
228type Lang = String;
229
230#[derive(Debug, Clone, PartialEq)]
232pub struct StanzaError {
233 pub type_: ErrorType,
235
236 pub by: Option<Jid>,
238
239 pub defined_condition: DefinedCondition,
241
242 pub texts: BTreeMap<Lang, String>,
244
245 pub other: Option<Element>,
247}
248
249impl MessagePayload for StanzaError {}
250impl PresencePayload for StanzaError {}
251
252impl StanzaError {
253 pub fn new<L, T>(
255 type_: ErrorType,
256 defined_condition: DefinedCondition,
257 lang: L,
258 text: T,
259 ) -> StanzaError
260 where
261 L: Into<Lang>,
262 T: Into<String>,
263 {
264 StanzaError {
265 type_,
266 by: None,
267 defined_condition,
268 texts: {
269 let mut map = BTreeMap::new();
270 map.insert(lang.into(), text.into());
271 map
272 },
273 other: None,
274 }
275 }
276}
277
278impl TryFrom<Element> for StanzaError {
279 type Error = FromElementError;
280
281 fn try_from(elem: Element) -> Result<StanzaError, FromElementError> {
282 check_self!(elem, "error", DEFAULT_NS);
283 check_no_unknown_attributes!(elem, "error", ["type", "by", "code"]);
286
287 let mut stanza_error = StanzaError {
288 type_: get_attr!(elem, "type", Required),
289 by: get_attr!(elem, "by", Option),
290 defined_condition: DefinedCondition::UndefinedCondition,
291 texts: BTreeMap::new(),
292 other: None,
293 };
294 let mut defined_condition = None;
295
296 for child in elem.children() {
297 if child.is("text", ns::XMPP_STANZAS) {
298 check_no_children!(child, "text");
299 check_no_unknown_attributes!(child, "text", ["xml:lang"]);
300 let lang = get_attr!(child, "xml:lang", Default);
301 if stanza_error.texts.insert(lang, child.text()).is_some() {
302 return Err(
303 Error::Other("Text element present twice for the same xml:lang.").into(),
304 );
305 }
306 } else if child.has_ns(ns::XMPP_STANZAS) {
307 if defined_condition.is_some() {
308 return Err(Error::Other(
309 "Error must not have more than one defined-condition.",
310 )
311 .into());
312 }
313 check_no_children!(child, "defined-condition");
314 check_no_attributes!(child, "defined-condition");
315 defined_condition = Some(DefinedCondition::try_from(child.clone())?);
316 } else {
317 if stanza_error.other.is_some() {
318 return Err(
319 Error::Other("Error must not have more than one other element.").into(),
320 );
321 }
322 stanza_error.other = Some(child.clone());
323 }
324 }
325 stanza_error.defined_condition =
326 defined_condition.ok_or(Error::Other("Error must have a defined-condition."))?;
327
328 Ok(stanza_error)
329 }
330}
331
332impl From<StanzaError> for Element {
333 fn from(err: StanzaError) -> Element {
334 Element::builder("error", ns::DEFAULT_NS)
335 .attr("type", err.type_)
336 .attr("by", err.by)
337 .append(err.defined_condition)
338 .append_all(err.texts.into_iter().map(|(lang, text)| {
339 Element::builder("text", ns::XMPP_STANZAS)
340 .attr("xml:lang", lang)
341 .append(text)
342 }))
343 .append_all(err.other)
344 .build()
345 }
346}
347
348#[cfg(test)]
349mod tests {
350 use super::*;
351
352 #[cfg(target_pointer_width = "32")]
353 #[test]
354 fn test_size() {
355 assert_size!(ErrorType, 1);
356 assert_size!(DefinedCondition, 16);
357 assert_size!(StanzaError, 108);
358 }
359
360 #[cfg(target_pointer_width = "64")]
361 #[test]
362 fn test_size() {
363 assert_size!(ErrorType, 1);
364 assert_size!(DefinedCondition, 32);
365 assert_size!(StanzaError, 216);
366 }
367
368 #[test]
369 fn test_simple() {
370 #[cfg(not(feature = "component"))]
371 let elem: Element = "<error xmlns='jabber:client' type='cancel'><undefined-condition xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'/></error>".parse().unwrap();
372 #[cfg(feature = "component")]
373 let elem: Element = "<error xmlns='jabber:component:accept' type='cancel'><undefined-condition xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'/></error>".parse().unwrap();
374 let error = StanzaError::try_from(elem).unwrap();
375 assert_eq!(error.type_, ErrorType::Cancel);
376 assert_eq!(
377 error.defined_condition,
378 DefinedCondition::UndefinedCondition
379 );
380 }
381
382 #[test]
383 fn test_invalid_type() {
384 #[cfg(not(feature = "component"))]
385 let elem: Element = "<error xmlns='jabber:client'/>".parse().unwrap();
386 #[cfg(feature = "component")]
387 let elem: Element = "<error xmlns='jabber:component:accept'/>".parse().unwrap();
388 let error = StanzaError::try_from(elem).unwrap_err();
389 let message = match error {
390 FromElementError::Invalid(Error::Other(string)) => string,
391 _ => panic!(),
392 };
393 assert_eq!(message, "Required attribute 'type' missing.");
394
395 #[cfg(not(feature = "component"))]
396 let elem: Element = "<error xmlns='jabber:client' type='coucou'/>"
397 .parse()
398 .unwrap();
399 #[cfg(feature = "component")]
400 let elem: Element = "<error xmlns='jabber:component:accept' type='coucou'/>"
401 .parse()
402 .unwrap();
403 let error = StanzaError::try_from(elem).unwrap_err();
404 let message = match error {
405 FromElementError::Invalid(Error::TextParseError(string)) => string,
406 _ => panic!(),
407 };
408 assert_eq!(message.to_string(), "Unknown value for 'type' attribute.");
409 }
410
411 #[test]
412 fn test_invalid_condition() {
413 #[cfg(not(feature = "component"))]
414 let elem: Element = "<error xmlns='jabber:client' type='cancel'/>"
415 .parse()
416 .unwrap();
417 #[cfg(feature = "component")]
418 let elem: Element = "<error xmlns='jabber:component:accept' type='cancel'/>"
419 .parse()
420 .unwrap();
421 let error = StanzaError::try_from(elem).unwrap_err();
422 let message = match error {
423 FromElementError::Invalid(Error::Other(string)) => string,
424 _ => panic!(),
425 };
426 assert_eq!(message, "Error must have a defined-condition.");
427 }
428
429 #[test]
430 fn test_error_code() {
431 let elem: Element = r#"<error code="501" type="cancel" xmlns='jabber:client'>
432 <feature-not-implemented xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'/>
433 <text xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'>The feature requested is not implemented by the recipient or server and therefore cannot be processed.</text>
434</error>"#
435 .parse()
436 .unwrap();
437 let stanza_error = StanzaError::try_from(elem).unwrap();
438 assert_eq!(stanza_error.type_, ErrorType::Cancel);
439 }
440
441 #[test]
442 fn test_error_multiple_text() {
443 let elem: Element = r#"<error type="cancel" xmlns='jabber:client'>
444 <item-not-found xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'/>
445 <text xmlns='urn:ietf:params:xml:ns:xmpp-stanzas' xml:lang="fr">Nœud non trouvé</text>
446 <text xmlns='urn:ietf:params:xml:ns:xmpp-stanzas' xml:lang="en">Node not found</text>
447</error>"#
448 .parse()
449 .unwrap();
450 let stanza_error = StanzaError::try_from(elem).unwrap();
451 assert_eq!(stanza_error.type_, ErrorType::Cancel);
452 }
453
454 #[test]
455 fn test_gone_with_new_address() {
456 #[cfg(not(feature = "component"))]
457 let elem: Element = "<error xmlns='jabber:client' type='cancel'><gone xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'>xmpp:room@muc.example.org?join</gone></error>"
458 .parse()
459 .unwrap();
460 #[cfg(feature = "component")]
461 let elem: Element = "<error xmlns='jabber:component:accept' type='cancel'><gone xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'>xmpp:room@muc.example.org?join</gone></error>"
462 .parse()
463 .unwrap();
464 let error = StanzaError::try_from(elem).unwrap();
465 assert_eq!(error.type_, ErrorType::Cancel);
466 assert_eq!(
467 error.defined_condition,
468 DefinedCondition::Gone {
469 new_address: Some("xmpp:room@muc.example.org?join".to_string()),
470 }
471 );
472 }
473
474 #[test]
475 fn test_gone_without_new_address() {
476 #[cfg(not(feature = "component"))]
477 let elem: Element = "<error xmlns='jabber:client' type='cancel'><gone xmlns='urn:ietf:params:xml:ns:xmpp-stanzas' /></error>"
478 .parse()
479 .unwrap();
480 #[cfg(feature = "component")]
481 let elem: Element = "<error xmlns='jabber:component:accept' type='cancel'><gone xmlns='urn:ietf:params:xml:ns:xmpp-stanzas' /></error>"
482 .parse()
483 .unwrap();
484 let error = StanzaError::try_from(elem).unwrap();
485 assert_eq!(error.type_, ErrorType::Cancel);
486 assert_eq!(
487 error.defined_condition,
488 DefinedCondition::Gone { new_address: None }
489 );
490 }
491
492 #[test]
493 fn test_redirect_with_alternate_address() {
494 #[cfg(not(feature = "component"))]
495 let elem: Element = "<error xmlns='jabber:client' type='modify'><redirect xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'>xmpp:characters@conference.example.org</redirect></error>"
496 .parse()
497 .unwrap();
498 #[cfg(feature = "component")]
499 let elem: Element = "<error xmlns='jabber:component:accept' type='modify'><redirect xmlns='urn:ietf:params:xml:ns:xmpp-stanzas'>xmpp:characters@conference.example.org</redirect></error>"
500 .parse()
501 .unwrap();
502 let error = StanzaError::try_from(elem).unwrap();
503 assert_eq!(error.type_, ErrorType::Modify);
504 assert_eq!(
505 error.defined_condition,
506 DefinedCondition::Redirect {
507 new_address: Some("xmpp:characters@conference.example.org".to_string()),
508 }
509 );
510 }
511
512 #[test]
513 fn test_redirect_without_alternate_address() {
514 #[cfg(not(feature = "component"))]
515 let elem: Element = "<error xmlns='jabber:client' type='modify'><redirect xmlns='urn:ietf:params:xml:ns:xmpp-stanzas' /></error>"
516 .parse()
517 .unwrap();
518 #[cfg(feature = "component")]
519 let elem: Element = "<error xmlns='jabber:component:accept' type='modify'><redirect xmlns='urn:ietf:params:xml:ns:xmpp-stanzas' /></error>"
520 .parse()
521 .unwrap();
522 let error = StanzaError::try_from(elem).unwrap();
523 assert_eq!(error.type_, ErrorType::Modify);
524 assert_eq!(
525 error.defined_condition,
526 DefinedCondition::Redirect { new_address: None }
527 );
528 }
529}