diff options
| author | 2025-04-01 21:40:23 +0100 | |
|---|---|---|
| committer | 2025-04-01 21:40:23 +0100 | |
| commit | 9ce3827a7d25714d17f266f0f50bb29f41090175 (patch) | |
| tree | b34f4d9c947cf7bc21e90f1a6723fd58801e8e8f | |
| parent | 448208337c3a404403e6b312dbe38555a2bf8ad5 (diff) | |
| download | luz-9ce3827a7d25714d17f266f0f50bb29f41090175.tar.gz luz-9ce3827a7d25714d17f266f0f50bb29f41090175.tar.bz2 luz-9ce3827a7d25714d17f266f0f50bb29f41090175.zip | |
feat(stanza): xep-0131: stanza headers
| -rw-r--r-- | stanza/Cargo.toml | 1 | ||||
| -rw-r--r-- | stanza/src/client/message.rs | 12 | ||||
| -rw-r--r-- | stanza/src/client/presence.rs | 12 | ||||
| -rw-r--r-- | stanza/src/lib.rs | 2 | ||||
| -rw-r--r-- | stanza/src/xep_0131.rs | 56 | 
5 files changed, 83 insertions, 0 deletions
| diff --git a/stanza/Cargo.toml b/stanza/Cargo.toml index ad3bd0b..64962b6 100644 --- a/stanza/Cargo.toml +++ b/stanza/Cargo.toml @@ -15,5 +15,6 @@ xep_0004 = []  xep_0030 = []  xep_0059 = []  xep_0060 = ["xep_0004", "dep:chrono"] +xep_0131 = []  xep_0199 = []  xep_0203 = ["dep:chrono"] diff --git a/stanza/src/client/message.rs b/stanza/src/client/message.rs index 192390b..e521613 100644 --- a/stanza/src/client/message.rs +++ b/stanza/src/client/message.rs @@ -6,6 +6,8 @@ use peanuts::{      DeserializeError, Element, XML_NS,  }; +#[cfg(feature = "xep_0131")] +use crate::xep_0131::Headers;  #[cfg(feature = "xep_0203")]  use crate::xep_0203::Delay; @@ -25,6 +27,8 @@ pub struct Message {      pub thread: Option<Thread>,      #[cfg(feature = "xep_0203")]      pub delay: Option<Delay>, +    #[cfg(feature = "xep_0131")] +    pub headers: Option<Headers>,  }  impl FromElement for Message { @@ -45,6 +49,9 @@ impl FromElement for Message {          #[cfg(feature = "xep_0203")]          let delay = element.child_opt()?; +        #[cfg(feature = "xep_0131")] +        let headers = element.child_opt()?; +          Ok(Message {              from,              id, @@ -56,6 +63,8 @@ impl FromElement for Message {              thread,              #[cfg(feature = "xep_0203")]              delay, +            #[cfg(feature = "xep_0131")] +            headers,          })      }  } @@ -81,6 +90,9 @@ impl IntoElement for Message {          #[cfg(feature = "xep_0203")]          let builder = builder.push_child_opt(self.delay.clone()); +        #[cfg(feature = "xep_0131")] +        let builder = builder.push_child_opt(self.headers.clone()); +          builder      }  } diff --git a/stanza/src/client/presence.rs b/stanza/src/client/presence.rs index ae38756..8fb96be 100644 --- a/stanza/src/client/presence.rs +++ b/stanza/src/client/presence.rs @@ -6,6 +6,8 @@ use peanuts::{      DeserializeError, Element, XML_NS,  }; +#[cfg(feature = "xep_0131")] +use crate::xep_0131::Headers;  #[cfg(feature = "xep_0203")]  use crate::xep_0203::Delay; @@ -24,6 +26,8 @@ pub struct Presence {      pub priority: Option<Priority>,      #[cfg(feature = "xep_0203")]      pub delay: Option<Delay>, +    #[cfg(feature = "xep_0131")] +    pub headers: Option<Headers>,      // TODO: ##other      // other: Vec<Other>,      pub errors: Vec<Error>, @@ -48,6 +52,9 @@ impl FromElement for Presence {          #[cfg(feature = "xep_0203")]          let delay = element.child_opt()?; +        #[cfg(feature = "xep_0131")] +        let headers = element.child_opt()?; +          Ok(Presence {              from,              id, @@ -60,6 +67,8 @@ impl FromElement for Presence {              errors,              #[cfg(feature = "xep_0203")]              delay, +            #[cfg(feature = "xep_0131")] +            headers,          })      }  } @@ -80,6 +89,9 @@ impl IntoElement for Presence {          #[cfg(feature = "xep_0203")]          let builder = builder.push_child_opt(self.delay.clone()); +        #[cfg(feature = "xep_0131")] +        let builder = builder.push_child_opt(self.headers.clone()); +          builder      }  } diff --git a/stanza/src/lib.rs b/stanza/src/lib.rs index 0bae7f7..0a71a26 100644 --- a/stanza/src/lib.rs +++ b/stanza/src/lib.rs @@ -17,6 +17,8 @@ pub mod xep_0030;  pub mod xep_0059;  #[cfg(feature = "xep_0060")]  pub mod xep_0060; +#[cfg(feature = "xep_0131")] +pub mod xep_0131;  #[cfg(feature = "xep_0199")]  pub mod xep_0199;  #[cfg(feature = "xep_0203")] diff --git a/stanza/src/xep_0131.rs b/stanza/src/xep_0131.rs new file mode 100644 index 0000000..27fc962 --- /dev/null +++ b/stanza/src/xep_0131.rs @@ -0,0 +1,56 @@ +use peanuts::{ +    element::{FromElement, IntoElement}, +    Element, +}; + +pub const XMLNS: &str = "http://jabber.org/protocol/disco#info"; + +#[derive(Clone, Debug)] +pub struct Headers(pub Vec<Header>); + +impl FromElement for Headers { +    fn from_element(mut element: Element) -> peanuts::element::DeserializeResult<Self> { +        element.check_name("headers")?; +        element.check_namespace(XMLNS)?; + +        Ok(Self(element.pop_children()?)) +    } +} + +impl IntoElement for Headers { +    fn builder(&self) -> peanuts::element::ElementBuilder { +        Element::builder("headers", Some(XMLNS)).push_children(self.0.clone()) +    } +} + +#[derive(Clone, Debug)] +pub struct Header { +    name: String, +    header: String, +} + +impl FromElement for Header { +    fn from_element(mut element: peanuts::Element) -> peanuts::element::DeserializeResult<Self> { +        element.check_name("header")?; +        element.check_namespace(XMLNS)?; + +        let name = element.attribute("name")?; + +        let header = element.pop_value_opt()?.unwrap_or_default(); + +        Ok(Self { name, header }) +    } +} + +impl IntoElement for Header { +    fn builder(&self) -> peanuts::element::ElementBuilder { +        let builder = +            Element::builder("header", Some(XMLNS)).push_attribute("name", self.name.clone()); + +        if self.header.is_empty() { +            builder +        } else { +            builder.push_text(self.header.clone()) +        } +    } +} | 
