aboutsummaryrefslogtreecommitdiffstats
path: root/jid
diff options
context:
space:
mode:
authorLibravatar cel 🌸 <cel@bunny.garden>2024-12-04 18:18:37 +0000
committerLibravatar cel 🌸 <cel@bunny.garden>2024-12-04 18:18:37 +0000
commit1b91ff690488b65b552c90bd5392b9a300c8c981 (patch)
tree9c290f69b26eba0393d7bbc05ba29c28ea74a26e /jid
parent03764f8cedb3f0a55a61be0f0a59faaa6357a83a (diff)
downloadluz-1b91ff690488b65b552c90bd5392b9a300c8c981.tar.gz
luz-1b91ff690488b65b552c90bd5392b9a300c8c981.tar.bz2
luz-1b91ff690488b65b552c90bd5392b9a300c8c981.zip
use cargo workspace
Diffstat (limited to 'jid')
-rw-r--r--jid/Cargo.toml6
-rw-r--r--jid/src/lib.rs170
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()))
+ )
+ }
+}