222 lines
6.9 KiB
Rust
222 lines
6.9 KiB
Rust
//! Matrix identifiers for places where a room ID or room alias ID are used interchangeably.
|
|
|
|
use std::{convert::TryFrom, hint::unreachable_unchecked, num::NonZeroU8};
|
|
|
|
use crate::{server_name::ServerName, Error, RoomAliasId, RoomId};
|
|
|
|
/// A Matrix room ID or a Matrix room alias ID.
|
|
///
|
|
/// `RoomIdOrAliasId` is useful for APIs that accept either kind of room identifier. It is converted
|
|
/// from a string slice, and can be converted back into a string as needed. When converted from a
|
|
/// string slice, the variant is determined by the leading sigil character.
|
|
///
|
|
/// ```
|
|
/// # use std::convert::TryFrom;
|
|
/// # use ruma_identifiers::RoomIdOrAliasId;
|
|
/// assert_eq!(
|
|
/// RoomIdOrAliasId::try_from("#ruma:example.com").unwrap().as_ref(),
|
|
/// "#ruma:example.com"
|
|
/// );
|
|
///
|
|
/// assert_eq!(
|
|
/// RoomIdOrAliasId::try_from("!n8f893n9:example.com").unwrap().as_ref(),
|
|
/// "!n8f893n9:example.com"
|
|
/// );
|
|
/// ```
|
|
#[derive(Clone, Debug)]
|
|
pub struct RoomIdOrAliasId {
|
|
full_id: Box<str>,
|
|
colon_idx: NonZeroU8,
|
|
}
|
|
|
|
impl RoomIdOrAliasId {
|
|
/// Returns the local part (everything after the `!` or `#` and before the first colon).
|
|
pub fn localpart(&self) -> &str {
|
|
&self.full_id[1..self.colon_idx.get() as usize]
|
|
}
|
|
|
|
/// Returns the server name of the room (alias) ID.
|
|
pub fn server_name(&self) -> &ServerName {
|
|
<&ServerName>::try_from(&self.full_id[self.colon_idx.get() as usize + 1..]).unwrap()
|
|
}
|
|
|
|
/// Whether this is a room id (starts with `'!'`)
|
|
pub fn is_room_id(&self) -> bool {
|
|
self.variant() == Variant::RoomId
|
|
}
|
|
|
|
/// Whether this is a room alias id (starts with `'#'`)
|
|
pub fn is_room_alias_id(&self) -> bool {
|
|
self.variant() == Variant::RoomAliasId
|
|
}
|
|
|
|
/// Turn this `RoomIdOrAliasId` into `Either<RoomId, RoomAliasId>`
|
|
#[cfg(feature = "either")]
|
|
#[cfg_attr(docsrs, doc(cfg(feature = "either")))]
|
|
pub fn into_either(self) -> either::Either<RoomId, RoomAliasId> {
|
|
match self.variant() {
|
|
Variant::RoomId => {
|
|
either::Either::Left(RoomId { full_id: self.full_id, colon_idx: self.colon_idx })
|
|
}
|
|
Variant::RoomAliasId => either::Either::Right(RoomAliasId {
|
|
full_id: self.full_id,
|
|
colon_idx: self.colon_idx,
|
|
}),
|
|
}
|
|
}
|
|
|
|
fn variant(&self) -> Variant {
|
|
match self.full_id.bytes().next() {
|
|
Some(b'!') => Variant::RoomId,
|
|
Some(b'#') => Variant::RoomAliasId,
|
|
_ => unsafe { unreachable_unchecked() },
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(PartialEq)]
|
|
enum Variant {
|
|
RoomId,
|
|
RoomAliasId,
|
|
}
|
|
|
|
/// Attempts to create a new Matrix room ID or a room alias ID from a string representation.
|
|
///
|
|
/// The string must either include the leading ! sigil, the localpart, a literal colon, and a
|
|
/// valid homeserver host or include the leading # sigil, the alias, a literal colon, and a
|
|
/// valid homeserver host.
|
|
fn try_from<S>(room_id_or_alias_id: S) -> Result<RoomIdOrAliasId, Error>
|
|
where
|
|
S: AsRef<str> + Into<Box<str>>,
|
|
{
|
|
let colon_idx =
|
|
ruma_identifiers_validation::room_id_or_alias_id::validate(room_id_or_alias_id.as_ref())?;
|
|
Ok(RoomIdOrAliasId { full_id: room_id_or_alias_id.into(), colon_idx })
|
|
}
|
|
|
|
common_impls!(RoomIdOrAliasId, try_from, "a Matrix room ID or room alias ID");
|
|
|
|
impl From<RoomId> for RoomIdOrAliasId {
|
|
fn from(RoomId { full_id, colon_idx }: RoomId) -> Self {
|
|
Self { full_id, colon_idx }
|
|
}
|
|
}
|
|
|
|
impl From<RoomAliasId> for RoomIdOrAliasId {
|
|
fn from(RoomAliasId { full_id, colon_idx }: RoomAliasId) -> Self {
|
|
Self { full_id, colon_idx }
|
|
}
|
|
}
|
|
|
|
impl TryFrom<RoomIdOrAliasId> for RoomId {
|
|
type Error = RoomAliasId;
|
|
|
|
fn try_from(id: RoomIdOrAliasId) -> Result<RoomId, RoomAliasId> {
|
|
match id.variant() {
|
|
Variant::RoomId => Ok(RoomId { full_id: id.full_id, colon_idx: id.colon_idx }),
|
|
Variant::RoomAliasId => {
|
|
Err(RoomAliasId { full_id: id.full_id, colon_idx: id.colon_idx })
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl TryFrom<RoomIdOrAliasId> for RoomAliasId {
|
|
type Error = RoomId;
|
|
|
|
fn try_from(id: RoomIdOrAliasId) -> Result<RoomAliasId, RoomId> {
|
|
match id.variant() {
|
|
Variant::RoomAliasId => {
|
|
Ok(RoomAliasId { full_id: id.full_id, colon_idx: id.colon_idx })
|
|
}
|
|
Variant::RoomId => Err(RoomId { full_id: id.full_id, colon_idx: id.colon_idx }),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use std::convert::TryFrom;
|
|
|
|
#[cfg(feature = "serde")]
|
|
use serde_json::{from_str, to_string};
|
|
|
|
use super::RoomIdOrAliasId;
|
|
use crate::Error;
|
|
|
|
#[test]
|
|
fn valid_room_id_or_alias_id_with_a_room_alias_id() {
|
|
assert_eq!(
|
|
RoomIdOrAliasId::try_from("#ruma:example.com")
|
|
.expect("Failed to create RoomAliasId.")
|
|
.as_ref(),
|
|
"#ruma:example.com"
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn valid_room_id_or_alias_id_with_a_room_id() {
|
|
assert_eq!(
|
|
RoomIdOrAliasId::try_from("!29fhd83h92h0:example.com")
|
|
.expect("Failed to create RoomId.")
|
|
.as_ref(),
|
|
"!29fhd83h92h0:example.com"
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn missing_sigil_for_room_id_or_alias_id() {
|
|
assert_eq!(
|
|
RoomIdOrAliasId::try_from("ruma:example.com").unwrap_err(),
|
|
Error::MissingLeadingSigil
|
|
);
|
|
}
|
|
|
|
#[cfg(feature = "serde")]
|
|
#[test]
|
|
fn serialize_valid_room_id_or_alias_id_with_a_room_alias_id() {
|
|
assert_eq!(
|
|
to_string(
|
|
&RoomIdOrAliasId::try_from("#ruma:example.com")
|
|
.expect("Failed to create RoomAliasId.")
|
|
)
|
|
.expect("Failed to convert RoomAliasId to JSON."),
|
|
r##""#ruma:example.com""##
|
|
);
|
|
}
|
|
|
|
#[cfg(feature = "serde")]
|
|
#[test]
|
|
fn serialize_valid_room_id_or_alias_id_with_a_room_id() {
|
|
assert_eq!(
|
|
to_string(
|
|
&RoomIdOrAliasId::try_from("!29fhd83h92h0:example.com")
|
|
.expect("Failed to create RoomId.")
|
|
)
|
|
.expect("Failed to convert RoomId to JSON."),
|
|
r#""!29fhd83h92h0:example.com""#
|
|
);
|
|
}
|
|
|
|
#[cfg(feature = "serde")]
|
|
#[test]
|
|
fn deserialize_valid_room_id_or_alias_id_with_a_room_alias_id() {
|
|
assert_eq!(
|
|
from_str::<RoomIdOrAliasId>(r##""#ruma:example.com""##)
|
|
.expect("Failed to convert JSON to RoomAliasId"),
|
|
RoomIdOrAliasId::try_from("#ruma:example.com").expect("Failed to create RoomAliasId.")
|
|
);
|
|
}
|
|
|
|
#[cfg(feature = "serde")]
|
|
#[test]
|
|
fn deserialize_valid_room_id_or_alias_id_with_a_room_id() {
|
|
assert_eq!(
|
|
from_str::<RoomIdOrAliasId>(r##""!29fhd83h92h0:example.com""##)
|
|
.expect("Failed to convert JSON to RoomId"),
|
|
RoomIdOrAliasId::try_from("!29fhd83h92h0:example.com")
|
|
.expect("Failed to create RoomAliasId.")
|
|
);
|
|
}
|
|
}
|