Files
continuwuity/src/api/client/user_directory.rs
T

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

100 lines
2.5 KiB
Rust
Raw Normal View History

2024-07-16 08:05:25 +00:00
use axum::extract::State;
2024-08-08 17:18:30 +00:00
use futures::{pin_mut, StreamExt};
use ruma::{
api::client::user_directory::search_users,
events::{
room::join_rules::{JoinRule, RoomJoinRulesEventContent},
StateEventType,
},
};
2020-07-30 18:14:47 +02:00
2024-07-16 08:05:25 +00:00
use crate::{Result, Ruma};
2024-03-05 19:48:54 -05:00
2021-08-31 19:14:37 +02:00
/// # `POST /_matrix/client/r0/user_directory/search`
///
/// Searches all known users for a match.
///
/// - Hides any local users that aren't in any public rooms (i.e. those that
2024-06-16 00:36:49 +00:00
/// have the join rule set to public) and don't share a room with the sender
2024-07-16 08:05:25 +00:00
pub(crate) async fn search_users_route(
State(services): State<crate::State>, body: Ruma<search_users::v3::Request>,
) -> Result<search_users::v3::Response> {
let sender_user = body.sender_user.as_ref().expect("user is authenticated");
2024-05-04 09:45:37 -04:00
let limit = usize::try_from(body.limit).unwrap_or(10); // default limit is 10
2020-07-30 18:14:47 +02:00
2024-08-08 17:18:30 +00:00
let users = services.users.stream().filter_map(|user_id| async {
2020-07-30 18:14:47 +02:00
// Filter out buggy users (they should not exist, but you never know...)
2022-02-18 15:33:14 +01:00
let user = search_users::v3::User {
2024-08-08 17:18:30 +00:00
user_id: user_id.to_owned(),
display_name: services.users.displayname(user_id).await.ok(),
avatar_url: services.users.avatar_url(user_id).await.ok(),
2020-07-30 18:14:47 +02:00
};
2024-03-25 17:05:11 -04:00
let user_id_matches = user
.user_id
.to_string()
.to_lowercase()
.contains(&body.search_term.to_lowercase());
2021-07-14 12:31:38 +02:00
let user_displayname_matches = user
.display_name
.as_ref()
2024-03-25 17:05:11 -04:00
.filter(|name| {
name.to_lowercase()
.contains(&body.search_term.to_lowercase())
})
2021-07-14 12:31:38 +02:00
.is_some();
if !user_id_matches && !user_displayname_matches {
2020-07-30 18:14:47 +02:00
return None;
}
// It's a matching user, but is the sender allowed to see them?
let mut user_visible = false;
2024-07-16 08:05:25 +00:00
let user_is_in_public_rooms = services
2024-03-25 17:05:11 -04:00
.rooms
.state_cache
2024-08-08 17:18:30 +00:00
.rooms_joined(&user.user_id)
.any(|room| async move {
2024-07-16 08:05:25 +00:00
services
2024-03-25 17:05:11 -04:00
.rooms
.state_accessor
2024-08-08 17:18:30 +00:00
.room_state_get(room, &StateEventType::RoomJoinRules, "")
.await
2024-03-25 17:05:11 -04:00
.map_or(false, |event| {
2024-08-08 17:18:30 +00:00
serde_json::from_str(event.content.get())
.map_or(false, |r: RoomJoinRulesEventContent| r.join_rule == JoinRule::Public)
2024-03-25 17:05:11 -04:00
})
2024-08-08 17:18:30 +00:00
})
.await;
if user_is_in_public_rooms {
user_visible = true;
} else {
2024-07-16 08:05:25 +00:00
let user_is_in_shared_rooms = services
2024-03-25 17:05:11 -04:00
.rooms
.user
2024-08-08 17:18:30 +00:00
.has_shared_rooms(sender_user, &user.user_id)
.await;
if user_is_in_shared_rooms {
user_visible = true;
}
}
2024-08-08 17:18:30 +00:00
user_visible.then_some(user)
2020-07-30 18:14:47 +02:00
});
2024-08-08 17:18:30 +00:00
pin_mut!(users);
let limited = users.by_ref().next().await.is_some();
let results = users.take(limit).collect().await;
2020-07-30 18:14:47 +02:00
2022-02-18 15:33:14 +01:00
Ok(search_users::v3::Response {
results,
limited,
})
2020-07-30 18:14:47 +02:00
}