Skip to content
Snippets Groups Projects
space.rs 1.42 KiB
Newer Older
use std::str::FromStr;
use axum::extract::State;
use ruma::{
	api::client::{error::ErrorKind, space::get_hierarchy},
use crate::{service::rooms::spaces::PaginationToken, Error, Result, Ruma};
/// # `GET /_matrix/client/v1/rooms/{room_id}/hierarchy`
Timo Kösters's avatar
Timo Kösters committed
///
/// Paginates over the space tree in a depth-first manner to locate child rooms
/// of a given space.
pub(crate) async fn get_hierarchy_route(
	State(services): State<crate::State>, body: Ruma<get_hierarchy::v1::Request>,
) -> Result<get_hierarchy::v1::Response> {
	let sender_user = body.sender_user.as_ref().expect("user is authenticated");
	let limit = body
🥺's avatar
🥺 committed
		.limit
		.unwrap_or_else(|| UInt::from(10_u32))
		.min(UInt::from(100_u32));
🥺's avatar
🥺 committed

	let max_depth = body
		.max_depth
		.unwrap_or_else(|| UInt::from(3_u32))
		.min(UInt::from(10_u32));

	let key = body
		.from
		.as_ref()
		.and_then(|s| PaginationToken::from_str(s).ok());
	// Should prevent unexpeded behaviour in (bad) clients
	if let Some(ref token) = key {
		if token.suggested_only != body.suggested_only || token.max_depth != max_depth {
			return Err(Error::BadRequest(
				ErrorKind::InvalidParam,
				"suggested_only and max_depth cannot change on paginated requests",
			));
		}
	}
	services
		.rooms
		.spaces
		.get_client_hierarchy(
			sender_user,
			&body.room_id,
			limit.try_into().unwrap_or(10),
			key.map_or(vec![], |token| token.short_room_ids),
Jason Volk's avatar
Jason Volk committed
			max_depth.into(),
			body.suggested_only,
		)
		.await