parsers: port stanza_error over to derive macros

Jonas Schäfer created

Change summary

parsers/ChangeLog           |  1 
parsers/src/stanza_error.rs | 99 +++++++++-----------------------------
2 files changed, 24 insertions(+), 76 deletions(-)

Detailed changes

parsers/ChangeLog 🔗

@@ -67,6 +67,7 @@ XXXX-YY-ZZ RELEASER <admin@example.com>
       - time::TimeResult has been ported to use xso. Use From/Into to convert
         it to/from chrono::DateTime values. The numbered member `0` does not
         exist anymore (!551).
+      - stanza_error::StanzaError has been ported to use xso (!552).
     * New parsers/serialisers:
       - Stream Features (RFC 6120) (!400)
       - Spam Reporting (XEP-0377) (!506)

parsers/src/stanza_error.rs 🔗

@@ -10,10 +10,8 @@ use crate::message::MessagePayload;
 use crate::ns;
 use crate::presence::PresencePayload;
 use alloc::collections::BTreeMap;
-use core::convert::TryFrom;
 use jid::Jid;
 use minidom::Element;
-use xso::error::{Error, FromElementError};
 
 generate_attribute!(
     /// The type of the error.
@@ -36,8 +34,10 @@ generate_attribute!(
 );
 
 /// List of valid error conditions.
+// NOTE: This MUST NOT be marked as exhaustive, because the <text/> elements
+// use the same namespace!
 #[derive(FromXml, AsXml, PartialEq, Debug, Clone)]
-#[xml(namespace = ns::XMPP_STANZAS, exhaustive)]
+#[xml(namespace = ns::XMPP_STANZAS)]
 pub enum DefinedCondition {
     /// The sender has sent a stanza containing XML that does not conform
     /// to the appropriate schema or that cannot be processed (e.g., an IQ
@@ -228,21 +228,30 @@ pub enum DefinedCondition {
 type Lang = String;
 
 /// The representation of a stanza error.
-#[derive(Debug, Clone, PartialEq)]
+#[derive(Debug, Clone, PartialEq, FromXml, AsXml)]
+#[xml(namespace = ns::DEFAULT_NS, name = "error", discard(attribute = "code"))]
 pub struct StanzaError {
     /// The type of this error.
+    #[xml(attribute = "type")]
     pub type_: ErrorType,
 
     /// The JID of the entity who set this error.
+    #[xml(attribute(name = "by", default))]
     pub by: Option<Jid>,
 
     /// One of the defined conditions for this error to happen.
+    #[xml(child)]
     pub defined_condition: DefinedCondition,
 
     /// Human-readable description of this error.
+    #[xml(extract(n = .., namespace = ns::XMPP_STANZAS, name = "text", fields(
+        attribute(name = "xml:lang", type_ = Lang, default),
+        text(type_ = String),
+    )))]
     pub texts: BTreeMap<Lang, String>,
 
     /// A protocol-specific extension for this error.
+    #[xml(element(default))]
     pub other: Option<Element>,
 }
 
@@ -275,80 +284,12 @@ impl StanzaError {
     }
 }
 
-impl TryFrom<Element> for StanzaError {
-    type Error = FromElementError;
-
-    fn try_from(elem: Element) -> Result<StanzaError, FromElementError> {
-        check_self!(elem, "error", DEFAULT_NS);
-        // The code attribute has been deprecated in [XEP-0086](https://xmpp.org/extensions/xep-0086.html)
-        // which was deprecated in 2007. We don't error when it's here, but don't include it in the final struct.
-        check_no_unknown_attributes!(elem, "error", ["type", "by", "code"]);
-
-        let mut stanza_error = StanzaError {
-            type_: get_attr!(elem, "type", Required),
-            by: get_attr!(elem, "by", Option),
-            defined_condition: DefinedCondition::UndefinedCondition,
-            texts: BTreeMap::new(),
-            other: None,
-        };
-        let mut defined_condition = None;
-
-        for child in elem.children() {
-            if child.is("text", ns::XMPP_STANZAS) {
-                check_no_children!(child, "text");
-                check_no_unknown_attributes!(child, "text", ["xml:lang"]);
-                let lang = get_attr!(child, "xml:lang", Default);
-                if stanza_error.texts.insert(lang, child.text()).is_some() {
-                    return Err(
-                        Error::Other("Text element present twice for the same xml:lang.").into(),
-                    );
-                }
-            } else if child.has_ns(ns::XMPP_STANZAS) {
-                if defined_condition.is_some() {
-                    return Err(Error::Other(
-                        "Error must not have more than one defined-condition.",
-                    )
-                    .into());
-                }
-                check_no_children!(child, "defined-condition");
-                check_no_attributes!(child, "defined-condition");
-                defined_condition = Some(DefinedCondition::try_from(child.clone())?);
-            } else {
-                if stanza_error.other.is_some() {
-                    return Err(
-                        Error::Other("Error must not have more than one other element.").into(),
-                    );
-                }
-                stanza_error.other = Some(child.clone());
-            }
-        }
-        stanza_error.defined_condition =
-            defined_condition.ok_or(Error::Other("Error must have a defined-condition."))?;
-
-        Ok(stanza_error)
-    }
-}
-
-impl From<StanzaError> for Element {
-    fn from(err: StanzaError) -> Element {
-        Element::builder("error", ns::DEFAULT_NS)
-            .attr("type", err.type_)
-            .attr("by", err.by)
-            .append(err.defined_condition)
-            .append_all(err.texts.into_iter().map(|(lang, text)| {
-                Element::builder("text", ns::XMPP_STANZAS)
-                    .attr("xml:lang", lang)
-                    .append(text)
-            }))
-            .append_all(err.other)
-            .build()
-    }
-}
-
 #[cfg(test)]
 mod tests {
     use super::*;
 
+    use xso::error::{Error, FromElementError};
+
     #[cfg(target_pointer_width = "32")]
     #[test]
     fn test_size() {
@@ -390,7 +331,10 @@ mod tests {
             FromElementError::Invalid(Error::Other(string)) => string,
             _ => panic!(),
         };
-        assert_eq!(message, "Required attribute 'type' missing.");
+        assert_eq!(
+            message,
+            "Required attribute field 'type_' on StanzaError element missing."
+        );
 
         #[cfg(not(feature = "component"))]
         let elem: Element = "<error xmlns='jabber:client' type='coucou'/>"
@@ -423,7 +367,10 @@ mod tests {
             FromElementError::Invalid(Error::Other(string)) => string,
             _ => panic!(),
         };
-        assert_eq!(message, "Error must have a defined-condition.");
+        assert_eq!(
+            message,
+            "Missing child field 'defined_condition' in StanzaError element."
+        );
     }
 
     #[test]