use std::{cmp::Ordering, collections::HashMap, sync::Arc};
use eyeball_im::{ObservableVector, VectorSubscriberBatchedStream};
use futures_util::pin_mut;
use imbl::Vector;
use itertools::Itertools;
use matrix_sdk::{
Client, Error as SDKError, deserialized_responses::SyncOrStrippedState, executor::AbortOnDrop,
};
use matrix_sdk_common::executor::spawn;
use ruma::{
OwnedRoomId, RoomId,
events::{
self, StateEventType, SyncStateEvent,
space::{child::SpaceChildEventContent, parent::SpaceParentEventContent},
},
};
use thiserror::Error;
use tokio::sync::Mutex as AsyncMutex;
use tracing::{error, warn};
use crate::spaces::{graph::SpaceGraph, leave::LeaveSpaceHandle};
pub use crate::spaces::{room::SpaceRoom, room_list::SpaceRoomList};
pub mod graph;
pub mod leave;
pub mod room;
pub mod room_list;
#[derive(Debug, Error)]
pub enum Error {
#[error("User ID not available from client")]
UserIdNotFound,
#[error("Room `{0}` not found")]
RoomNotFound(OwnedRoomId),
#[error("Missing `{0}` for `{1}`")]
MissingState(StateEventType, OwnedRoomId),
#[error("Failed updating space parent/child relationship")]
UpdateRelationship(SDKError),
#[error("Failed to leave space")]
LeaveSpace(SDKError),
#[error("Failed to load members")]
LoadRoomMembers(SDKError),
}
struct SpaceState {
graph: SpaceGraph,
joined_rooms: ObservableVector<SpaceRoom>,
}
pub struct SpaceService {
client: Client,
space_state: Arc<AsyncMutex<SpaceState>>,
room_update_handle: AsyncMutex<Option<AbortOnDrop<()>>>,
}
impl SpaceService {
pub fn new(client: Client) -> Self {
Self {
client,
space_state: Arc::new(AsyncMutex::new(SpaceState {
graph: SpaceGraph::new(),
joined_rooms: ObservableVector::new(),
})),
room_update_handle: AsyncMutex::new(None),
}
}
pub async fn subscribe_to_joined_spaces(
&self,
) -> (Vector<SpaceRoom>, VectorSubscriberBatchedStream<SpaceRoom>) {
let mut room_update_handle = self.room_update_handle.lock().await;
if room_update_handle.is_none() {
let client = self.client.clone();
let space_state = Arc::clone(&self.space_state);
let all_room_updates_receiver = self.client.subscribe_to_all_room_updates();
*room_update_handle = Some(AbortOnDrop::new(spawn(async move {
pin_mut!(all_room_updates_receiver);
loop {
match all_room_updates_receiver.recv().await {
Ok(updates) => {
if updates.is_empty() {
continue;
}
let (spaces, graph) = Self::joined_spaces_for(&client).await;
Self::update_joined_spaces_if_needed(
Vector::from(spaces),
graph,
&space_state,
)
.await;
}
Err(err) => {
error!("error when listening to room updates: {err}");
}
}
}
})));
let (spaces, graph) = Self::joined_spaces_for(&self.client).await;
Self::update_joined_spaces_if_needed(Vector::from(spaces), graph, &self.space_state)
.await;
}
self.space_state.lock().await.joined_rooms.subscribe().into_values_and_batched_stream()
}
pub async fn joined_spaces(&self) -> Vec<SpaceRoom> {
let (spaces, graph) = Self::joined_spaces_for(&self.client).await;
Self::update_joined_spaces_if_needed(
Vector::from(spaces.clone()),
graph,
&self.space_state,
)
.await;
spaces
}
pub async fn editable_spaces(&self) -> Vec<SpaceRoom> {
let Some(user_id) = self.client.user_id() else {
return vec![];
};
let graph = &self.space_state.lock().await.graph;
let rooms = self.client.joined_space_rooms();
let mut editable_spaces = Vec::new();
for room in &rooms {
if let Ok(power_levels) = room.power_levels().await
&& power_levels.user_can_send_state(user_id, StateEventType::SpaceChild)
{
let room_id = room.room_id();
editable_spaces
.push(SpaceRoom::new_from_known(room, graph.children_of(room_id).len() as u64));
}
}
editable_spaces
}
pub async fn space_room_list(&self, space_id: OwnedRoomId) -> SpaceRoomList {
SpaceRoomList::new(self.client.clone(), space_id).await
}
pub async fn joined_parents_of_child(&self, child_id: &RoomId) -> Vec<SpaceRoom> {
let graph = &self.space_state.lock().await.graph;
graph
.parents_of(child_id)
.into_iter()
.filter_map(|parent_id| self.client.get_room(parent_id))
.map(|room| {
SpaceRoom::new_from_known(&room, graph.children_of(room.room_id()).len() as u64)
})
.collect()
}
pub async fn add_child_to_space(
&self,
child_id: OwnedRoomId,
space_id: OwnedRoomId,
) -> Result<(), Error> {
let user_id = self.client.user_id().ok_or(Error::UserIdNotFound)?;
let space_room =
self.client.get_room(&space_id).ok_or(Error::RoomNotFound(space_id.to_owned()))?;
let child_room =
self.client.get_room(&child_id).ok_or(Error::RoomNotFound(child_id.to_owned()))?;
let child_power_levels = child_room
.power_levels()
.await
.map_err(|error| Error::UpdateRelationship(matrix_sdk::Error::from(error)))?;
let child_route = child_room.route().await.map_err(Error::UpdateRelationship)?;
space_room
.send_state_event_for_key(&child_id, SpaceChildEventContent::new(child_route))
.await
.map_err(Error::UpdateRelationship)?;
if child_power_levels.user_can_send_state(user_id, StateEventType::SpaceParent) {
let parent_route = space_room.route().await.map_err(Error::UpdateRelationship)?;
child_room
.send_state_event_for_key(&space_id, SpaceParentEventContent::new(parent_route))
.await
.map_err(Error::UpdateRelationship)?;
} else {
warn!("The current user doesn't have permission to set the child's parent.");
}
Ok(())
}
pub async fn remove_child_from_space(
&self,
child_id: OwnedRoomId,
space_id: OwnedRoomId,
) -> Result<(), Error> {
let space_room =
self.client.get_room(&space_id).ok_or(Error::RoomNotFound(space_id.to_owned()))?;
let child_room =
self.client.get_room(&child_id).ok_or(Error::RoomNotFound(child_id.to_owned()))?;
if let Ok(Some(_)) =
space_room.get_state_event_static_for_key::<SpaceChildEventContent, _>(&child_id).await
{
space_room
.send_state_event_raw("m.space.child", child_id.as_str(), serde_json::json!({}))
.await
.map_err(Error::UpdateRelationship)?;
} else {
warn!("A space child event wasn't found on the parent, ignoring.");
}
if let Ok(Some(_)) =
child_room.get_state_event_static_for_key::<SpaceParentEventContent, _>(&space_id).await
{
child_room
.send_state_event_raw("m.space.parent", space_id.as_str(), serde_json::json!({}))
.await
.map_err(Error::UpdateRelationship)?;
} else {
warn!("A space parent event wasn't found on the child, ignoring.");
}
Ok(())
}
pub async fn leave_space(&self, space_id: &RoomId) -> Result<LeaveSpaceHandle, Error> {
let space_state = self.space_state.lock().await;
if !space_state.graph.has_node(space_id) {
return Err(Error::RoomNotFound(space_id.to_owned()));
}
let room_ids = space_state.graph.flattened_bottom_up_subtree(space_id);
let handle = LeaveSpaceHandle::new(self.client.clone(), room_ids).await;
Ok(handle)
}
async fn update_joined_spaces_if_needed(
new_spaces: Vector<SpaceRoom>,
new_graph: SpaceGraph,
space_state: &Arc<AsyncMutex<SpaceState>>,
) {
let mut space_state = space_state.lock().await;
if new_spaces != space_state.joined_rooms.clone() {
space_state.joined_rooms.clear();
space_state.joined_rooms.append(new_spaces);
}
space_state.graph = new_graph;
}
async fn joined_spaces_for(client: &Client) -> (Vec<SpaceRoom>, SpaceGraph) {
let joined_spaces = client.joined_space_rooms();
let mut graph = SpaceGraph::new();
for space in joined_spaces.iter() {
graph.add_node(space.room_id().to_owned());
if let Ok(parents) = space.get_state_events_static::<SpaceParentEventContent>().await {
parents.into_iter()
.flat_map(|parent_event| match parent_event.deserialize() {
Ok(SyncOrStrippedState::Sync(SyncStateEvent::Original(e))) => {
Some(e.state_key)
}
Ok(SyncOrStrippedState::Sync(SyncStateEvent::Redacted(_))) => None,
Ok(SyncOrStrippedState::Stripped(e)) => Some(e.state_key),
Err(e) => {
error!(room_id = ?space.room_id(), "Could not deserialize m.space.parent: {e}");
None
}
}).for_each(|parent| graph.add_edge(parent, space.room_id().to_owned()));
} else {
error!(room_id = ?space.room_id(), "Could not get m.space.parent events");
}
if let Ok(children) = space.get_state_events_static::<SpaceChildEventContent>().await {
children.into_iter()
.filter_map(|child_event| match child_event.deserialize() {
Ok(SyncOrStrippedState::Sync(SyncStateEvent::Original(e))) => {
Some(e.state_key)
}
Ok(SyncOrStrippedState::Sync(SyncStateEvent::Redacted(_))) => None,
Ok(SyncOrStrippedState::Stripped(e)) => Some(e.state_key),
Err(e) => {
error!(room_id = ?space.room_id(), "Could not deserialize m.space.child: {e}");
None
}
}).for_each(|child| graph.add_edge(space.room_id().to_owned(), child));
} else {
error!(room_id = ?space.room_id(), "Could not get m.space.child events");
}
}
graph.remove_cycles();
let root_nodes = graph.root_nodes();
let top_level_spaces = joined_spaces
.iter()
.filter(|room| root_nodes.contains(&room.room_id()))
.collect::<Vec<_>>();
let mut top_level_space_order = HashMap::new();
for space in &top_level_spaces {
if let Ok(Some(raw_event)) =
space.account_data_static::<events::space_order::SpaceOrderEventContent>().await
&& let Ok(event) = raw_event.deserialize()
{
top_level_space_order.insert(space.room_id().to_owned(), event.content.order);
}
}
let top_level_spaces = top_level_spaces
.iter()
.sorted_by(|a, b| {
match (
top_level_space_order.get(a.room_id()),
top_level_space_order.get(b.room_id()),
) {
(Some(a_order), Some(b_order)) => {
a_order.cmp(b_order).then(a.room_id().cmp(b.room_id()))
}
(Some(_), None) => Ordering::Less,
(None, Some(_)) => Ordering::Greater,
(None, None) => a.room_id().cmp(b.room_id()),
}
})
.map(|room| {
SpaceRoom::new_from_known(room, graph.children_of(room.room_id()).len() as u64)
})
.collect();
(top_level_spaces, graph)
}
}
#[cfg(test)]
mod tests {
use std::collections::BTreeMap;
use assert_matches2::assert_let;
use eyeball_im::VectorDiff;
use futures_util::{StreamExt, pin_mut};
use matrix_sdk::{room::ParentSpace, test_utils::mocks::MatrixMockServer};
use matrix_sdk_test::{
JoinedRoomBuilder, LeftRoomBuilder, RoomAccountDataTestEvent, async_test,
event_factory::EventFactory,
};
use ruma::{RoomVersionId, UserId, event_id, owned_room_id, room_id};
use serde_json::json;
use stream_assert::{assert_next_eq, assert_pending};
use super::*;
#[async_test]
async fn test_spaces_hierarchy() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let space_service = SpaceService::new(client.clone());
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
let parent_space_id = room_id!("!parent_space:example.org");
let child_space_id_1 = room_id!("!child_space_1:example.org");
let child_space_id_2 = room_id!("!child_space_2:example.org");
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: child_space_id_1,
order: None,
parents: vec![parent_space_id],
children: vec![],
power_level: None,
},
MockSpaceRoomParameters {
room_id: child_space_id_2,
order: None,
parents: vec![parent_space_id],
children: vec![],
power_level: None,
},
MockSpaceRoomParameters {
room_id: parent_space_id,
order: None,
parents: vec![],
children: vec![child_space_id_1, child_space_id_2],
power_level: None,
},
],
&client,
&server,
&factory,
user_id,
)
.await;
assert_eq!(
space_service
.joined_spaces()
.await
.iter()
.map(|s| s.room_id.to_owned())
.collect::<Vec<_>>(),
vec![parent_space_id]
);
assert_eq!(
space_service
.joined_spaces()
.await
.iter()
.map(|s| s.children_count)
.collect::<Vec<_>>(),
vec![2]
);
let parent_space = client.get_room(parent_space_id).unwrap();
assert!(parent_space.is_space());
let spaces: Vec<ParentSpace> = client
.get_room(child_space_id_1)
.unwrap()
.parent_spaces()
.await
.unwrap()
.map(Result::unwrap)
.collect()
.await;
assert_let!(ParentSpace::Reciprocal(parent) = spaces.first().unwrap());
assert_eq!(parent.room_id(), parent_space.room_id());
let spaces: Vec<ParentSpace> = client
.get_room(child_space_id_2)
.unwrap()
.parent_spaces()
.await
.unwrap()
.map(Result::unwrap)
.collect()
.await;
assert_let!(ParentSpace::Reciprocal(parent) = spaces.last().unwrap());
assert_eq!(parent.room_id(), parent_space.room_id());
}
#[async_test]
async fn test_joined_spaces_updates() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
let first_space_id = room_id!("!first_space:example.org");
let second_space_id = room_id!("!second_space:example.org");
server
.sync_room(
&client,
JoinedRoomBuilder::new(first_space_id)
.add_state_event(factory.create(user_id, RoomVersionId::V1).with_space_type()),
)
.await;
let space_service = SpaceService::new(client.clone());
let (initial_values, joined_spaces_subscriber) =
space_service.subscribe_to_joined_spaces().await;
pin_mut!(joined_spaces_subscriber);
assert_pending!(joined_spaces_subscriber);
assert_eq!(
initial_values,
vec![SpaceRoom::new_from_known(&client.get_room(first_space_id).unwrap(), 0)].into()
);
assert_eq!(
space_service.joined_spaces().await,
vec![SpaceRoom::new_from_known(&client.get_room(first_space_id).unwrap(), 0)]
);
assert_pending!(joined_spaces_subscriber);
server
.sync_room(
&client,
JoinedRoomBuilder::new(second_space_id)
.add_state_event(factory.create(user_id, RoomVersionId::V1).with_space_type())
.add_state_event(
factory
.space_child(
second_space_id.to_owned(),
owned_room_id!("!child:example.org"),
)
.sender(user_id),
),
)
.await;
assert_eq!(
space_service.joined_spaces().await,
vec![
SpaceRoom::new_from_known(&client.get_room(first_space_id).unwrap(), 0),
SpaceRoom::new_from_known(&client.get_room(second_space_id).unwrap(), 1)
]
);
assert_next_eq!(
joined_spaces_subscriber,
vec![
VectorDiff::Clear,
VectorDiff::Append {
values: vec![
SpaceRoom::new_from_known(&client.get_room(first_space_id).unwrap(), 0),
SpaceRoom::new_from_known(&client.get_room(second_space_id).unwrap(), 1)
]
.into()
},
]
);
server.sync_room(&client, LeftRoomBuilder::new(second_space_id)).await;
assert_next_eq!(
joined_spaces_subscriber,
vec![
VectorDiff::Clear,
VectorDiff::Append {
values: vec![SpaceRoom::new_from_known(
&client.get_room(first_space_id).unwrap(),
0
)]
.into()
},
]
);
server
.sync_room(
&client,
JoinedRoomBuilder::new(room_id!("!room:example.org"))
.add_state_event(factory.create(user_id, RoomVersionId::V1)),
)
.await;
assert_pending!(joined_spaces_subscriber);
assert_eq!(
space_service.joined_spaces().await,
vec![SpaceRoom::new_from_known(&client.get_room(first_space_id).unwrap(), 0)]
);
}
#[async_test]
async fn test_top_level_space_order() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
server.mock_room_state_encryption().plain().mount().await;
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: room_id!("!2:a.b"),
order: Some("2"),
parents: vec![],
children: vec![],
power_level: None,
},
MockSpaceRoomParameters {
room_id: room_id!("!4:a.b"),
order: None,
parents: vec![],
children: vec![],
power_level: None,
},
MockSpaceRoomParameters {
room_id: room_id!("!3:a.b"),
order: None,
parents: vec![],
children: vec![],
power_level: None,
},
MockSpaceRoomParameters {
room_id: room_id!("!1:a.b"),
order: Some("1"),
parents: vec![],
children: vec![],
power_level: None,
},
],
&client,
&server,
&EventFactory::new(),
client.user_id().unwrap(),
)
.await;
let space_service = SpaceService::new(client.clone());
assert_eq!(
space_service.joined_spaces().await,
vec![
SpaceRoom::new_from_known(&client.get_room(room_id!("!1:a.b")).unwrap(), 0),
SpaceRoom::new_from_known(&client.get_room(room_id!("!2:a.b")).unwrap(), 0),
SpaceRoom::new_from_known(&client.get_room(room_id!("!3:a.b")).unwrap(), 0),
SpaceRoom::new_from_known(&client.get_room(room_id!("!4:a.b")).unwrap(), 0),
]
);
}
#[async_test]
async fn test_editable_spaces() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
let admin_space_id = room_id!("!admin_space:example.org");
let admin_subspace_id = room_id!("!admin_subspace:example.org");
let regular_space_id = room_id!("!regular_space:example.org");
let regular_subspace_id = room_id!("!regular_subspace:example.org");
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: admin_space_id,
order: None,
parents: vec![],
children: vec![regular_subspace_id],
power_level: Some(100),
},
MockSpaceRoomParameters {
room_id: admin_subspace_id,
order: None,
parents: vec![regular_space_id],
children: vec![],
power_level: Some(100),
},
MockSpaceRoomParameters {
room_id: regular_space_id,
order: None,
parents: vec![],
children: vec![admin_subspace_id],
power_level: Some(0),
},
MockSpaceRoomParameters {
room_id: regular_subspace_id,
order: None,
parents: vec![admin_space_id],
children: vec![],
power_level: Some(0),
},
],
&client,
&server,
&factory,
user_id,
)
.await;
let space_service = SpaceService::new(client.clone());
_ = space_service.joined_spaces().await;
let editable_spaces = space_service.editable_spaces().await;
assert_eq!(
editable_spaces.iter().map(|room| room.room_id.to_owned()).collect::<Vec<_>>(),
vec![admin_space_id.to_owned(), admin_subspace_id.to_owned()]
);
}
#[async_test]
async fn test_joined_parents_of_child() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
let parent_space_id_1 = room_id!("!parent_space_1:example.org");
let parent_space_id_2 = room_id!("!parent_space_2:example.org");
let unknown_parent_space_id = room_id!("!unknown_parent_space:example.org");
let child_space_id = room_id!("!child_space:example.org");
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: child_space_id,
order: None,
parents: vec![parent_space_id_1, parent_space_id_2, unknown_parent_space_id],
children: vec![],
power_level: None,
},
MockSpaceRoomParameters {
room_id: parent_space_id_1,
order: None,
parents: vec![],
children: vec![child_space_id],
power_level: None,
},
MockSpaceRoomParameters {
room_id: parent_space_id_2,
order: None,
parents: vec![],
children: vec![child_space_id],
power_level: None,
},
],
&client,
&server,
&factory,
user_id,
)
.await;
let space_service = SpaceService::new(client.clone());
_ = space_service.joined_spaces().await;
let parents = space_service.joined_parents_of_child(child_space_id).await;
assert_eq!(
parents.iter().map(|space| space.room_id.to_owned()).collect::<Vec<_>>(),
vec![parent_space_id_1, parent_space_id_2]
);
}
#[async_test]
async fn test_add_child_to_space() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
let space_child_event_id = event_id!("$1");
let space_parent_event_id = event_id!("$2");
server.mock_set_space_child().ok(space_child_event_id.to_owned()).expect(1).mount().await;
server.mock_set_space_parent().ok(space_parent_event_id.to_owned()).expect(1).mount().await;
let space_id = room_id!("!my_space:example.org");
let child_id = room_id!("!my_child:example.org");
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: space_id,
order: None,
parents: vec![],
children: vec![],
power_level: Some(100),
},
MockSpaceRoomParameters {
room_id: child_id,
order: None,
parents: vec![],
children: vec![],
power_level: Some(100),
},
],
&client,
&server,
&factory,
user_id,
)
.await;
let space_service = SpaceService::new(client.clone());
let result =
space_service.add_child_to_space(child_id.to_owned(), space_id.to_owned()).await;
assert!(result.is_ok());
}
#[async_test]
async fn test_add_child_to_space_without_space_admin() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
server.mock_set_space_child().unauthorized().expect(1).mount().await;
server.mock_set_space_parent().unauthorized().expect(0).mount().await;
let space_id = room_id!("!my_space:example.org");
let child_id = room_id!("!my_child:example.org");
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: space_id,
order: None,
parents: vec![],
children: vec![],
power_level: Some(0),
},
MockSpaceRoomParameters {
room_id: child_id,
order: None,
parents: vec![],
children: vec![],
power_level: Some(0),
},
],
&client,
&server,
&factory,
user_id,
)
.await;
let space_service = SpaceService::new(client.clone());
let result =
space_service.add_child_to_space(child_id.to_owned(), space_id.to_owned()).await;
assert!(result.is_err());
}
#[async_test]
async fn test_add_child_to_space_without_child_admin() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
let space_child_event_id = event_id!("$1");
server.mock_set_space_child().ok(space_child_event_id.to_owned()).expect(1).mount().await;
server.mock_set_space_parent().unauthorized().expect(0).mount().await;
let space_id = room_id!("!my_space:example.org");
let child_id = room_id!("!my_child:example.org");
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: space_id,
order: None,
parents: vec![],
children: vec![],
power_level: Some(100),
},
MockSpaceRoomParameters {
room_id: child_id,
order: None,
parents: vec![],
children: vec![],
power_level: Some(0),
},
],
&client,
&server,
&factory,
user_id,
)
.await;
let space_service = SpaceService::new(client.clone());
let result =
space_service.add_child_to_space(child_id.to_owned(), space_id.to_owned()).await;
error!("result: {:?}", result);
assert!(result.is_ok());
}
#[async_test]
async fn test_remove_child_from_space() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
let space_child_event_id = event_id!("$1");
let space_parent_event_id = event_id!("$2");
server.mock_set_space_child().ok(space_child_event_id.to_owned()).expect(1).mount().await;
server.mock_set_space_parent().ok(space_parent_event_id.to_owned()).expect(1).mount().await;
let parent_id = room_id!("!parent_space:example.org");
let child_id = room_id!("!child_space:example.org");
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: parent_id,
order: None,
parents: vec![],
children: vec![child_id],
power_level: None,
},
MockSpaceRoomParameters {
room_id: child_id,
order: None,
parents: vec![parent_id],
children: vec![],
power_level: None,
},
],
&client,
&server,
&factory,
user_id,
)
.await;
let space_service = SpaceService::new(client.clone());
let result =
space_service.remove_child_from_space(child_id.to_owned(), parent_id.to_owned()).await;
assert!(result.is_ok());
}
#[async_test]
async fn test_remove_child_from_space_without_parent_event() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
let space_child_event_id = event_id!("$1");
server.mock_set_space_child().ok(space_child_event_id.to_owned()).expect(1).mount().await;
server.mock_set_space_parent().unauthorized().expect(0).mount().await;
let parent_id = room_id!("!parent_space:example.org");
let child_id = room_id!("!child_space:example.org");
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: parent_id,
order: None,
parents: vec![],
children: vec![child_id],
power_level: None,
},
MockSpaceRoomParameters {
room_id: child_id,
order: None,
parents: vec![],
children: vec![],
power_level: None,
},
],
&client,
&server,
&factory,
user_id,
)
.await;
let space_service = SpaceService::new(client.clone());
let result =
space_service.remove_child_from_space(child_id.to_owned(), parent_id.to_owned()).await;
assert!(result.is_ok());
}
#[async_test]
async fn test_remove_child_from_space_without_child_event() {
let server = MatrixMockServer::new().await;
let client = server.client_builder().build().await;
let user_id = client.user_id().unwrap();
let factory = EventFactory::new();
server.mock_room_state_encryption().plain().mount().await;
let space_parent_event_id = event_id!("$2");
server.mock_set_space_child().unauthorized().expect(0).mount().await;
server.mock_set_space_parent().ok(space_parent_event_id.to_owned()).expect(1).mount().await;
let parent_id = room_id!("!parent_space:example.org");
let child_id = room_id!("!child_space:example.org");
add_space_rooms(
vec![
MockSpaceRoomParameters {
room_id: parent_id,
order: None,
parents: vec![],
children: vec![],
power_level: None,
},
MockSpaceRoomParameters {
room_id: child_id,
order: None,
parents: vec![parent_id],
children: vec![],
power_level: None,
},
],
&client,
&server,
&factory,
user_id,
)
.await;
let space_service = SpaceService::new(client.clone());
let result =
space_service.remove_child_from_space(child_id.to_owned(), parent_id.to_owned()).await;
assert!(result.is_ok());
}
async fn add_space_rooms(
rooms: Vec<MockSpaceRoomParameters>,
client: &Client,
server: &MatrixMockServer,
factory: &EventFactory,
user_id: &UserId,
) {
for parameters in rooms {
let mut builder = JoinedRoomBuilder::new(parameters.room_id)
.add_state_event(factory.create(user_id, RoomVersionId::V1).with_space_type());
if let Some(order) = parameters.order {
builder = builder.add_account_data(RoomAccountDataTestEvent::Custom(json!({
"type": "m.space_order",
"content": {
"order": order
}
})));
}
for parent_id in parameters.parents {
builder = builder.add_state_event(
factory
.space_parent(parent_id.to_owned(), parameters.room_id.to_owned())
.sender(user_id),
);
}
for child_id in parameters.children {
builder = builder.add_state_event(
factory
.space_child(parameters.room_id.to_owned(), child_id.to_owned())
.sender(user_id),
);
}
if let Some(power_level) = parameters.power_level {
let mut power_levels = BTreeMap::from([(user_id.to_owned(), power_level.into())]);
builder = builder.add_state_event(
factory.power_levels(&mut power_levels).state_key("").sender(user_id),
);
}
server.sync_room(client, builder).await;
}
}
struct MockSpaceRoomParameters {
room_id: &'static RoomId,
order: Option<&'static str>,
parents: Vec<&'static RoomId>,
children: Vec<&'static RoomId>,
power_level: Option<i32>,
}
}