#![no_std]
#![deny(missing_docs)]
#![cfg_attr(docsrs, feature(doc_auto_cfg))]
extern crate alloc;
#[cfg(any(feature = "std", test))]
extern crate std;
use alloc::borrow::Cow;
use alloc::format;
use alloc::string::{String, ToString};
use core::borrow::Borrow;
use core::cmp::Ordering;
use core::fmt;
use core::hash::{Hash, Hasher};
use core::mem;
use core::num::NonZeroU16;
use core::ops::Deref;
use core::str::FromStr;
use memchr::memchr;
use stringprep::{nameprep, nodeprep, resourceprep};
#[cfg(feature = "serde")]
use serde::{de, Deserialize, Deserializer, Serialize, Serializer};
#[cfg(feature = "quote")]
use proc_macro2::TokenStream;
#[cfg(feature = "quote")]
use quote::{quote, ToTokens};
#[cfg(feature = "minidom")]
use minidom::{IntoAttributeValue, Node};
mod error;
pub use crate::error::Error;
mod parts;
pub use parts::{DomainPart, DomainRef, NodePart, NodeRef, ResourcePart, ResourceRef};
fn length_check(len: usize, error_empty: Error, error_too_long: Error) -> Result<(), Error> {
if len == 0 {
Err(error_empty)
} else if len > 1023 {
Err(error_too_long)
} else {
Ok(())
}
}
#[derive(Clone, Eq)]
pub struct Jid {
normalized: String,
at: Option<NonZeroU16>,
slash: Option<NonZeroU16>,
}
impl PartialEq for Jid {
fn eq(&self, other: &Jid) -> bool {
self.normalized == other.normalized
}
}
impl PartialOrd for Jid {
fn partial_cmp(&self, other: &Jid) -> Option<Ordering> {
Some(self.cmp(other))
}
}
impl Ord for Jid {
fn cmp(&self, other: &Jid) -> Ordering {
self.normalized.cmp(&other.normalized)
}
}
impl Hash for Jid {
fn hash<H: Hasher>(&self, state: &mut H) {
self.normalized.hash(state)
}
}
impl FromStr for Jid {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
Self::new(s)
}
}
impl From<BareJid> for Jid {
fn from(other: BareJid) -> Self {
other.inner
}
}
impl From<FullJid> for Jid {
fn from(other: FullJid) -> Self {
other.inner
}
}
impl fmt::Display for Jid {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
fmt.write_str(&self.normalized)
}
}
impl Jid {
pub fn new(unnormalized: &str) -> Result<Jid, Error> {
let bytes = unnormalized.as_bytes();
let mut orig_at = memchr(b'@', bytes);
let mut orig_slash = memchr(b'/', bytes);
if orig_at.is_some() && orig_slash.is_some() && orig_at > orig_slash {
orig_at = None;
}
let normalized = match (orig_at, orig_slash) {
(Some(at), Some(slash)) => {
let node = nodeprep(&unnormalized[..at]).map_err(|_| Error::NodePrep)?;
length_check(node.len(), Error::NodeEmpty, Error::NodeTooLong)?;
let domain = nameprep(&unnormalized[at + 1..slash]).map_err(|_| Error::NamePrep)?;
length_check(domain.len(), Error::DomainEmpty, Error::DomainTooLong)?;
let resource =
resourceprep(&unnormalized[slash + 1..]).map_err(|_| Error::ResourcePrep)?;
length_check(resource.len(), Error::ResourceEmpty, Error::ResourceTooLong)?;
orig_at = Some(node.len());
orig_slash = Some(node.len() + domain.len() + 1);
match (node, domain, resource) {
(Cow::Borrowed(_), Cow::Borrowed(_), Cow::Borrowed(_)) => {
unnormalized.to_string()
}
(node, domain, resource) => format!("{node}@{domain}/{resource}"),
}
}
(Some(at), None) => {
let node = nodeprep(&unnormalized[..at]).map_err(|_| Error::NodePrep)?;
length_check(node.len(), Error::NodeEmpty, Error::NodeTooLong)?;
let domain = nameprep(&unnormalized[at + 1..]).map_err(|_| Error::NamePrep)?;
length_check(domain.len(), Error::DomainEmpty, Error::DomainTooLong)?;
orig_at = Some(node.len());
match (node, domain) {
(Cow::Borrowed(_), Cow::Borrowed(_)) => unnormalized.to_string(),
(node, domain) => format!("{node}@{domain}"),
}
}
(None, Some(slash)) => {
let domain = nameprep(&unnormalized[..slash]).map_err(|_| Error::NamePrep)?;
length_check(domain.len(), Error::DomainEmpty, Error::DomainTooLong)?;
let resource =
resourceprep(&unnormalized[slash + 1..]).map_err(|_| Error::ResourcePrep)?;
length_check(resource.len(), Error::ResourceEmpty, Error::ResourceTooLong)?;
orig_slash = Some(domain.len());
match (domain, resource) {
(Cow::Borrowed(_), Cow::Borrowed(_)) => unnormalized.to_string(),
(domain, resource) => format!("{domain}/{resource}"),
}
}
(None, None) => {
let domain = nameprep(unnormalized).map_err(|_| Error::NamePrep)?;
length_check(domain.len(), Error::DomainEmpty, Error::DomainTooLong)?;
domain.into_owned()
}
};
Ok(Self {
normalized,
at: orig_at.and_then(|x| NonZeroU16::new(x as u16)),
slash: orig_slash.and_then(|x| NonZeroU16::new(x as u16)),
})
}
pub fn into_inner(self) -> String {
self.normalized
}
pub fn from_parts(
node: Option<&NodeRef>,
domain: &DomainRef,
resource: Option<&ResourceRef>,
) -> Self {
match resource {
Some(resource) => FullJid::from_parts(node, domain, resource).into(),
None => BareJid::from_parts(node, domain).into(),
}
}
pub fn node(&self) -> Option<&NodeRef> {
self.at.map(|at| {
let at = u16::from(at) as usize;
NodeRef::from_str_unchecked(&self.normalized[..at])
})
}
pub fn domain(&self) -> &DomainRef {
match (self.at, self.slash) {
(Some(at), Some(slash)) => {
let at = u16::from(at) as usize;
let slash = u16::from(slash) as usize;
DomainRef::from_str_unchecked(&self.normalized[at + 1..slash])
}
(Some(at), None) => {
let at = u16::from(at) as usize;
DomainRef::from_str_unchecked(&self.normalized[at + 1..])
}
(None, Some(slash)) => {
let slash = u16::from(slash) as usize;
DomainRef::from_str_unchecked(&self.normalized[..slash])
}
(None, None) => DomainRef::from_str_unchecked(&self.normalized),
}
}
pub fn resource(&self) -> Option<&ResourceRef> {
self.slash.map(|slash| {
let slash = u16::from(slash) as usize;
ResourceRef::from_str_unchecked(&self.normalized[slash + 1..])
})
}
pub fn to_bare(&self) -> BareJid {
BareJid::from_parts(self.node(), self.domain())
}
pub fn into_bare(mut self) -> BareJid {
if let Some(slash) = self.slash {
self.normalized.truncate(slash.get() as usize);
self.slash = None;
}
BareJid { inner: self }
}
pub fn is_full(&self) -> bool {
self.slash.is_some()
}
pub fn is_bare(&self) -> bool {
self.slash.is_none()
}
pub fn as_str(&self) -> &str {
&self.normalized
}
pub fn try_into_full(self) -> Result<FullJid, BareJid> {
if self.slash.is_some() {
Ok(FullJid { inner: self })
} else {
Err(BareJid { inner: self })
}
}
pub fn try_as_full(&self) -> Result<&FullJid, &BareJid> {
if self.slash.is_some() {
Ok(unsafe {
mem::transmute::<&Jid, &FullJid>(self)
})
} else {
Err(unsafe {
mem::transmute::<&Jid, &BareJid>(self)
})
}
}
pub fn try_as_full_mut(&mut self) -> Result<&mut FullJid, &mut BareJid> {
if self.slash.is_some() {
Ok(unsafe {
mem::transmute::<&mut Jid, &mut FullJid>(self)
})
} else {
Err(unsafe {
mem::transmute::<&mut Jid, &mut BareJid>(self)
})
}
}
#[doc(hidden)]
#[allow(non_snake_case)]
#[deprecated(
since = "0.11.0",
note = "use Jid::from (for construction of Jid values) or Jid::try_into_full/Jid::try_as_full (for match blocks) instead"
)]
pub fn Bare(other: BareJid) -> Self {
Self::from(other)
}
#[doc(hidden)]
#[allow(non_snake_case)]
#[deprecated(
since = "0.11.0",
note = "use Jid::from (for construction of Jid values) or Jid::try_into_full/Jid::try_as_full (for match blocks) instead"
)]
pub fn Full(other: FullJid) -> Self {
Self::from(other)
}
}
impl TryFrom<Jid> for FullJid {
type Error = Error;
fn try_from(inner: Jid) -> Result<Self, Self::Error> {
if inner.slash.is_none() {
return Err(Error::ResourceMissingInFullJid);
}
Ok(Self { inner })
}
}
impl TryFrom<Jid> for BareJid {
type Error = Error;
fn try_from(inner: Jid) -> Result<Self, Self::Error> {
if inner.slash.is_some() {
return Err(Error::ResourceInBareJid);
}
Ok(Self { inner })
}
}
impl PartialEq<Jid> for FullJid {
fn eq(&self, other: &Jid) -> bool {
&self.inner == other
}
}
impl PartialEq<Jid> for BareJid {
fn eq(&self, other: &Jid) -> bool {
&self.inner == other
}
}
impl PartialEq<FullJid> for Jid {
fn eq(&self, other: &FullJid) -> bool {
self == &other.inner
}
}
impl PartialEq<BareJid> for Jid {
fn eq(&self, other: &BareJid) -> bool {
self == &other.inner
}
}
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
#[repr(transparent)] pub struct FullJid {
inner: Jid,
}
#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
#[repr(transparent)] pub struct BareJid {
inner: Jid,
}
impl Deref for FullJid {
type Target = Jid;
fn deref(&self) -> &Self::Target {
&self.inner
}
}
impl Deref for BareJid {
type Target = Jid;
fn deref(&self) -> &Self::Target {
&self.inner
}
}
impl Borrow<Jid> for FullJid {
fn borrow(&self) -> &Jid {
&self.inner
}
}
impl Borrow<Jid> for BareJid {
fn borrow(&self) -> &Jid {
&self.inner
}
}
impl fmt::Debug for Jid {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
fmt.debug_tuple("Jid").field(&self.normalized).finish()
}
}
impl fmt::Debug for FullJid {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
fmt.debug_tuple("FullJid")
.field(&self.inner.normalized)
.finish()
}
}
impl fmt::Debug for BareJid {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
fmt.debug_tuple("BareJid")
.field(&self.inner.normalized)
.finish()
}
}
impl fmt::Display for FullJid {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
fmt::Display::fmt(&self.inner, fmt)
}
}
impl fmt::Display for BareJid {
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
fmt::Display::fmt(&self.inner, fmt)
}
}
#[cfg(feature = "serde")]
impl Serialize for Jid {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
serializer.serialize_str(&self.normalized)
}
}
#[cfg(feature = "serde")]
impl Serialize for FullJid {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
self.inner.serialize(serializer)
}
}
#[cfg(feature = "serde")]
impl Serialize for BareJid {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
self.inner.serialize(serializer)
}
}
impl FromStr for FullJid {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
Self::new(s)
}
}
#[cfg(feature = "serde")]
impl<'de> Deserialize<'de> for Jid {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
let s = String::deserialize(deserializer)?;
Jid::new(&s).map_err(de::Error::custom)
}
}
#[cfg(feature = "serde")]
impl<'de> Deserialize<'de> for FullJid {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
let jid = Jid::deserialize(deserializer)?;
jid.try_into().map_err(de::Error::custom)
}
}
#[cfg(feature = "serde")]
impl<'de> Deserialize<'de> for BareJid {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
let jid = Jid::deserialize(deserializer)?;
jid.try_into().map_err(de::Error::custom)
}
}
#[cfg(feature = "quote")]
impl ToTokens for Jid {
fn to_tokens(&self, tokens: &mut TokenStream) {
let s = &self.normalized;
tokens.extend(quote! {
::jid::Jid::new(#s).unwrap()
});
}
}
#[cfg(feature = "quote")]
impl ToTokens for FullJid {
fn to_tokens(&self, tokens: &mut TokenStream) {
let s = &self.inner.normalized;
tokens.extend(quote! {
::jid::FullJid::new(#s).unwrap()
});
}
}
#[cfg(feature = "quote")]
impl ToTokens for BareJid {
fn to_tokens(&self, tokens: &mut TokenStream) {
let s = &self.inner.normalized;
tokens.extend(quote! {
::jid::BareJid::new(#s).unwrap()
});
}
}
impl FullJid {
pub fn new(unnormalized: &str) -> Result<Self, Error> {
Jid::new(unnormalized)?.try_into()
}
pub fn from_parts(
node: Option<&NodeRef>,
domain: &DomainRef,
resource: &ResourceRef,
) -> FullJid {
let (at, slash, normalized);
if let Some(node) = node {
at = NonZeroU16::new(node.len() as u16);
slash = NonZeroU16::new((node.len() + 1 + domain.len()) as u16);
normalized = format!("{node}@{domain}/{resource}");
} else {
at = None;
slash = NonZeroU16::new(domain.len() as u16);
normalized = format!("{domain}/{resource}");
}
let inner = Jid {
normalized,
at,
slash,
};
Self { inner }
}
pub fn resource(&self) -> &ResourceRef {
self.inner.resource().unwrap()
}
pub fn into_inner(self) -> String {
self.inner.into_inner()
}
pub fn into_bare(self) -> BareJid {
self.inner.into_bare()
}
}
impl FromStr for BareJid {
type Err = Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
Self::new(s)
}
}
impl BareJid {
pub fn new(unnormalized: &str) -> Result<Self, Error> {
Jid::new(unnormalized)?.try_into()
}
pub fn from_parts(node: Option<&NodeRef>, domain: &DomainRef) -> Self {
let (at, normalized);
if let Some(node) = node {
at = NonZeroU16::new(node.len() as u16);
normalized = format!("{node}@{domain}");
} else {
at = None;
normalized = domain.to_string();
}
let inner = Jid {
normalized,
at,
slash: None,
};
Self { inner }
}
pub fn with_resource(&self, resource: &ResourceRef) -> FullJid {
let slash = NonZeroU16::new(self.inner.normalized.len() as u16);
let normalized = format!("{}/{resource}", self.inner.normalized);
let inner = Jid {
normalized,
at: self.inner.at,
slash,
};
FullJid { inner }
}
pub fn with_resource_str(&self, resource: &str) -> Result<FullJid, Error> {
let resource = ResourcePart::new(resource)?;
Ok(self.with_resource(&resource))
}
pub fn into_inner(self) -> String {
self.inner.into_inner()
}
}
#[cfg(feature = "minidom")]
impl IntoAttributeValue for Jid {
fn into_attribute_value(self) -> Option<String> {
Some(self.to_string())
}
}
#[cfg(feature = "minidom")]
impl From<Jid> for Node {
fn from(jid: Jid) -> Self {
Node::Text(jid.to_string())
}
}
#[cfg(feature = "minidom")]
impl IntoAttributeValue for FullJid {
fn into_attribute_value(self) -> Option<String> {
self.inner.into_attribute_value()
}
}
#[cfg(feature = "minidom")]
impl From<FullJid> for Node {
fn from(jid: FullJid) -> Self {
jid.inner.into()
}
}
#[cfg(feature = "minidom")]
impl IntoAttributeValue for BareJid {
fn into_attribute_value(self) -> Option<String> {
self.inner.into_attribute_value()
}
}
#[cfg(feature = "minidom")]
impl From<BareJid> for Node {
fn from(other: BareJid) -> Self {
other.inner.into()
}
}
#[cfg(test)]
mod tests {
use super::*;
use std::collections::{HashMap, HashSet};
use std::vec::Vec;
macro_rules! assert_size (
($t:ty, $sz:expr) => (
assert_eq!(::core::mem::size_of::<$t>(), $sz);
);
);
#[cfg(target_pointer_width = "32")]
#[test]
fn test_size() {
assert_size!(BareJid, 16);
assert_size!(FullJid, 16);
assert_size!(Jid, 16);
}
#[cfg(target_pointer_width = "64")]
#[test]
fn test_size() {
assert_size!(BareJid, 32);
assert_size!(FullJid, 32);
assert_size!(Jid, 32);
}
#[test]
fn can_parse_full_jids() {
assert_eq!(
FullJid::from_str("a@b.c/d"),
Ok(FullJid::new("a@b.c/d").unwrap())
);
assert_eq!(
FullJid::from_str("b.c/d"),
Ok(FullJid::new("b.c/d").unwrap())
);
assert_eq!(
FullJid::from_str("a@b.c"),
Err(Error::ResourceMissingInFullJid)
);
assert_eq!(
FullJid::from_str("b.c"),
Err(Error::ResourceMissingInFullJid)
);
}
#[test]
fn can_parse_bare_jids() {
assert_eq!(
BareJid::from_str("a@b.c"),
Ok(BareJid::new("a@b.c").unwrap())
);
assert_eq!(BareJid::from_str("b.c"), Ok(BareJid::new("b.c").unwrap()));
}
#[test]
fn can_parse_jids() {
let full = FullJid::from_str("a@b.c/d").unwrap();
let bare = BareJid::from_str("e@f.g").unwrap();
assert_eq!(Jid::from_str("a@b.c/d").unwrap(), full);
assert_eq!(Jid::from_str("e@f.g").unwrap(), bare);
}
#[test]
fn full_to_bare_jid() {
let bare: BareJid = FullJid::new("a@b.c/d").unwrap().to_bare();
assert_eq!(bare, BareJid::new("a@b.c").unwrap());
}
#[test]
fn bare_to_full_jid_str() {
assert_eq!(
BareJid::new("a@b.c")
.unwrap()
.with_resource_str("d")
.unwrap(),
FullJid::new("a@b.c/d").unwrap()
);
}
#[test]
fn bare_to_full_jid() {
assert_eq!(
BareJid::new("a@b.c")
.unwrap()
.with_resource(&ResourcePart::new("d").unwrap()),
FullJid::new("a@b.c/d").unwrap()
)
}
#[test]
fn node_from_jid() {
let jid = Jid::new("a@b.c/d").unwrap();
assert_eq!(jid.node().map(|x| x.as_str()), Some("a"),);
}
#[test]
fn domain_from_jid() {
let jid = Jid::new("a@b.c").unwrap();
assert_eq!(jid.domain().as_str(), "b.c");
}
#[test]
fn resource_from_jid() {
let jid = Jid::new("a@b.c/d").unwrap();
assert_eq!(jid.resource().map(|x| x.as_str()), Some("d"),);
}
#[test]
fn jid_to_full_bare() {
let full = FullJid::new("a@b.c/d").unwrap();
let bare = BareJid::new("a@b.c").unwrap();
assert_eq!(FullJid::try_from(Jid::from(full.clone())), Ok(full.clone()));
assert_eq!(
FullJid::try_from(Jid::from(bare.clone())),
Err(Error::ResourceMissingInFullJid),
);
assert_eq!(Jid::from(full.clone().to_bare()), bare.clone());
assert_eq!(Jid::from(bare.clone()), bare);
}
#[test]
fn serialise() {
assert_eq!(FullJid::new("a@b/c").unwrap().to_string(), "a@b/c");
assert_eq!(BareJid::new("a@b").unwrap().to_string(), "a@b");
}
#[test]
fn hash() {
let _map: HashMap<Jid, String> = HashMap::new();
}
#[test]
fn invalid_jids() {
assert_eq!(BareJid::from_str(""), Err(Error::DomainEmpty));
assert_eq!(BareJid::from_str("/c"), Err(Error::DomainEmpty));
assert_eq!(BareJid::from_str("a@/c"), Err(Error::DomainEmpty));
assert_eq!(BareJid::from_str("@b"), Err(Error::NodeEmpty));
assert_eq!(BareJid::from_str("b/"), Err(Error::ResourceEmpty));
assert_eq!(FullJid::from_str(""), Err(Error::DomainEmpty));
assert_eq!(FullJid::from_str("/c"), Err(Error::DomainEmpty));
assert_eq!(FullJid::from_str("a@/c"), Err(Error::DomainEmpty));
assert_eq!(FullJid::from_str("@b"), Err(Error::NodeEmpty));
assert_eq!(FullJid::from_str("b/"), Err(Error::ResourceEmpty));
assert_eq!(
FullJid::from_str("a@b"),
Err(Error::ResourceMissingInFullJid)
);
assert_eq!(BareJid::from_str("a@b/c"), Err(Error::ResourceInBareJid));
}
#[test]
fn display_jids() {
assert_eq!(FullJid::new("a@b/c").unwrap().to_string(), "a@b/c");
assert_eq!(BareJid::new("a@b").unwrap().to_string(), "a@b");
assert_eq!(
Jid::from(FullJid::new("a@b/c").unwrap()).to_string(),
"a@b/c"
);
assert_eq!(Jid::from(BareJid::new("a@b").unwrap()).to_string(), "a@b");
}
#[cfg(feature = "minidom")]
#[test]
fn minidom() {
let elem: minidom::Element = "<message xmlns='ns1' from='a@b/c'/>".parse().unwrap();
let to: Jid = elem.attr("from").unwrap().parse().unwrap();
assert_eq!(to, Jid::from(FullJid::new("a@b/c").unwrap()));
let elem: minidom::Element = "<message xmlns='ns1' from='a@b'/>".parse().unwrap();
let to: Jid = elem.attr("from").unwrap().parse().unwrap();
assert_eq!(to, Jid::from(BareJid::new("a@b").unwrap()));
let elem: minidom::Element = "<message xmlns='ns1' from='a@b/c'/>".parse().unwrap();
let to: FullJid = elem.attr("from").unwrap().parse().unwrap();
assert_eq!(to, FullJid::new("a@b/c").unwrap());
let elem: minidom::Element = "<message xmlns='ns1' from='a@b'/>".parse().unwrap();
let to: BareJid = elem.attr("from").unwrap().parse().unwrap();
assert_eq!(to, BareJid::new("a@b").unwrap());
}
#[cfg(feature = "minidom")]
#[test]
fn minidom_into_attr() {
let full = FullJid::new("a@b/c").unwrap();
let elem = minidom::Element::builder("message", "jabber:client")
.attr("from", full.clone())
.build();
assert_eq!(elem.attr("from"), Some(full.to_string().as_str()));
let bare = BareJid::new("a@b").unwrap();
let elem = minidom::Element::builder("message", "jabber:client")
.attr("from", bare.clone())
.build();
assert_eq!(elem.attr("from"), Some(bare.to_string().as_str()));
let jid = Jid::from(bare.clone());
let _elem = minidom::Element::builder("message", "jabber:client")
.attr("from", jid)
.build();
assert_eq!(elem.attr("from"), Some(bare.to_string().as_str()));
}
#[test]
fn stringprep() {
let full = FullJid::from_str("Test@☃.coM/Test™").unwrap();
let equiv = FullJid::new("test@☃.com/TestTM").unwrap();
assert_eq!(full, equiv);
}
#[test]
fn invalid_stringprep() {
FullJid::from_str("a@b/🎉").unwrap_err();
}
#[test]
fn jid_from_parts() {
let node = NodePart::new("node").unwrap();
let domain = DomainPart::new("domain").unwrap();
let resource = ResourcePart::new("resource").unwrap();
let jid = Jid::from_parts(Some(&node), &domain, Some(&resource));
assert_eq!(jid, Jid::new("node@domain/resource").unwrap());
let barejid = BareJid::from_parts(Some(&node), &domain);
assert_eq!(barejid, BareJid::new("node@domain").unwrap());
let fulljid = FullJid::from_parts(Some(&node), &domain, &resource);
assert_eq!(fulljid, FullJid::new("node@domain/resource").unwrap());
}
#[test]
#[cfg(feature = "serde")]
fn jid_ser_de() {
let jid: Jid = Jid::new("node@domain").unwrap();
serde_test::assert_tokens(&jid, &[serde_test::Token::Str("node@domain")]);
let jid: Jid = Jid::new("node@domain/resource").unwrap();
serde_test::assert_tokens(&jid, &[serde_test::Token::Str("node@domain/resource")]);
let jid: BareJid = BareJid::new("node@domain").unwrap();
serde_test::assert_tokens(&jid, &[serde_test::Token::Str("node@domain")]);
let jid: FullJid = FullJid::new("node@domain/resource").unwrap();
serde_test::assert_tokens(&jid, &[serde_test::Token::Str("node@domain/resource")]);
}
#[test]
fn jid_into_parts_and_from_parts() {
let node = NodePart::new("node").unwrap();
let domain = DomainPart::new("domain").unwrap();
let jid1 = domain.with_node(&node);
let jid2 = node.with_domain(&domain);
let jid3 = BareJid::new("node@domain").unwrap();
assert_eq!(jid1, jid2);
assert_eq!(jid2, jid3);
}
#[test]
fn jid_match_replacement_try_as() {
let jid1 = Jid::new("foo@bar").unwrap();
let jid2 = Jid::new("foo@bar/baz").unwrap();
match jid1.try_as_full() {
Err(_) => (),
other => panic!("unexpected result: {:?}", other),
};
match jid2.try_as_full() {
Ok(_) => (),
other => panic!("unexpected result: {:?}", other),
};
}
#[test]
fn jid_match_replacement_try_as_mut() {
let mut jid1 = Jid::new("foo@bar").unwrap();
let mut jid2 = Jid::new("foo@bar/baz").unwrap();
match jid1.try_as_full_mut() {
Err(_) => (),
other => panic!("unexpected result: {:?}", other),
};
match jid2.try_as_full_mut() {
Ok(_) => (),
other => panic!("unexpected result: {:?}", other),
};
}
#[test]
fn jid_match_replacement_try_into() {
let jid1 = Jid::new("foo@bar").unwrap();
let jid2 = Jid::new("foo@bar/baz").unwrap();
match jid1.try_as_full() {
Err(_) => (),
other => panic!("unexpected result: {:?}", other),
};
match jid2.try_as_full() {
Ok(_) => (),
other => panic!("unexpected result: {:?}", other),
};
}
#[test]
fn lookup_jid_by_full_jid() {
let mut map: HashSet<Jid> = HashSet::new();
let jid1 = Jid::new("foo@bar").unwrap();
let jid2 = Jid::new("foo@bar/baz").unwrap();
let jid3 = FullJid::new("foo@bar/baz").unwrap();
map.insert(jid1);
assert!(!map.contains(&jid2));
assert!(!map.contains(&jid3));
map.insert(jid2);
assert!(map.contains(&jid3));
}
#[test]
fn lookup_full_jid_by_jid() {
let mut map: HashSet<FullJid> = HashSet::new();
let jid1 = FullJid::new("foo@bar/baz").unwrap();
let jid2 = FullJid::new("foo@bar/fnord").unwrap();
let jid3 = Jid::new("foo@bar/fnord").unwrap();
map.insert(jid1);
assert!(!map.contains(&jid2));
assert!(!map.contains(&jid3));
map.insert(jid2);
assert!(map.contains(&jid3));
}
#[test]
fn lookup_bare_jid_by_jid() {
let mut map: HashSet<BareJid> = HashSet::new();
let jid1 = BareJid::new("foo@bar").unwrap();
let jid2 = BareJid::new("foo@baz").unwrap();
let jid3 = Jid::new("foo@baz").unwrap();
map.insert(jid1);
assert!(!map.contains(&jid2));
assert!(!map.contains(&jid3));
map.insert(jid2);
assert!(map.contains(&jid3));
}
#[test]
fn normalizes_all_parts() {
assert_eq!(
Jid::new("ßA@IX.test/\u{2168}").unwrap().as_str(),
"ssa@ix.test/IX"
);
}
#[test]
fn rejects_unassigned_codepoints() {
match Jid::new("\u{01f601}@example.com") {
Err(Error::NodePrep) => (),
other => panic!("unexpected result: {:?}", other),
};
match Jid::new("foo@\u{01f601}.example.com") {
Err(Error::NamePrep) => (),
other => panic!("unexpected result: {:?}", other),
};
match Jid::new("foo@example.com/\u{01f601}") {
Err(Error::ResourcePrep) => (),
other => panic!("unexpected result: {:?}", other),
};
}
#[test]
fn accepts_domain_only_jid() {
match Jid::new("example.com") {
Ok(_) => (),
other => panic!("unexpected result: {:?}", other),
};
match BareJid::new("example.com") {
Ok(_) => (),
other => panic!("unexpected result: {:?}", other),
};
match FullJid::new("example.com/x") {
Ok(_) => (),
other => panic!("unexpected result: {:?}", other),
};
}
#[test]
fn is_bare_returns_true_iff_bare() {
let bare = Jid::new("foo@bar").unwrap();
let full = Jid::new("foo@bar/baz").unwrap();
assert!(bare.is_bare());
assert!(!full.is_bare());
}
#[test]
fn is_full_returns_true_iff_full() {
let bare = Jid::new("foo@bar").unwrap();
let full = Jid::new("foo@bar/baz").unwrap();
assert!(!bare.is_full());
assert!(full.is_full());
}
#[test]
fn reject_long_localpart() {
let mut long = Vec::with_capacity(1028);
long.resize(1024, b'a');
let mut long = String::from_utf8(long).unwrap();
long.push_str("@foo");
match Jid::new(&long) {
Err(Error::NodeTooLong) => (),
other => panic!("unexpected result: {:?}", other),
}
match BareJid::new(&long) {
Err(Error::NodeTooLong) => (),
other => panic!("unexpected result: {:?}", other),
}
}
#[test]
fn reject_long_domainpart() {
let mut long = Vec::with_capacity(1028);
long.push(b'x');
long.push(b'@');
long.resize(1026, b'a');
let long = String::from_utf8(long).unwrap();
match Jid::new(&long) {
Err(Error::DomainTooLong) => (),
other => panic!("unexpected result: {:?}", other),
}
match BareJid::new(&long) {
Err(Error::DomainTooLong) => (),
other => panic!("unexpected result: {:?}", other),
}
}
#[test]
fn reject_long_resourcepart() {
let mut long = Vec::with_capacity(1028);
long.push(b'x');
long.push(b'@');
long.push(b'y');
long.push(b'/');
long.resize(1028, b'a');
let long = String::from_utf8(long).unwrap();
match Jid::new(&long) {
Err(Error::ResourceTooLong) => (),
other => panic!("unexpected result: {:?}", other),
}
match FullJid::new(&long) {
Err(Error::ResourceTooLong) => (),
other => panic!("unexpected result: {:?}", other),
}
}
}