use ruma::{api::client::error::ErrorKind, OwnedRoomAliasId, OwnedRoomId, RoomAliasId, RoomId}; use crate::{services, utils, Error, KeyValueDatabase, Result}; pub trait Data: Send + Sync { /// Creates or updates the alias to the given room id. fn set_alias(&self, alias: &RoomAliasId, room_id: &RoomId) -> Result<()>; /// Forgets about an alias. Returns an error if the alias did not exist. fn remove_alias(&self, alias: &RoomAliasId) -> Result<()>; /// Looks up the roomid for the given alias. fn resolve_local_alias(&self, alias: &RoomAliasId) -> Result>; /// Returns all local aliases that point to the given room fn local_aliases_for_room<'a>( &'a self, room_id: &RoomId, ) -> Box> + 'a>; /// Returns all local aliases on the server fn all_local_aliases<'a>(&'a self) -> Box> + 'a>; } impl Data for KeyValueDatabase { fn set_alias(&self, alias: &RoomAliasId, room_id: &RoomId) -> Result<()> { self.alias_roomid .insert(alias.alias().as_bytes(), room_id.as_bytes())?; let mut aliasid = room_id.as_bytes().to_vec(); aliasid.push(0xFF); aliasid.extend_from_slice(&services().globals.next_count()?.to_be_bytes()); self.aliasid_alias.insert(&aliasid, alias.as_bytes())?; Ok(()) } fn remove_alias(&self, alias: &RoomAliasId) -> Result<()> { if let Some(room_id) = self.alias_roomid.get(alias.alias().as_bytes())? { let mut prefix = room_id; prefix.push(0xFF); for (key, _) in self.aliasid_alias.scan_prefix(prefix) { self.aliasid_alias.remove(&key)?; } self.alias_roomid.remove(alias.alias().as_bytes())?; } else { return Err(Error::BadRequest(ErrorKind::NotFound, "Alias does not exist.")); } Ok(()) } fn resolve_local_alias(&self, alias: &RoomAliasId) -> Result> { self.alias_roomid .get(alias.alias().as_bytes())? .map(|bytes| { RoomId::parse( utils::string_from_bytes(&bytes) .map_err(|_| Error::bad_database("Room ID in alias_roomid is invalid unicode."))?, ) .map_err(|_| Error::bad_database("Room ID in alias_roomid is invalid.")) }) .transpose() } fn local_aliases_for_room<'a>( &'a self, room_id: &RoomId, ) -> Box> + 'a> { let mut prefix = room_id.as_bytes().to_vec(); prefix.push(0xFF); Box::new(self.aliasid_alias.scan_prefix(prefix).map(|(_, bytes)| { utils::string_from_bytes(&bytes) .map_err(|_| Error::bad_database("Invalid alias bytes in aliasid_alias."))? .try_into() .map_err(|_| Error::bad_database("Invalid alias in aliasid_alias.")) })) } fn all_local_aliases<'a>(&'a self) -> Box> + 'a> { Box::new( self.alias_roomid .iter() .map(|(room_alias_bytes, room_id_bytes)| { let room_alias_localpart = utils::string_from_bytes(&room_alias_bytes) .map_err(|_| Error::bad_database("Invalid alias bytes in aliasid_alias."))?; let room_id = utils::string_from_bytes(&room_id_bytes) .map_err(|_| Error::bad_database("Invalid room_id bytes in aliasid_alias."))? .try_into() .map_err(|_| Error::bad_database("Invalid room_id in aliasid_alias."))?; Ok((room_id, room_alias_localpart)) }), ) } }