203 lines
6.1 KiB
Rust
203 lines
6.1 KiB
Rust
use crate::{services, Error, Result, Ruma};
|
|
use rand::seq::SliceRandom;
|
|
use regex::Regex;
|
|
use ruma::{
|
|
api::{
|
|
appservice,
|
|
client::{
|
|
alias::{create_alias, delete_alias, get_alias},
|
|
error::ErrorKind,
|
|
},
|
|
federation,
|
|
},
|
|
OwnedRoomAliasId, OwnedServerName,
|
|
};
|
|
|
|
/// # `PUT /_matrix/client/r0/directory/room/{roomAlias}`
|
|
///
|
|
/// Creates a new room alias on this server.
|
|
pub async fn create_alias_route(
|
|
body: Ruma<create_alias::v3::Request>,
|
|
) -> Result<create_alias::v3::Response> {
|
|
if body.room_alias.server_name() != services().globals.server_name() {
|
|
return Err(Error::BadRequest(
|
|
ErrorKind::InvalidParam,
|
|
"Alias is from another server.",
|
|
));
|
|
}
|
|
|
|
if services()
|
|
.rooms
|
|
.alias
|
|
.resolve_local_alias(&body.room_alias)?
|
|
.is_some()
|
|
{
|
|
return Err(Error::Conflict("Alias already exists."));
|
|
}
|
|
|
|
services()
|
|
.rooms
|
|
.alias
|
|
.set_alias(&body.room_alias, &body.room_id)?;
|
|
|
|
Ok(create_alias::v3::Response::new())
|
|
}
|
|
|
|
/// # `DELETE /_matrix/client/r0/directory/room/{roomAlias}`
|
|
///
|
|
/// Deletes a room alias from this server.
|
|
///
|
|
/// - TODO: additional access control checks
|
|
/// - TODO: Update canonical alias event
|
|
pub async fn delete_alias_route(
|
|
body: Ruma<delete_alias::v3::Request>,
|
|
) -> Result<delete_alias::v3::Response> {
|
|
if body.room_alias.server_name() != services().globals.server_name() {
|
|
return Err(Error::BadRequest(
|
|
ErrorKind::InvalidParam,
|
|
"Alias is from another server.",
|
|
));
|
|
}
|
|
|
|
services().rooms.alias.remove_alias(&body.room_alias)?;
|
|
|
|
// TODO: update alt_aliases?
|
|
|
|
Ok(delete_alias::v3::Response::new())
|
|
}
|
|
|
|
/// # `GET /_matrix/client/v3/directory/room/{roomAlias}`
|
|
///
|
|
/// Resolve an alias locally or over federation.
|
|
pub async fn get_alias_route(
|
|
body: Ruma<get_alias::v3::Request>,
|
|
) -> Result<get_alias::v3::Response> {
|
|
get_alias_helper(body.body.room_alias).await
|
|
}
|
|
|
|
pub(crate) async fn get_alias_helper(
|
|
room_alias: OwnedRoomAliasId,
|
|
) -> Result<get_alias::v3::Response> {
|
|
if room_alias.server_name() != services().globals.server_name() {
|
|
let response = services()
|
|
.sending
|
|
.send_federation_request(
|
|
room_alias.server_name(),
|
|
federation::query::get_room_information::v1::Request {
|
|
room_alias: room_alias.to_owned(),
|
|
},
|
|
)
|
|
.await?;
|
|
|
|
let room_id = response.room_id;
|
|
|
|
let mut servers = response.servers;
|
|
|
|
// find active servers in room state cache to suggest
|
|
for extra_servers in services()
|
|
.rooms
|
|
.state_cache
|
|
.room_servers(&room_id)
|
|
.filter_map(|r| r.ok())
|
|
{
|
|
servers.push(extra_servers);
|
|
}
|
|
|
|
// shuffle list of servers randomly
|
|
servers.shuffle(&mut rand::thread_rng());
|
|
|
|
// insert our server as the very first choice if in list
|
|
if let Some(server_index) = servers
|
|
.clone()
|
|
.into_iter()
|
|
.position(|server| server == services().globals.server_name())
|
|
{
|
|
servers.remove(server_index);
|
|
servers.insert(0, services().globals.server_name().to_owned());
|
|
}
|
|
|
|
return Ok(get_alias::v3::Response::new(room_id, servers));
|
|
}
|
|
|
|
let mut room_id = None;
|
|
match services().rooms.alias.resolve_local_alias(&room_alias)? {
|
|
Some(r) => room_id = Some(r),
|
|
None => {
|
|
for (_id, registration) in services().appservice.all()? {
|
|
let aliases = registration
|
|
.get("namespaces")
|
|
.and_then(|ns| ns.get("aliases"))
|
|
.and_then(|aliases| aliases.as_sequence())
|
|
.map_or_else(Vec::new, |aliases| {
|
|
aliases
|
|
.iter()
|
|
.filter_map(|aliases| Regex::new(aliases.get("regex")?.as_str()?).ok())
|
|
.collect::<Vec<_>>()
|
|
});
|
|
|
|
if aliases
|
|
.iter()
|
|
.any(|aliases| aliases.is_match(room_alias.as_str()))
|
|
&& services()
|
|
.sending
|
|
.send_appservice_request(
|
|
registration,
|
|
appservice::query::query_room_alias::v1::Request {
|
|
room_alias: room_alias.clone(),
|
|
},
|
|
)
|
|
.await
|
|
.is_ok()
|
|
{
|
|
room_id = Some(
|
|
services()
|
|
.rooms
|
|
.alias
|
|
.resolve_local_alias(&room_alias)?
|
|
.ok_or_else(|| {
|
|
Error::bad_config("Appservice lied to us. Room does not exist.")
|
|
})?,
|
|
);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
let room_id = match room_id {
|
|
Some(room_id) => room_id,
|
|
None => {
|
|
return Err(Error::BadRequest(
|
|
ErrorKind::NotFound,
|
|
"Room with alias not found.",
|
|
))
|
|
}
|
|
};
|
|
|
|
let mut servers: Vec<OwnedServerName> = Vec::new();
|
|
|
|
// find active servers in room state cache to suggest
|
|
for extra_servers in services()
|
|
.rooms
|
|
.state_cache
|
|
.room_servers(&room_id)
|
|
.filter_map(|r| r.ok())
|
|
{
|
|
servers.push(extra_servers);
|
|
}
|
|
|
|
// shuffle list of servers randomly
|
|
servers.shuffle(&mut rand::thread_rng());
|
|
|
|
// insert our server as the very first choice if in list
|
|
if let Some(server_index) = servers
|
|
.clone()
|
|
.into_iter()
|
|
.position(|server| server == services().globals.server_name())
|
|
{
|
|
servers.remove(server_index);
|
|
servers.insert(0, services().globals.server_name().to_owned());
|
|
}
|
|
|
|
Ok(get_alias::v3::Response::new(room_id, servers))
|
|
}
|