2020-07-30 16:14:47 +00:00
|
|
|
use super::State;
|
|
|
|
use crate::{pdu::PduBuilder, ConduitResult, Database, Error, Ruma};
|
|
|
|
use ruma::{
|
|
|
|
api::client::{
|
|
|
|
error::ErrorKind,
|
|
|
|
r0::state::{
|
2020-08-12 21:32:39 +00:00
|
|
|
get_state_events, get_state_events_for_empty_key, get_state_events_for_key,
|
|
|
|
send_state_event_for_empty_key, send_state_event_for_key,
|
2020-07-30 16:14:47 +00:00
|
|
|
},
|
|
|
|
},
|
2020-08-06 12:29:59 +00:00
|
|
|
events::{AnyStateEventContent, EventContent},
|
2020-08-21 21:19:18 +00:00
|
|
|
RoomId, UserId,
|
2020-07-30 16:14:47 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#[cfg(feature = "conduit_bin")]
|
|
|
|
use rocket::{get, put};
|
|
|
|
|
|
|
|
#[cfg_attr(
|
|
|
|
feature = "conduit_bin",
|
|
|
|
put("/_matrix/client/r0/rooms/<_>/state/<_>/<_>", data = "<body>")
|
|
|
|
)]
|
2020-08-12 21:32:39 +00:00
|
|
|
pub fn send_state_event_for_key_route(
|
2020-07-30 16:14:47 +00:00
|
|
|
db: State<'_, Database>,
|
2020-08-12 21:32:39 +00:00
|
|
|
body: Ruma<send_state_event_for_key::IncomingRequest>,
|
|
|
|
) -> ConduitResult<send_state_event_for_key::Response> {
|
2020-07-30 16:14:47 +00:00
|
|
|
let sender_id = body.sender_id.as_ref().expect("user is authenticated");
|
|
|
|
|
|
|
|
let content = serde_json::from_str::<serde_json::Value>(
|
|
|
|
body.json_body
|
|
|
|
.as_ref()
|
|
|
|
.ok_or(Error::BadRequest(ErrorKind::BadJson, "Invalid JSON body."))?
|
|
|
|
.get(),
|
|
|
|
)
|
|
|
|
.map_err(|_| Error::BadRequest(ErrorKind::BadJson, "Invalid JSON body."))?;
|
|
|
|
|
2020-08-21 21:19:18 +00:00
|
|
|
send_state_event_for_key_helper(
|
|
|
|
&db,
|
|
|
|
sender_id,
|
|
|
|
&body.content,
|
|
|
|
content,
|
|
|
|
&body.room_id,
|
|
|
|
Some(body.state_key.clone()),
|
|
|
|
)
|
2020-07-30 16:14:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg_attr(
|
|
|
|
feature = "conduit_bin",
|
|
|
|
put("/_matrix/client/r0/rooms/<_>/state/<_>", data = "<body>")
|
|
|
|
)]
|
2020-08-12 21:32:39 +00:00
|
|
|
pub fn send_state_event_for_empty_key_route(
|
2020-07-30 16:14:47 +00:00
|
|
|
db: State<'_, Database>,
|
2020-08-12 21:32:39 +00:00
|
|
|
body: Ruma<send_state_event_for_empty_key::IncomingRequest>,
|
|
|
|
) -> ConduitResult<send_state_event_for_empty_key::Response> {
|
|
|
|
// This just calls send_state_event_for_key_route
|
2020-07-30 16:14:47 +00:00
|
|
|
let Ruma {
|
2020-08-21 21:19:18 +00:00
|
|
|
body,
|
2020-07-30 16:14:47 +00:00
|
|
|
sender_id,
|
2020-08-21 21:19:18 +00:00
|
|
|
device_id: _,
|
2020-07-30 16:14:47 +00:00
|
|
|
json_body,
|
|
|
|
} = body;
|
|
|
|
|
2020-08-21 21:19:18 +00:00
|
|
|
let json = serde_json::from_str::<serde_json::Value>(
|
|
|
|
json_body
|
|
|
|
.as_ref()
|
|
|
|
.ok_or(Error::BadRequest(ErrorKind::BadJson, "Invalid JSON body."))?
|
|
|
|
.get(),
|
|
|
|
)
|
|
|
|
.map_err(|_| Error::BadRequest(ErrorKind::BadJson, "Invalid JSON body."))?;
|
|
|
|
|
2020-08-06 12:29:59 +00:00
|
|
|
Ok(send_state_event_for_empty_key::Response::new(
|
2020-08-21 21:19:18 +00:00
|
|
|
send_state_event_for_key_helper(
|
|
|
|
&db,
|
|
|
|
sender_id
|
|
|
|
.as_ref()
|
|
|
|
.expect("no user for send state empty key rout"),
|
|
|
|
&body.content,
|
|
|
|
json,
|
|
|
|
&body.room_id,
|
|
|
|
None,
|
2020-07-30 16:14:47 +00:00
|
|
|
)?
|
|
|
|
.0
|
|
|
|
.event_id,
|
2020-08-06 12:29:59 +00:00
|
|
|
)
|
2020-07-30 16:14:47 +00:00
|
|
|
.into())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg_attr(
|
|
|
|
feature = "conduit_bin",
|
|
|
|
get("/_matrix/client/r0/rooms/<_>/state", data = "<body>")
|
|
|
|
)]
|
|
|
|
pub fn get_state_events_route(
|
|
|
|
db: State<'_, Database>,
|
|
|
|
body: Ruma<get_state_events::Request>,
|
|
|
|
) -> ConduitResult<get_state_events::Response> {
|
|
|
|
let sender_id = body.sender_id.as_ref().expect("user is authenticated");
|
|
|
|
|
|
|
|
if !db.rooms.is_joined(sender_id, &body.room_id)? {
|
|
|
|
return Err(Error::BadRequest(
|
|
|
|
ErrorKind::Forbidden,
|
|
|
|
"You don't have permission to view the room state.",
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(get_state_events::Response {
|
|
|
|
room_state: db
|
|
|
|
.rooms
|
|
|
|
.room_state_full(&body.room_id)?
|
|
|
|
.values()
|
|
|
|
.map(|pdu| pdu.to_state_event())
|
|
|
|
.collect(),
|
|
|
|
}
|
|
|
|
.into())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg_attr(
|
|
|
|
feature = "conduit_bin",
|
|
|
|
get("/_matrix/client/r0/rooms/<_>/state/<_>/<_>", data = "<body>")
|
|
|
|
)]
|
|
|
|
pub fn get_state_events_for_key_route(
|
|
|
|
db: State<'_, Database>,
|
|
|
|
body: Ruma<get_state_events_for_key::Request>,
|
|
|
|
) -> ConduitResult<get_state_events_for_key::Response> {
|
|
|
|
let sender_id = body.sender_id.as_ref().expect("user is authenticated");
|
|
|
|
|
|
|
|
if !db.rooms.is_joined(sender_id, &body.room_id)? {
|
|
|
|
return Err(Error::BadRequest(
|
|
|
|
ErrorKind::Forbidden,
|
|
|
|
"You don't have permission to view the room state.",
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
let event = db
|
|
|
|
.rooms
|
|
|
|
.room_state_get(&body.room_id, &body.event_type, &body.state_key)?
|
|
|
|
.ok_or(Error::BadRequest(
|
|
|
|
ErrorKind::NotFound,
|
|
|
|
"State event not found.",
|
|
|
|
))?;
|
|
|
|
|
|
|
|
Ok(get_state_events_for_key::Response {
|
|
|
|
content: serde_json::value::to_raw_value(&event.content)
|
|
|
|
.map_err(|_| Error::bad_database("Invalid event content in database"))?,
|
|
|
|
}
|
|
|
|
.into())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg_attr(
|
|
|
|
feature = "conduit_bin",
|
|
|
|
get("/_matrix/client/r0/rooms/<_>/state/<_>", data = "<body>")
|
|
|
|
)]
|
|
|
|
pub fn get_state_events_for_empty_key_route(
|
|
|
|
db: State<'_, Database>,
|
|
|
|
body: Ruma<get_state_events_for_empty_key::Request>,
|
|
|
|
) -> ConduitResult<get_state_events_for_empty_key::Response> {
|
|
|
|
let sender_id = body.sender_id.as_ref().expect("user is authenticated");
|
|
|
|
|
|
|
|
if !db.rooms.is_joined(sender_id, &body.room_id)? {
|
|
|
|
return Err(Error::BadRequest(
|
|
|
|
ErrorKind::Forbidden,
|
|
|
|
"You don't have permission to view the room state.",
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
let event = db
|
|
|
|
.rooms
|
|
|
|
.room_state_get(&body.room_id, &body.event_type, "")?
|
|
|
|
.ok_or(Error::BadRequest(
|
|
|
|
ErrorKind::NotFound,
|
|
|
|
"State event not found.",
|
|
|
|
))?;
|
|
|
|
|
|
|
|
Ok(get_state_events_for_empty_key::Response {
|
|
|
|
content: serde_json::value::to_raw_value(&event)
|
|
|
|
.map_err(|_| Error::bad_database("Invalid event content in database"))?,
|
|
|
|
}
|
|
|
|
.into())
|
|
|
|
}
|
2020-08-21 21:19:18 +00:00
|
|
|
|
|
|
|
pub fn send_state_event_for_key_helper(
|
|
|
|
db: &Database,
|
|
|
|
sender: &UserId,
|
|
|
|
content: &AnyStateEventContent,
|
|
|
|
json: serde_json::Value,
|
|
|
|
room_id: &RoomId,
|
|
|
|
state_key: Option<String>,
|
|
|
|
) -> ConduitResult<send_state_event_for_key::Response> {
|
|
|
|
let sender_id = sender;
|
|
|
|
|
|
|
|
if let AnyStateEventContent::RoomCanonicalAlias(canonical_alias) = content {
|
|
|
|
let mut aliases = canonical_alias.alt_aliases.clone();
|
|
|
|
|
|
|
|
if let Some(alias) = canonical_alias.alias.clone() {
|
|
|
|
aliases.push(alias);
|
|
|
|
}
|
|
|
|
|
|
|
|
for alias in aliases {
|
|
|
|
if alias.server_name() != db.globals.server_name()
|
|
|
|
|| db
|
|
|
|
.rooms
|
|
|
|
.id_from_alias(&alias)?
|
|
|
|
.filter(|room| room == room_id) // Make sure it's the right room
|
|
|
|
.is_none()
|
|
|
|
{
|
|
|
|
return Err(Error::BadRequest(
|
|
|
|
ErrorKind::Forbidden,
|
|
|
|
"You are only allowed to send canonical_alias \
|
|
|
|
events when it's aliases already exists",
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let event_id = db.rooms.build_and_append_pdu(
|
|
|
|
PduBuilder {
|
|
|
|
room_id: room_id.clone(),
|
|
|
|
sender: sender_id.clone(),
|
|
|
|
event_type: content.event_type().into(),
|
|
|
|
content: json,
|
|
|
|
unsigned: None,
|
|
|
|
state_key,
|
|
|
|
redacts: None,
|
|
|
|
},
|
|
|
|
&db.globals,
|
|
|
|
&db.account_data,
|
|
|
|
)?;
|
|
|
|
|
|
|
|
Ok(send_state_event_for_key::Response::new(event_id).into())
|
|
|
|
}
|