224 lines
7.2 KiB
Rust
224 lines
7.2 KiB
Rust
//! [POST /_matrix/client/r0/createRoom](https://matrix.org/docs/spec/client_server/r0.6.0#post-matrix-client-r0-createroom)
|
|
|
|
use assign::assign;
|
|
use ruma_api::ruma_api;
|
|
#[cfg(feature = "unstable-pre-spec")]
|
|
use ruma_events::room::create::RoomType;
|
|
use ruma_events::{
|
|
room::{
|
|
create::{CreateEventContent, PreviousRoom},
|
|
power_levels::PowerLevelsEventContent,
|
|
},
|
|
AnyInitialStateEvent,
|
|
};
|
|
use ruma_identifiers::{RoomId, RoomName, RoomVersionId, UserId};
|
|
use ruma_serde::{Raw, StringEnum};
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use super::Visibility;
|
|
use crate::r0::membership::{IncomingInvite3pid, Invite3pid};
|
|
|
|
ruma_api! {
|
|
metadata: {
|
|
description: "Create a new room.",
|
|
method: POST,
|
|
name: "create_room",
|
|
path: "/_matrix/client/r0/createRoom",
|
|
rate_limited: false,
|
|
authentication: AccessToken,
|
|
}
|
|
|
|
#[derive(Default)]
|
|
request: {
|
|
/// Extra keys to be added to the content of the `m.room.create`.
|
|
#[serde(default, skip_serializing_if = "CreationContent::is_empty")]
|
|
pub creation_content: CreationContent,
|
|
|
|
/// List of state events to send to the new room.
|
|
///
|
|
/// Takes precedence over events set by preset, but gets overridden by
|
|
/// name and topic keys.
|
|
#[serde(default, skip_serializing_if = "<[_]>::is_empty")]
|
|
pub initial_state: &'a [Raw<AnyInitialStateEvent>],
|
|
|
|
/// A list of user IDs to invite to the room.
|
|
///
|
|
/// This will tell the server to invite everyone in the list to the newly created room.
|
|
#[serde(default, skip_serializing_if = "<[_]>::is_empty")]
|
|
pub invite: &'a [UserId],
|
|
|
|
/// List of third party IDs of users to invite.
|
|
#[serde(default, skip_serializing_if = "<[_]>::is_empty")]
|
|
pub invite_3pid: &'a [Invite3pid<'a>],
|
|
|
|
/// If set, this sets the `is_direct` flag on room invites.
|
|
#[serde(default, skip_serializing_if = "ruma_serde::is_default")]
|
|
pub is_direct: bool,
|
|
|
|
/// If this is included, an `m.room.name` event will be sent into the room to indicate
|
|
/// the name of the room.
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
pub name: Option<&'a RoomName>,
|
|
|
|
/// Power level content to override in the default power level event.
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
pub power_level_content_override: Option<Raw<PowerLevelsEventContent>>,
|
|
|
|
/// Convenience parameter for setting various default state events based on a preset.
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
pub preset: Option<RoomPreset>,
|
|
|
|
/// The desired room alias local part.
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
pub room_alias_name: Option<&'a str>,
|
|
|
|
/// Room version to set for the room. Defaults to homeserver's default if not specified.
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
pub room_version: Option<&'a RoomVersionId>,
|
|
|
|
/// If this is included, an `m.room.topic` event will be sent into the room to indicate
|
|
/// the topic for the room.
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
pub topic: Option<&'a str>,
|
|
|
|
/// A public visibility indicates that the room will be shown in the published room
|
|
/// list. A private visibility will hide the room from the published room list.
|
|
///
|
|
/// Defaults to `Private`.
|
|
#[serde(default, skip_serializing_if = "ruma_serde::is_default")]
|
|
pub visibility: Visibility,
|
|
}
|
|
|
|
response: {
|
|
/// The created room's ID.
|
|
pub room_id: RoomId,
|
|
}
|
|
|
|
error: crate::Error
|
|
}
|
|
|
|
impl Request<'_> {
|
|
/// Creates a new `Request` will all-default parameters.
|
|
pub fn new() -> Self {
|
|
Default::default()
|
|
}
|
|
}
|
|
|
|
impl Response {
|
|
/// Creates a new `Response` with the given room id.
|
|
pub fn new(room_id: RoomId) -> Self {
|
|
Self { room_id }
|
|
}
|
|
}
|
|
|
|
/// Extra options to be added to the `m.room.create` event.
|
|
///
|
|
/// This is the same as the event content struct for `m.room.create`, but without some fields that
|
|
/// servers are supposed to ignore.
|
|
#[derive(Clone, Debug, Deserialize, Serialize)]
|
|
#[cfg_attr(not(feature = "unstable-exhaustive-types"), non_exhaustive)]
|
|
pub struct CreationContent {
|
|
/// Whether users on other servers can join this room.
|
|
///
|
|
/// Defaults to `true` if key does not exist.
|
|
#[serde(
|
|
rename = "m.federate",
|
|
default = "ruma_serde::default_true",
|
|
skip_serializing_if = "ruma_serde::is_true"
|
|
)]
|
|
pub federate: bool,
|
|
|
|
/// A reference to the room this room replaces, if the previous room was upgraded.
|
|
#[serde(skip_serializing_if = "Option::is_none")]
|
|
pub predecessor: Option<PreviousRoom>,
|
|
|
|
/// The room type.
|
|
///
|
|
/// This is currently only used for spaces.
|
|
#[cfg(feature = "unstable-pre-spec")]
|
|
#[serde(skip_serializing_if = "Option::is_none", rename = "type")]
|
|
pub room_type: Option<RoomType>,
|
|
}
|
|
|
|
impl CreationContent {
|
|
/// Creates a new `CreationContent` with all fields defaulted.
|
|
pub fn new() -> Self {
|
|
Self {
|
|
federate: true,
|
|
predecessor: None,
|
|
#[cfg(feature = "unstable-pre-spec")]
|
|
room_type: None,
|
|
}
|
|
}
|
|
|
|
/// Given a `CreationContent` and the other fields that a homeserver has to fill, construct
|
|
/// a `CreateEventContent`.
|
|
pub fn into_event_content(
|
|
self,
|
|
creator: UserId,
|
|
room_version: RoomVersionId,
|
|
) -> CreateEventContent {
|
|
#[allow(unused_mut)]
|
|
let mut content = assign!(CreateEventContent::new(creator), {
|
|
federate: self.federate,
|
|
room_version: room_version,
|
|
predecessor: self.predecessor,
|
|
});
|
|
|
|
#[cfg(feature = "unstable-pre-spec")]
|
|
{
|
|
content.room_type = self.room_type;
|
|
}
|
|
|
|
content
|
|
}
|
|
|
|
/// Returns whether all fields have their default value.
|
|
pub fn is_empty(&self) -> bool {
|
|
let stable_fields = self.federate && self.predecessor.is_none();
|
|
|
|
#[cfg(feature = "unstable-pre-spec")]
|
|
{
|
|
stable_fields && self.room_type.is_none()
|
|
}
|
|
#[cfg(not(feature = "unstable-pre-spec"))]
|
|
{
|
|
stable_fields
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Default for CreationContent {
|
|
fn default() -> Self {
|
|
Self::new()
|
|
}
|
|
}
|
|
|
|
/// A convenience parameter for setting a few default state events.
|
|
///
|
|
/// This type can hold an arbitrary string. To check for formats that are not available as a
|
|
/// documented variant here, use its string representation, obtained through `.as_str()`.
|
|
#[derive(Clone, Debug, PartialEq, Eq, StringEnum)]
|
|
#[ruma_enum(rename_all = "snake_case")]
|
|
#[non_exhaustive]
|
|
pub enum RoomPreset {
|
|
/// `join_rules` is set to `invite` and `history_visibility` is set to `shared`.
|
|
PrivateChat,
|
|
|
|
/// `join_rules` is set to `public` and `history_visibility` is set to `shared`.
|
|
PublicChat,
|
|
|
|
/// Same as `PrivateChat`, but all initial invitees get the same power level as the creator.
|
|
TrustedPrivateChat,
|
|
|
|
#[doc(hidden)]
|
|
_Custom(String),
|
|
}
|
|
|
|
impl RoomPreset {
|
|
/// Creates a string slice from this `RoomPreset`.
|
|
pub fn as_str(&self) -> &str {
|
|
self.as_ref()
|
|
}
|
|
}
|