parallelise IO of user searching, improve perf, raise max limit to 500

Signed-off-by: June Clementine Strawberry <june@3.dog>
This commit is contained in:
June Clementine Strawberry 2025-04-05 20:09:22 -04:00
parent 3cc92b32ec
commit 6578b83bce
No known key found for this signature in database

View file

@ -1,16 +1,20 @@
use axum::extract::State; use axum::extract::State;
use conduwuit::{Result, utils::TryFutureExtExt}; use conduwuit::{
use futures::{StreamExt, pin_mut}; Result,
utils::{future::BoolExt, stream::BroadbandExt},
};
use futures::{FutureExt, StreamExt, pin_mut};
use ruma::{ use ruma::{
api::client::user_directory::search_users, api::client::user_directory::search_users::{self},
events::{ events::room::join_rules::JoinRule,
StateEventType,
room::join_rules::{JoinRule, RoomJoinRulesEventContent},
},
}; };
use crate::Ruma; use crate::Ruma;
// conduwuit can handle a lot more results than synapse
const LIMIT_MAX: usize = 500;
const LIMIT_DEFAULT: usize = 10;
/// # `POST /_matrix/client/r0/user_directory/search` /// # `POST /_matrix/client/r0/user_directory/search`
/// ///
/// Searches all known users for a match. /// Searches all known users for a match.
@ -21,78 +25,63 @@ pub(crate) async fn search_users_route(
State(services): State<crate::State>, State(services): State<crate::State>,
body: Ruma<search_users::v3::Request>, body: Ruma<search_users::v3::Request>,
) -> Result<search_users::v3::Response> { ) -> Result<search_users::v3::Response> {
let sender_user = body.sender_user.as_ref().expect("user is authenticated"); let sender_user = body.sender_user();
let limit = usize::try_from(body.limit).map_or(10, usize::from).min(100); // default limit is 10 let limit = usize::try_from(body.limit)
.map_or(LIMIT_DEFAULT, usize::from)
.min(LIMIT_MAX);
let users = services.users.stream().filter_map(|user_id| async { let mut users = services
// Filter out buggy users (they should not exist, but you never know...) .users
.stream()
.map(ToOwned::to_owned)
.broad_filter_map(async |user_id| {
let user = search_users::v3::User { let user = search_users::v3::User {
user_id: user_id.to_owned(), user_id: user_id.clone(),
display_name: services.users.displayname(user_id).await.ok(), display_name: services.users.displayname(&user_id).await.ok(),
avatar_url: services.users.avatar_url(user_id).await.ok(), avatar_url: services.users.avatar_url(&user_id).await.ok(),
}; };
let user_id_matches = user let user_id_matches = user
.user_id .user_id
.to_string() .as_str()
.to_lowercase() .to_lowercase()
.contains(&body.search_term.to_lowercase()); .contains(&body.search_term.to_lowercase());
let user_displayname_matches = user let user_displayname_matches = user.display_name.as_ref().is_some_and(|name| {
.display_name
.as_ref()
.filter(|name| {
name.to_lowercase() name.to_lowercase()
.contains(&body.search_term.to_lowercase()) .contains(&body.search_term.to_lowercase())
}) });
.is_some();
if !user_id_matches && !user_displayname_matches { if !user_id_matches && !user_displayname_matches {
return None; return None;
} }
// It's a matching user, but is the sender allowed to see them? let user_in_public_room = services
let mut user_visible = false;
let user_is_in_public_rooms = services
.rooms .rooms
.state_cache .state_cache
.rooms_joined(&user.user_id) .rooms_joined(&user_id)
.any(|room| { .map(ToOwned::to_owned)
.any(|room| async move {
services services
.rooms .rooms
.state_accessor .state_accessor
.room_state_get_content::<RoomJoinRulesEventContent>( .get_join_rules(&room)
room, .map(|rule| matches!(rule, JoinRule::Public))
&StateEventType::RoomJoinRules, .await
"",
)
.map_ok_or(false, |content| content.join_rule == JoinRule::Public)
})
.await;
if user_is_in_public_rooms {
user_visible = true;
} else {
let user_is_in_shared_rooms = services
.rooms
.state_cache
.user_sees_user(sender_user, &user.user_id)
.await;
if user_is_in_shared_rooms {
user_visible = true;
}
}
user_visible.then_some(user)
}); });
pin_mut!(users); let user_sees_user = services
.rooms
.state_cache
.user_sees_user(sender_user, &user_id);
let limited = users.by_ref().next().await.is_some(); pin_mut!(user_in_public_room, user_sees_user);
let results = users.take(limit).collect().await; user_in_public_room.or(user_sees_user).await.then_some(user)
});
let results = users.by_ref().take(limit).collect().await;
let limited = users.next().await.is_some();
Ok(search_users::v3::Response { results, limited }) Ok(search_users::v3::Response { results, limited })
} }