Skip to content

Commit

Permalink
feat(room_preview): Use room directory search as another data source
Browse files Browse the repository at this point in the history
  • Loading branch information
jmartinesp committed Nov 15, 2024
1 parent 232391c commit 95083a9
Show file tree
Hide file tree
Showing 3 changed files with 253 additions and 13 deletions.
98 changes: 92 additions & 6 deletions crates/matrix-sdk/src/room_preview.rs
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
use matrix_sdk_base::{RoomInfo, RoomState};
use ruma::{
api::client::{membership::joined_members, state::get_state_events},
directory::PublicRoomJoinRule,
events::room::{history_visibility::HistoryVisibility, join_rules::JoinRule},
room::RoomType,
space::SpaceRoomJoinRule,
Expand All @@ -29,7 +30,10 @@ use ruma::{
use tokio::try_join;
use tracing::{instrument, warn};

use crate::{Client, Room};
use crate::{
room_directory_search::{RoomDescription, RoomDirectorySearch},
Client, Room,
};

/// The preview of a room, be it invited/joined/left, or not.
#[derive(Debug, Clone)]
Expand Down Expand Up @@ -119,7 +123,7 @@ impl RoomPreview {

/// Create a room preview from a known room (i.e. one we've been invited to,
/// we've joined or we've left).
pub(crate) async fn from_known(room: &Room) -> Self {
pub async fn from_known(room: &Room) -> Self {
let is_direct = room.is_direct().await.ok();

Self::from_room_info(
Expand All @@ -140,21 +144,76 @@ impl RoomPreview {
) -> crate::Result<Self> {
// Use the room summary endpoint, if available, as described in
// https://github.com/deepbluev7/matrix-doc/blob/room-summaries/proposals/3266-room-summary.md
match Self::from_room_summary(client, room_id.clone(), room_or_alias_id, via).await {
match Self::from_room_summary(client, room_id.clone(), room_or_alias_id, via.clone()).await
{
Ok(res) => return Ok(res),
Err(err) => {
warn!("error when previewing room from the room summary endpoint: {err}");
}
}

// TODO: (optimization) Use the room search directory, if available:
// - if the room directory visibility is public,
// - then use a public room filter set to this room id
// Try room directory search next.
match Self::from_room_directory_search(client, &room_id, room_or_alias_id, via).await {
Ok(res) => return Ok(res),
Err(err) => {
warn!("Room '{room_or_alias_id}' not found in room directory search: {err}");
}
}

// Resort to using the room state endpoint, as well as the joined members one.
Self::from_state_events(client, &room_id).await
}

/// Get a [`RoomPreview`] by searching in the room directory for the
/// provided room alias or room id and transforming the [`RoomDescription`]
/// into a preview.
pub(crate) async fn from_room_directory_search(
client: &Client,
room_id: &RoomId,
room_or_alias_id: &RoomOrAliasId,
via: Vec<OwnedServerName>,
) -> crate::Result<Self> {
let mut directory_search = RoomDirectorySearch::new(client.clone());

// Get either the room alias or the room id without the leading identifier char
let search_term = if room_or_alias_id.is_room_alias_id() {
Some(room_or_alias_id.to_string().chars().skip(1).collect::<String>())
} else {
None
};

// If we have no alias, filtering using a room id is impossible, so just take
// the first 100 results and try to find the current room #YOLO
let batch_size = if search_term.is_some() { 20 } else { 100 };

if via.is_empty() {
// Just search in the current homeserver
let _ = directory_search.search(search_term, batch_size, None).await;
} else {
// Iterate all servers until one returns some results
for server in via {
if directory_search.loaded_pages() > 0 {
// If we previously got results don't search anymore
break;
}

let _ =
directory_search.search(search_term.clone(), batch_size, Some(server)).await;
}
}

let (values, _) = directory_search.results();
while let Some(room_description) = values.iter().next() {
// Iterate until we find a room description with a matching room id
if room_description.room_id != room_id {
continue;
}
return Ok(RoomPreview::from_room_description(room_description.to_owned()));
}

Err(crate::Error::InsufficientData)
}

/// Get a [`RoomPreview`] using MSC3266, if available on the remote server.
///
/// Will fail with a 404 if the API is not available.
Expand Down Expand Up @@ -258,4 +317,31 @@ impl RoomPreview {
state,
))
}

pub(crate) fn from_room_description(room_description: RoomDescription) -> Self {
RoomPreview {
room_id: room_description.room_id,
canonical_alias: room_description.alias,
name: room_description.name,
topic: room_description.topic,
avatar_url: room_description.avatar_url,
num_joined_members: room_description.joined_members,
num_active_members: None,
// Assume it's a room
room_type: None,
join_rule: space_rule_from_public_rule(room_description.join_rule),
is_world_readable: room_description.is_world_readable,
state: None,
is_direct: None,
}
}
}

fn space_rule_from_public_rule(join_rule: PublicRoomJoinRule) -> SpaceRoomJoinRule {
match join_rule {
PublicRoomJoinRule::Public => SpaceRoomJoinRule::Public,
PublicRoomJoinRule::Knock => SpaceRoomJoinRule::Knock,
PublicRoomJoinRule::_Custom(rule) => SpaceRoomJoinRule::_Custom(rule),
_ => panic!("Unexpected PublicRoomJoinRule {:?}", join_rule),
}
}
74 changes: 70 additions & 4 deletions crates/matrix-sdk/src/test_utils/mocks.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,24 +17,31 @@
#![allow(missing_debug_implementations)]

use std::sync::{Arc, Mutex};
use std::{
collections::BTreeMap,
sync::{Arc, Mutex},
};

use matrix_sdk_base::{deserialized_responses::TimelineEvent, store::StoreConfig, SessionMeta};
use matrix_sdk_test::{
test_json, InvitedRoomBuilder, JoinedRoomBuilder, KnockedRoomBuilder, LeftRoomBuilder,
SyncResponseBuilder,
};
use ruma::{api::MatrixVersion, device_id, user_id, MxcUri, OwnedEventId, OwnedRoomId, RoomId};
use ruma::{
api::MatrixVersion, device_id, directory::PublicRoomsChunk, user_id, MxcUri, OwnedEventId,
OwnedRoomId, RoomId, ServerName,
};
use serde::Deserialize;
use serde_json::json;
use wiremock::{
matchers::{body_partial_json, header, method, path, path_regex},
Mock, MockBuilder, MockGuard, MockServer, Respond, ResponseTemplate, Times,
Mock, MockBuilder, MockGuard, MockServer, Request, Respond, ResponseTemplate, Times,
};

use crate::{
config::RequestConfig,
matrix_auth::{MatrixSession, MatrixSessionTokens},
Client, ClientBuilder, Room,
Client, ClientBuilder, OwnedServerName, Room,
};

/// A [`wiremock`] [`MockServer`] along with useful methods to help mocking
Expand Down Expand Up @@ -452,6 +459,12 @@ impl MatrixMockServer {
Mock::given(method("PUT")).and(path_regex(r"/_matrix/client/v3/directory/room/.*"));
MockEndpoint { mock, server: &self.server, endpoint: CreateRoomAliasEndpoint }
}

/// Create a prebuilt mock for creating room aliases.
pub fn mock_public_rooms(&self) -> MockEndpoint<'_, PublicRoomsEndpoint> {
let mock = Mock::given(method("POST")).and(path_regex(r"/_matrix/client/v3/publicRooms"));
MockEndpoint { mock, server: &self.server, endpoint: PublicRoomsEndpoint }
}
}

/// Parameter to [`MatrixMockServer::sync_room`].
Expand Down Expand Up @@ -1033,6 +1046,59 @@ impl<'a> MockEndpoint<'a, CreateRoomAliasEndpoint> {
}
}

/// A prebuilt mock for paginating the public room list.
pub struct PublicRoomsEndpoint;

impl<'a> MockEndpoint<'a, PublicRoomsEndpoint> {
/// Returns a data endpoint for paginating the public room list.
pub fn ok(
self,
chunk: Vec<PublicRoomsChunk>,
next_batch: Option<String>,
prev_batch: Option<String>,
total_room_count_estimate: Option<u64>,
) -> MatrixMock<'a> {
let mock = self.mock.respond_with(ResponseTemplate::new(200).set_body_json(json!({
"chunk": chunk,
"next_batch": next_batch,
"prev_batch": prev_batch,
"total_room_count_estimate": total_room_count_estimate,
})));
MatrixMock { server: self.server, mock }
}

/// Returns a data endpoint for paginating the public room list with several
/// `via` params.
///
/// Each `via` param must be in the `server_map` parameter, otherwise it'll
/// fail.
pub fn with_via_params(
self,
server_map: BTreeMap<OwnedServerName, Vec<PublicRoomsChunk>>,
) -> MatrixMock<'a> {
let mock = self.mock.respond_with(move |req: &Request| {
#[derive(Deserialize)]
struct PartialRequest {
server: Option<OwnedServerName>,
}

let (_, server) = req
.url
.query_pairs()
.into_iter()
.find(|(key, _)| key == "server")
.expect("Server param not found in request URL");
let server = ServerName::parse(server).expect("Couldn't parse server name");
let chunk = server_map.get(&server).expect("Chunk for the server param not found");
ResponseTemplate::new(200).set_body_json(json!({
"chunk": chunk,
"total_room_count_estimate": chunk.len(),
}))
});
MatrixMock { server: self.server, mock }
}
}

/// An augmented [`ClientBuilder`] that also allows for handling session login.
pub struct MockClientBuilder {
builder: ClientBuilder,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
#![allow(unused)]

use std::{
collections::BTreeMap,
sync::{Arc, Mutex as StdMutex},
time::Duration,
};
Expand All @@ -21,6 +22,7 @@ use matrix_sdk::{
room::create_room::v3::{Request as CreateRoomRequest, RoomPreset},
},
assign,
directory::PublicRoomsChunkInit,
events::{
receipt::ReceiptThread,
room::{
Expand All @@ -30,14 +32,16 @@ use matrix_sdk::{
},
AnySyncMessageLikeEvent, InitialStateEvent, Mentions, StateEventType,
},
mxc_uri,
mxc_uri, owned_server_name, room_id,
space::SpaceRoomJoinRule,
RoomId,
uint, RoomId,
},
sliding_sync::VersionBuilder,
test_utils::{logged_in_client_with_server, mocks::MatrixMockServer},
Client, RoomInfo, RoomMemberships, RoomState, SlidingSyncList, SlidingSyncMode,
};
use matrix_sdk_base::sliding_sync::http;
use matrix_sdk_base::{ruma::room_alias_id, sliding_sync::http};
use matrix_sdk_test::async_test;
use matrix_sdk_ui::{
room_list_service::filters::new_filter_all, sync_service::SyncService, timeline::RoomExt,
RoomListService,
Expand Down Expand Up @@ -1092,6 +1096,90 @@ async fn test_room_preview() -> Result<()> {
Ok(())
}

#[async_test]
async fn test_room_preview_with_room_directory_search_and_room_alias_only() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;

let room_alias = room_alias_id!("#a-room:matrix.org");
let expected_room_id = room_id!("!a-room:matrix.org");

// Allow retrieving the room id
server
.mock_room_directory_resolve_alias()
.ok(expected_room_id.as_ref(), Vec::new())
.mock_once()
.mount()
.await;

// Given a successful public room search
let chunks = vec![PublicRoomsChunkInit {
num_joined_members: uint!(0),
room_id: expected_room_id.to_owned(),
world_readable: true,
guest_can_join: true,
}
.into()];
server.mock_public_rooms().ok(chunks, None, None, Some(1)).mock_once().mount().await;

// The room preview is found
let preview = client
.get_room_preview(room_alias.into(), Vec::new())
.await
.expect("room preview couldn't be retrieved");
assert_eq!(preview.room_id, expected_room_id);
}

#[async_test]
async fn test_room_preview_with_room_directory_search_and_room_alias_only_in_several_homeservers() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;

let room_alias = room_alias_id!("#a-room:matrix.org");
let expected_room_id = room_id!("!a-room:matrix.org");

// Allow retrieving the room id
server
.mock_room_directory_resolve_alias()
.ok(expected_room_id.as_ref(), Vec::new())
.mock_once()
.mount()
.await;

let via_1 = owned_server_name!("server1.com");
let via_2 = owned_server_name!("server2.com");

// Given a couple of successful public room search responses
let via_map = BTreeMap::from_iter(vec![
(
via_1.to_owned(),
vec![PublicRoomsChunkInit {
num_joined_members: uint!(0),
room_id: expected_room_id.to_owned(),
world_readable: true,
guest_can_join: true,
}
.into()],
),
(via_2.to_owned(), Vec::new()),
]);
server
.mock_public_rooms()
.with_via_params(via_map)
// Called only once as the first response returns the room, so no need to check other
// homeservers
.expect(1)
.mount()
.await;

// The room preview is found in the first response
let preview = client
.get_room_preview(room_alias.into(), vec![via_1, via_2])
.await
.expect("room preview couldn't be retrieved");
assert_eq!(preview.room_id, expected_room_id);
}

fn assert_room_preview(preview: &RoomPreview, room_alias: &str) {
assert_eq!(preview.canonical_alias.as_ref().unwrap().alias(), room_alias);
assert_eq!(preview.name.as_ref().unwrap(), "Alice's Room");
Expand Down

0 comments on commit 95083a9

Please sign in to comment.