diff options
author | 2024-12-04 18:18:37 +0000 | |
---|---|---|
committer | 2024-12-04 18:18:37 +0000 | |
commit | 1b91ff690488b65b552c90bd5392b9a300c8c981 (patch) | |
tree | 9c290f69b26eba0393d7bbc05ba29c28ea74a26e /jid | |
parent | 03764f8cedb3f0a55a61be0f0a59faaa6357a83a (diff) | |
download | luz-1b91ff690488b65b552c90bd5392b9a300c8c981.tar.gz luz-1b91ff690488b65b552c90bd5392b9a300c8c981.tar.bz2 luz-1b91ff690488b65b552c90bd5392b9a300c8c981.zip |
use cargo workspace
Diffstat (limited to 'jid')
-rw-r--r-- | jid/Cargo.toml | 6 | ||||
-rw-r--r-- | jid/src/lib.rs | 170 |
2 files changed, 176 insertions, 0 deletions
diff --git a/jid/Cargo.toml b/jid/Cargo.toml new file mode 100644 index 0000000..15049c9 --- /dev/null +++ b/jid/Cargo.toml @@ -0,0 +1,6 @@ +[package] +name = "jid" +version = "0.1.0" +edition = "2021" + +[dependencies] diff --git a/jid/src/lib.rs b/jid/src/lib.rs new file mode 100644 index 0000000..ff1d82b --- /dev/null +++ b/jid/src/lib.rs @@ -0,0 +1,170 @@ +use std::str::FromStr; + +#[derive(PartialEq, Debug, Clone)] +pub struct JID { + // TODO: validate localpart (length, char] + pub localpart: Option<String>, + pub domainpart: String, + pub resourcepart: Option<String>, +} + +pub enum JIDError { + NoResourcePart, + ParseError(ParseError), +} + +#[derive(Debug)] +pub enum ParseError { + Empty, + Malformed(String), +} + +impl JID { + pub fn new( + localpart: Option<String>, + domainpart: String, + resourcepart: Option<String>, + ) -> Self { + Self { + localpart, + domainpart: domainpart.parse().unwrap(), + resourcepart, + } + } + + pub fn as_bare(&self) -> Self { + Self { + localpart: self.localpart.clone(), + domainpart: self.domainpart.clone(), + resourcepart: None, + } + } + + pub fn as_full(&self) -> Result<&Self, JIDError> { + if let Some(_) = self.resourcepart { + Ok(&self) + } else { + Err(JIDError::NoResourcePart) + } + } +} + +impl FromStr for JID { + type Err = ParseError; + + fn from_str(s: &str) -> Result<Self, Self::Err> { + let split: Vec<&str> = s.split('@').collect(); + match split.len() { + 0 => Err(ParseError::Empty), + 1 => { + let split: Vec<&str> = split[0].split('/').collect(); + match split.len() { + 1 => Ok(JID::new(None, split[0].to_string(), None)), + 2 => Ok(JID::new( + None, + split[0].to_string(), + Some(split[1].to_string()), + )), + _ => Err(ParseError::Malformed(s.to_string())), + } + } + 2 => { + let split2: Vec<&str> = split[1].split('/').collect(); + match split2.len() { + 1 => Ok(JID::new( + Some(split[0].to_string()), + split2[0].to_string(), + None, + )), + 2 => Ok(JID::new( + Some(split[0].to_string()), + split2[0].to_string(), + Some(split2[1].to_string()), + )), + _ => Err(ParseError::Malformed(s.to_string())), + } + } + _ => Err(ParseError::Malformed(s.to_string())), + } + } +} + +impl TryFrom<String> for JID { + type Error = ParseError; + + fn try_from(value: String) -> Result<Self, Self::Error> { + value.parse() + } +} + +impl TryFrom<&str> for JID { + type Error = ParseError; + + fn try_from(value: &str) -> Result<Self, Self::Error> { + value.parse() + } +} + +impl std::fmt::Display for JID { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + write!( + f, + "{}{}{}", + self.localpart.clone().map(|l| l + "@").unwrap_or_default(), + self.domainpart, + self.resourcepart + .clone() + .map(|r| "/".to_owned() + &r) + .unwrap_or_default() + ) + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn jid_to_string() { + assert_eq!( + JID::new(Some("cel".into()), "blos.sm".into(), None).to_string(), + "cel@blos.sm".to_owned() + ); + } + + #[test] + fn parse_full_jid() { + assert_eq!( + "cel@blos.sm/greenhouse".parse::<JID>().unwrap(), + JID::new( + Some("cel".into()), + "blos.sm".into(), + Some("greenhouse".into()) + ) + ) + } + + #[test] + fn parse_bare_jid() { + assert_eq!( + "cel@blos.sm".parse::<JID>().unwrap(), + JID::new(Some("cel".into()), "blos.sm".into(), None) + ) + } + + #[test] + fn parse_domain_jid() { + assert_eq!( + "component.blos.sm".parse::<JID>().unwrap(), + JID::new(None, "component.blos.sm".into(), None) + ) + } + + #[test] + fn parse_full_domain_jid() { + assert_eq!( + "component.blos.sm/bot".parse::<JID>().unwrap(), + JID::new(None, "component.blos.sm".into(), Some("bot".into())) + ) + } +} |