disco: Split query and result.

Emmanuel Gil Peyrot created

Change summary

src/caps.rs   | 17 +++++-------
src/disco.rs  | 69 ++++++++++++++++++++++++++++++++++++++--------------
src/ecaps2.rs | 17 +++++-------
src/iq.rs     | 19 ++++++++++---
4 files changed, 78 insertions(+), 44 deletions(-)

Detailed changes

src/caps.rs 🔗

@@ -6,7 +6,7 @@
 
 use std::convert::TryFrom;
 
-use disco::{Feature, Identity, Disco};
+use disco::{Feature, Identity, DiscoInfoResult, DiscoInfoQuery};
 use data_forms::DataForm;
 use hashes::{Hash, Algo};
 
@@ -122,7 +122,7 @@ fn compute_extensions(extensions: &[DataForm]) -> Vec<u8> {
     })
 }
 
-pub fn compute_disco(disco: &Disco) -> Vec<u8> {
+pub fn compute_disco(disco: &DiscoInfoResult) -> Vec<u8> {
     let identities_string = compute_identities(&disco.identities);
     let features_string = compute_features(&disco.features);
     let extensions_string = compute_extensions(&disco.extensions);
@@ -193,12 +193,9 @@ pub fn hash_caps(data: &[u8], algo: Algo) -> Result<Hash, String> {
     })
 }
 
-pub fn query_caps(caps: Caps) -> Disco {
-    Disco {
+pub fn query_caps(caps: Caps) -> DiscoInfoQuery {
+    DiscoInfoQuery {
         node: Some(format!("{}#{}", caps.node, base64::encode(&caps.hash.hash))),
-        identities: vec!(),
-        features: vec!(),
-        extensions: vec!(),
     }
 }
 
@@ -231,7 +228,7 @@ mod tests {
     #[test]
     fn test_simple() {
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/><feature var='http://jabber.org/protocol/disco#info'/></query>".parse().unwrap();
-        let disco = Disco::try_from(elem).unwrap();
+        let disco = DiscoInfoResult::try_from(elem).unwrap();
         let caps = caps::compute_disco(&disco);
         assert_eq!(caps.len(), 50);
     }
@@ -252,7 +249,7 @@ mod tests {
         let data = b"client/pc//Exodus 0.9.1<http://jabber.org/protocol/caps<http://jabber.org/protocol/disco#info<http://jabber.org/protocol/disco#items<http://jabber.org/protocol/muc<";
         let mut expected = Vec::with_capacity(data.len());
         expected.extend_from_slice(data);
-        let disco = Disco::try_from(elem).unwrap();
+        let disco = DiscoInfoResult::try_from(elem).unwrap();
         let caps = caps::compute_disco(&disco);
         assert_eq!(caps, expected);
 
@@ -297,7 +294,7 @@ mod tests {
         let data = b"client/pc/el/\xce\xa8 0.11<client/pc/en/Psi 0.11<http://jabber.org/protocol/caps<http://jabber.org/protocol/disco#info<http://jabber.org/protocol/disco#items<http://jabber.org/protocol/muc<urn:xmpp:dataforms:softwareinfo<ip_version<ipv4<ipv6<os<Mac<os_version<10.5.1<software<Psi<software_version<0.11<";
         let mut expected = Vec::with_capacity(data.len());
         expected.extend_from_slice(data);
-        let disco = Disco::try_from(elem).unwrap();
+        let disco = DiscoInfoResult::try_from(elem).unwrap();
         let caps = caps::compute_disco(&disco);
         assert_eq!(caps, expected);
 

src/disco.rs 🔗

@@ -13,6 +13,41 @@ use ns;
 
 use data_forms::{DataForm, DataFormType};
 
+#[derive(Debug, Clone)]
+pub struct DiscoInfoQuery {
+    pub node: Option<String>,
+}
+
+impl TryFrom<Element> for DiscoInfoQuery {
+    type Error = Error;
+
+    fn try_from(elem: Element) -> Result<DiscoInfoQuery, Error> {
+        if !elem.is("query", ns::DISCO_INFO) {
+            return Err(Error::ParseError("This is not a disco#info element."));
+        }
+        for _ in elem.children() {
+            return Err(Error::ParseError("Unknown child in disco#info."));
+        }
+        for (attr, _) in elem.attrs() {
+            if attr != "node" {
+                return Err(Error::ParseError("Unknown attribute in disco#info."));
+            }
+        }
+        Ok(DiscoInfoQuery {
+            node: get_attr!(elem, "node", optional),
+        })
+    }
+}
+
+impl From<DiscoInfoQuery> for Element {
+    fn from(disco: DiscoInfoQuery) -> Element {
+        Element::builder("query")
+                .ns(ns::DISCO_INFO)
+                .attr("node", disco.node)
+                .build()
+    }
+}
+
 #[derive(Debug, Clone, PartialEq)]
 pub struct Feature {
     pub var: String,
@@ -60,17 +95,17 @@ impl IntoElements for Identity {
 }
 
 #[derive(Debug, Clone)]
-pub struct Disco {
+pub struct DiscoInfoResult {
     pub node: Option<String>,
     pub identities: Vec<Identity>,
     pub features: Vec<Feature>,
     pub extensions: Vec<DataForm>,
 }
 
-impl TryFrom<Element> for Disco {
+impl TryFrom<Element> for DiscoInfoResult {
     type Error = Error;
 
-    fn try_from(elem: Element) -> Result<Disco, Error> {
+    fn try_from(elem: Element) -> Result<DiscoInfoResult, Error> {
         if !elem.is("query", ns::DISCO_INFO) {
             return Err(Error::ParseError("This is not a disco#info element."));
         }
@@ -120,8 +155,6 @@ impl TryFrom<Element> for Disco {
             }
         }
 
-        /*
-        // TODO: encode these restrictions only for result disco#info, not get ones.
         if identities.is_empty() {
             return Err(Error::ParseError("There must be at least one identity in disco#info."));
         }
@@ -131,9 +164,8 @@ impl TryFrom<Element> for Disco {
         if !features.contains(&Feature { var: ns::DISCO_INFO.to_owned() }) {
             return Err(Error::ParseError("disco#info feature not present in disco#info."));
         }
-        */
 
-        Ok(Disco {
+        Ok(DiscoInfoResult {
             node: node,
             identities: identities,
             features: features,
@@ -142,7 +174,7 @@ impl TryFrom<Element> for Disco {
     }
 }
 
-impl Into<Element> for Disco {
+impl Into<Element> for DiscoInfoResult {
     fn into(self) -> Element {
         for _ in self.extensions {
             panic!("Not yet implemented!");
@@ -163,7 +195,7 @@ mod tests {
     #[test]
     fn test_simple() {
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/><feature var='http://jabber.org/protocol/disco#info'/></query>".parse().unwrap();
-        let query = Disco::try_from(elem).unwrap();
+        let query = DiscoInfoResult::try_from(elem).unwrap();
         assert!(query.node.is_none());
         assert_eq!(query.identities.len(), 1);
         assert_eq!(query.features.len(), 1);
@@ -173,7 +205,7 @@ mod tests {
     #[test]
     fn test_invalid() {
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><coucou/></query>".parse().unwrap();
-        let error = Disco::try_from(elem).unwrap_err();
+        let error = DiscoInfoResult::try_from(elem).unwrap_err();
         let message = match error {
             Error::ParseError(string) => string,
             _ => panic!(),
@@ -184,7 +216,7 @@ mod tests {
     #[test]
     fn test_invalid_identity() {
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity/></query>".parse().unwrap();
-        let error = Disco::try_from(elem).unwrap_err();
+        let error = DiscoInfoResult::try_from(elem).unwrap_err();
         let message = match error {
             Error::ParseError(string) => string,
             _ => panic!(),
@@ -192,7 +224,7 @@ mod tests {
         assert_eq!(message, "Required attribute 'category' missing.");
 
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category=''/></query>".parse().unwrap();
-        let error = Disco::try_from(elem).unwrap_err();
+        let error = DiscoInfoResult::try_from(elem).unwrap_err();
         let message = match error {
             Error::ParseError(string) => string,
             _ => panic!(),
@@ -200,7 +232,7 @@ mod tests {
         assert_eq!(message, "Identity must have a non-empty 'category' attribute.");
 
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='coucou'/></query>".parse().unwrap();
-        let error = Disco::try_from(elem).unwrap_err();
+        let error = DiscoInfoResult::try_from(elem).unwrap_err();
         let message = match error {
             Error::ParseError(string) => string,
             _ => panic!(),
@@ -208,7 +240,7 @@ mod tests {
         assert_eq!(message, "Required attribute 'type' missing.");
 
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='coucou' type=''/></query>".parse().unwrap();
-        let error = Disco::try_from(elem).unwrap_err();
+        let error = DiscoInfoResult::try_from(elem).unwrap_err();
         let message = match error {
             Error::ParseError(string) => string,
             _ => panic!(),
@@ -219,7 +251,7 @@ mod tests {
     #[test]
     fn test_invalid_feature() {
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><feature/></query>".parse().unwrap();
-        let error = Disco::try_from(elem).unwrap_err();
+        let error = DiscoInfoResult::try_from(elem).unwrap_err();
         let message = match error {
             Error::ParseError(string) => string,
             _ => panic!(),
@@ -228,10 +260,9 @@ mod tests {
     }
 
     #[test]
-    #[ignore]
     fn test_invalid_result() {
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'/>".parse().unwrap();
-        let error = Disco::try_from(elem).unwrap_err();
+        let error = DiscoInfoResult::try_from(elem).unwrap_err();
         let message = match error {
             Error::ParseError(string) => string,
             _ => panic!(),
@@ -239,7 +270,7 @@ mod tests {
         assert_eq!(message, "There must be at least one identity in disco#info.");
 
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/></query>".parse().unwrap();
-        let error = Disco::try_from(elem).unwrap_err();
+        let error = DiscoInfoResult::try_from(elem).unwrap_err();
         let message = match error {
             Error::ParseError(string) => string,
             _ => panic!(),
@@ -247,7 +278,7 @@ mod tests {
         assert_eq!(message, "There must be at least one feature in disco#info.");
 
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/><feature var='http://jabber.org/protocol/disco#items'/></query>".parse().unwrap();
-        let error = Disco::try_from(elem).unwrap_err();
+        let error = DiscoInfoResult::try_from(elem).unwrap_err();
         let message = match error {
             Error::ParseError(string) => string,
             _ => panic!(),

src/ecaps2.rs 🔗

@@ -6,7 +6,7 @@
 
 use std::convert::TryFrom;
 
-use disco::{Feature, Identity, Disco};
+use disco::{Feature, Identity, DiscoInfoResult, DiscoInfoQuery};
 use data_forms::DataForm;
 use hashes::{Hash, Algo};
 
@@ -106,7 +106,7 @@ fn compute_extensions(extensions: &[DataForm]) -> Vec<u8> {
     })
 }
 
-pub fn compute_disco(disco: &Disco) -> Vec<u8> {
+pub fn compute_disco(disco: &DiscoInfoResult) -> Vec<u8> {
     let features_string = compute_features(&disco.features);
     let identities_string = compute_identities(&disco.identities);
     let extensions_string = compute_extensions(&disco.extensions);
@@ -172,12 +172,9 @@ pub fn hash_ecaps2(data: &[u8], algo: Algo) -> Result<Hash, String> {
     })
 }
 
-pub fn query_ecaps2(hash: Hash) -> Disco {
-    Disco {
+pub fn query_ecaps2(hash: Hash) -> DiscoInfoQuery {
+    DiscoInfoQuery {
         node: Some(format!("{}#{}.{}", ns::ECAPS2, String::from(hash.algo), base64::encode(&hash.hash))),
-        identities: vec!(),
-        features: vec!(),
-        extensions: vec!(),
     }
 }
 
@@ -211,7 +208,7 @@ mod tests {
     #[test]
     fn test_simple() {
         let elem: Element = "<query xmlns='http://jabber.org/protocol/disco#info'><identity category='client' type='pc'/><feature var='http://jabber.org/protocol/disco#info'/></query>".parse().unwrap();
-        let disco = Disco::try_from(elem).unwrap();
+        let disco = DiscoInfoResult::try_from(elem).unwrap();
         let ecaps2 = ecaps2::compute_disco(&disco);
         assert_eq!(ecaps2.len(), 54);
     }
@@ -274,7 +271,7 @@ mod tests {
             105, 109, 101, 31, 28, 99, 108, 105, 101, 110, 116, 31, 109, 111,
             98, 105, 108, 101, 31, 31, 66, 111, 109, 98, 117, 115, 77, 111,
             100, 31, 30, 28, 28];
-        let disco = Disco::try_from(elem).unwrap();
+        let disco = DiscoInfoResult::try_from(elem).unwrap();
         let ecaps2 = ecaps2::compute_disco(&disco);
         assert_eq!(ecaps2.len(), 0x1d9);
         assert_eq!(ecaps2, expected);
@@ -446,7 +443,7 @@ mod tests {
             111, 110, 31, 48, 46, 49, 49, 46, 49, 45, 115, 118, 110, 45, 50,
             48, 49, 49, 49, 50, 49, 54, 45, 109, 111, 100, 32, 40, 84, 99, 108,
             47, 84, 107, 32, 56, 46,54, 98, 50, 41, 31, 30, 29, 28];
-        let disco = Disco::try_from(elem).unwrap();
+        let disco = DiscoInfoResult::try_from(elem).unwrap();
         let ecaps2 = ecaps2::compute_disco(&disco);
         assert_eq!(ecaps2.len(), 0x543);
         assert_eq!(ecaps2, expected);

src/iq.rs 🔗

@@ -18,7 +18,7 @@ use ns;
 
 use stanza_error::StanzaError;
 use roster::Roster;
-use disco::Disco;
+use disco::{DiscoInfoResult, DiscoInfoQuery};
 use ibb::IBB;
 use jingle::Jingle;
 use ping::Ping;
@@ -28,7 +28,8 @@ use mam::{Query as MamQuery, Fin as MamFin, Prefs as MamPrefs};
 #[derive(Debug, Clone)]
 pub enum IqPayload {
     Roster(Roster),
-    Disco(Disco),
+    DiscoInfoResult(DiscoInfoResult),
+    DiscoInfoQuery(DiscoInfoQuery),
     IBB(IBB),
     Jingle(Jingle),
     Ping(Ping),
@@ -48,7 +49,14 @@ impl TryFrom<Element> for IqPayload {
             ("query", ns::ROSTER) => IqPayload::Roster(Roster::try_from(elem)?),
 
             // XEP-0030
-            ("query", ns::DISCO_INFO) => IqPayload::Disco(Disco::try_from(elem)?),
+            ("query", ns::DISCO_INFO) => {
+                // TODO: separate all three types of payloads.
+                match DiscoInfoQuery::try_from(elem.clone()) {
+                    Ok(payload) => IqPayload::DiscoInfoQuery(payload),
+                    // TODO: put that error somewhere too?
+                    Err(_) => IqPayload::DiscoInfoResult(DiscoInfoResult::try_from(elem)?),
+                }
+            },
 
             // XEP-0047
             ("open", ns::IBB)
@@ -171,7 +179,8 @@ impl Into<Element> for IqPayload {
     fn into(self) -> Element {
         match self {
             IqPayload::Roster(roster) => roster.into(),
-            IqPayload::Disco(disco) => disco.into(),
+            IqPayload::DiscoInfoResult(disco) => disco.into(),
+            IqPayload::DiscoInfoQuery(disco) => disco.into(),
             IqPayload::IBB(ibb) => ibb.into(),
             IqPayload::Jingle(jingle) => jingle.into(),
             IqPayload::Ping(ping) => ping.into(),
@@ -339,7 +348,7 @@ mod tests {
             _ => panic!(),
         };
         assert!(match payload {
-            IqPayload::Disco(Disco { .. }) => true,
+            IqPayload::DiscoInfoQuery(DiscoInfoQuery { .. }) => true,
             _ => false,
         });
     }