1
0
Fork 0
mirror of https://gitlab.com/famedly/conduit.git synced 2025-06-27 16:35:59 +00:00
conduit/src/api/client_server/alias.rs

191 lines
5.5 KiB
Rust
Raw Normal View History

2022-10-05 20:34:31 +02:00
use crate::{services, Error, Result, Ruma};
use rand::seq::SliceRandom;
use ruma::{
api::{
appservice,
client::{
2022-02-18 15:33:14 +01:00
alias::{create_alias, delete_alias, get_alias},
error::ErrorKind,
},
federation,
2020-08-14 11:34:15 +02:00
},
2022-12-14 13:09:10 +01:00
OwnedRoomAliasId,
2020-08-14 11:34:15 +02:00
};
2020-07-30 18:14:47 +02:00
2021-08-31 19:14:37 +02:00
/// # `PUT /_matrix/client/r0/directory/room/{roomAlias}`
///
/// Creates a new room alias on this server.
pub async fn create_alias_route(
2022-12-14 13:09:10 +01:00
body: Ruma<create_alias::v3::Request>,
2022-02-18 15:33:14 +01:00
) -> Result<create_alias::v3::Response> {
2024-06-11 23:15:02 +02:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
if body.room_alias.server_name() != services().globals.server_name() {
2021-08-31 19:14:37 +02:00
return Err(Error::BadRequest(
ErrorKind::InvalidParam,
"Alias is from another server.",
));
}
if let Some(ref info) = body.appservice_info {
if !info.aliases.is_match(body.room_alias.as_str()) {
return Err(Error::BadRequest(
ErrorKind::Exclusive,
"Room alias is not in namespace.",
));
}
} else if services()
.appservice
.is_exclusive_alias(&body.room_alias)
.await
{
return Err(Error::BadRequest(
ErrorKind::Exclusive,
"Room alias reserved by appservice.",
));
}
2022-10-05 20:34:31 +02:00
if services()
.rooms
.alias
.resolve_local_alias(&body.room_alias)?
.is_some()
{
2020-07-30 18:14:47 +02:00
return Err(Error::Conflict("Alias already exists."));
}
2022-10-05 20:34:31 +02:00
services()
.rooms
.alias
2024-06-22 21:22:43 +07:00
.set_alias(&body.room_alias, &body.room_id, sender_user)
.await?;
2022-02-18 15:33:14 +01:00
Ok(create_alias::v3::Response::new())
2020-07-30 18:14:47 +02:00
}
2021-08-31 19:14:37 +02:00
/// # `DELETE /_matrix/client/r0/directory/room/{roomAlias}`
///
/// Deletes a room alias from this server.
///
/// - TODO: Update canonical alias event
pub async fn delete_alias_route(
2022-12-14 13:09:10 +01:00
body: Ruma<delete_alias::v3::Request>,
2022-02-18 15:33:14 +01:00
) -> Result<delete_alias::v3::Response> {
2024-06-11 23:15:02 +02:00
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
if body.room_alias.server_name() != services().globals.server_name() {
2021-08-31 19:14:37 +02:00
return Err(Error::BadRequest(
ErrorKind::InvalidParam,
"Alias is from another server.",
));
}
if let Some(ref info) = body.appservice_info {
if !info.aliases.is_match(body.room_alias.as_str()) {
return Err(Error::BadRequest(
ErrorKind::Exclusive,
"Room alias is not in namespace.",
));
}
} else if services()
.appservice
.is_exclusive_alias(&body.room_alias)
.await
{
return Err(Error::BadRequest(
ErrorKind::Exclusive,
"Room alias reserved by appservice.",
));
}
2024-06-11 23:15:02 +02:00
services()
.rooms
.alias
.remove_alias(&body.room_alias, sender_user)?;
2020-07-30 18:14:47 +02:00
2021-08-31 19:14:37 +02:00
// TODO: update alt_aliases?
2022-02-18 15:33:14 +01:00
Ok(delete_alias::v3::Response::new())
2020-07-30 18:14:47 +02:00
}
2021-08-31 19:14:37 +02:00
/// # `GET /_matrix/client/r0/directory/room/{roomAlias}`
///
/// Resolve an alias locally or over federation.
///
/// - TODO: Suggest more servers to join via
pub async fn get_alias_route(
2022-12-14 13:09:10 +01:00
body: Ruma<get_alias::v3::Request>,
2022-02-18 15:33:14 +01:00
) -> Result<get_alias::v3::Response> {
2022-12-14 13:09:10 +01:00
get_alias_helper(body.body.room_alias).await
}
2022-12-14 13:09:10 +01:00
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()
2020-12-19 16:00:11 +01:00
.sending
.send_federation_request(
room_alias.server_name(),
2022-12-14 13:09:10 +01:00
federation::query::get_room_information::v1::Request {
room_alias: room_alias.to_owned(),
},
2020-12-19 16:00:11 +01:00
)
.await?;
let mut servers = response.servers;
servers.shuffle(&mut rand::thread_rng());
return Ok(get_alias::v3::Response::new(response.room_id, servers));
2020-07-30 18:14:47 +02:00
}
let mut room_id = None;
2022-12-14 13:09:10 +01:00
match services().rooms.alias.resolve_local_alias(&room_alias)? {
Some(r) => room_id = Some(r),
None => {
for appservice in services().appservice.read().await.values() {
if appservice.aliases.is_match(room_alias.as_str())
&& matches!(
services()
.sending
.send_appservice_request(
appservice.registration.clone(),
appservice::query::query_room_alias::v1::Request {
room_alias: room_alias.clone(),
},
)
.await,
Ok(Some(_opt_result))
)
{
2022-10-05 20:34:31 +02:00
room_id = Some(
services()
.rooms
.alias
2022-12-14 13:09:10 +01:00
.resolve_local_alias(&room_alias)?
2022-10-05 20:34:31 +02:00
.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.",
))
}
};
2020-07-30 18:14:47 +02:00
2022-02-18 15:33:14 +01:00
Ok(get_alias::v3::Response::new(
room_id,
vec![services().globals.server_name().to_owned()],
))
2020-07-30 18:14:47 +02:00
}