2020-05-03 16:25:31 +01:00
|
|
|
mod edus;
|
|
|
|
|
|
|
|
pub use edus::RoomEdus;
|
|
|
|
|
|
|
|
use crate::{utils, Error, PduEvent, Result};
|
2020-05-24 17:25:52 +01:00
|
|
|
use log::error;
|
2020-06-05 17:19:26 +01:00
|
|
|
use ruma::{
|
2020-06-09 14:13:17 +01:00
|
|
|
api::client::error::ErrorKind,
|
2020-06-05 17:19:26 +01:00
|
|
|
events::{
|
|
|
|
room::{
|
|
|
|
join_rules, member,
|
|
|
|
power_levels::{self, PowerLevelsEventContent},
|
|
|
|
redaction,
|
|
|
|
},
|
2020-07-26 14:41:28 +01:00
|
|
|
EventType,
|
2020-05-24 17:25:52 +01:00
|
|
|
},
|
2020-07-26 14:41:28 +01:00
|
|
|
EventId, Raw, RoomAliasId, RoomId, UserId,
|
2020-05-24 17:25:52 +01:00
|
|
|
};
|
2020-05-26 09:27:51 +01:00
|
|
|
use sled::IVec;
|
2020-05-03 16:25:31 +01:00
|
|
|
use std::{
|
2020-05-24 17:25:52 +01:00
|
|
|
collections::{BTreeMap, HashMap},
|
2020-05-03 16:25:31 +01:00
|
|
|
convert::{TryFrom, TryInto},
|
|
|
|
mem,
|
|
|
|
};
|
|
|
|
|
|
|
|
pub struct Rooms {
|
|
|
|
pub edus: edus::RoomEdus,
|
|
|
|
pub(super) pduid_pdu: sled::Tree, // PduId = RoomId + Count
|
|
|
|
pub(super) eventid_pduid: sled::Tree,
|
|
|
|
pub(super) roomid_pduleaves: sled::Tree,
|
2020-05-08 20:13:52 +01:00
|
|
|
pub(super) roomstateid_pdu: sled::Tree, // RoomStateId = Room + StateType + StateKey
|
2020-05-03 16:25:31 +01:00
|
|
|
|
2020-05-24 07:30:57 +01:00
|
|
|
pub(super) alias_roomid: sled::Tree,
|
2020-05-25 22:24:13 +01:00
|
|
|
pub(super) aliasid_alias: sled::Tree, // AliasId = RoomId + Count
|
|
|
|
pub(super) publicroomids: sled::Tree,
|
2020-05-24 07:30:57 +01:00
|
|
|
|
2020-05-03 16:25:31 +01:00
|
|
|
pub(super) userroomid_joined: sled::Tree,
|
|
|
|
pub(super) roomuserid_joined: sled::Tree,
|
|
|
|
pub(super) userroomid_invited: sled::Tree,
|
|
|
|
pub(super) roomuserid_invited: sled::Tree,
|
|
|
|
pub(super) userroomid_left: sled::Tree,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Rooms {
|
|
|
|
/// Checks if a room exists.
|
|
|
|
pub fn exists(&self, room_id: &RoomId) -> Result<bool> {
|
|
|
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
|
|
|
prefix.push(0xff);
|
|
|
|
|
2020-05-08 20:13:52 +01:00
|
|
|
// Look for PDUs in that room.
|
2020-05-03 16:25:31 +01:00
|
|
|
Ok(self
|
|
|
|
.pduid_pdu
|
|
|
|
.get_gt(&prefix)?
|
|
|
|
.filter(|(k, _)| k.starts_with(&prefix))
|
|
|
|
.is_some())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the full room state.
|
2020-06-12 12:18:25 +01:00
|
|
|
pub fn room_state_full(
|
|
|
|
&self,
|
|
|
|
room_id: &RoomId,
|
|
|
|
) -> Result<HashMap<(EventType, String), PduEvent>> {
|
2020-05-03 16:25:31 +01:00
|
|
|
let mut hashmap = HashMap::new();
|
|
|
|
for pdu in self
|
|
|
|
.roomstateid_pdu
|
|
|
|
.scan_prefix(&room_id.to_string().as_bytes())
|
|
|
|
.values()
|
2020-06-09 14:13:17 +01:00
|
|
|
.map(|value| {
|
|
|
|
Ok::<_, Error>(
|
|
|
|
serde_json::from_slice::<PduEvent>(&value?)
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("Invalid PDU in db."))?,
|
2020-06-09 14:13:17 +01:00
|
|
|
)
|
|
|
|
})
|
2020-05-03 16:25:31 +01:00
|
|
|
{
|
|
|
|
let pdu = pdu?;
|
2020-06-11 09:03:08 +01:00
|
|
|
let state_key = pdu.state_key.clone().ok_or_else(|| {
|
|
|
|
Error::bad_database("Room state contains event without state_key.")
|
|
|
|
})?;
|
2020-06-09 14:13:17 +01:00
|
|
|
hashmap.insert((pdu.kind.clone(), state_key), pdu);
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
|
|
|
Ok(hashmap)
|
|
|
|
}
|
|
|
|
|
2020-07-10 07:49:05 +01:00
|
|
|
/// Returns the all state entries for this type.
|
2020-06-12 12:18:25 +01:00
|
|
|
pub fn room_state_type(
|
|
|
|
&self,
|
|
|
|
room_id: &RoomId,
|
|
|
|
event_type: &EventType,
|
|
|
|
) -> Result<HashMap<String, PduEvent>> {
|
|
|
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
|
|
|
prefix.push(0xff);
|
|
|
|
prefix.extend_from_slice(&event_type.to_string().as_bytes());
|
|
|
|
|
|
|
|
let mut hashmap = HashMap::new();
|
|
|
|
for pdu in self
|
|
|
|
.roomstateid_pdu
|
|
|
|
.scan_prefix(&prefix)
|
|
|
|
.values()
|
|
|
|
.map(|value| {
|
|
|
|
Ok::<_, Error>(
|
|
|
|
serde_json::from_slice::<PduEvent>(&value?)
|
|
|
|
.map_err(|_| Error::bad_database("Invalid PDU in db."))?,
|
|
|
|
)
|
|
|
|
})
|
|
|
|
{
|
|
|
|
let pdu = pdu?;
|
|
|
|
let state_key = pdu.state_key.clone().ok_or_else(|| {
|
|
|
|
Error::bad_database("Room state contains event without state_key.")
|
|
|
|
})?;
|
|
|
|
hashmap.insert(state_key, pdu);
|
|
|
|
}
|
|
|
|
Ok(hashmap)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the full room state.
|
|
|
|
pub fn room_state_get(
|
|
|
|
&self,
|
|
|
|
room_id: &RoomId,
|
|
|
|
event_type: &EventType,
|
|
|
|
state_key: &str,
|
|
|
|
) -> Result<Option<PduEvent>> {
|
|
|
|
let mut key = room_id.to_string().as_bytes().to_vec();
|
|
|
|
key.push(0xff);
|
|
|
|
key.extend_from_slice(&event_type.to_string().as_bytes());
|
|
|
|
key.push(0xff);
|
|
|
|
key.extend_from_slice(&state_key.as_bytes());
|
|
|
|
|
|
|
|
self.roomstateid_pdu.get(&key)?.map_or(Ok(None), |value| {
|
|
|
|
Ok::<_, Error>(Some(
|
|
|
|
serde_json::from_slice::<PduEvent>(&value)
|
|
|
|
.map_err(|_| Error::bad_database("Invalid PDU in db."))?,
|
|
|
|
))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-05-03 16:25:31 +01:00
|
|
|
/// Returns the `count` of this pdu's id.
|
|
|
|
pub fn get_pdu_count(&self, event_id: &EventId) -> Result<Option<u64>> {
|
2020-06-09 14:13:17 +01:00
|
|
|
self.eventid_pduid
|
2020-05-03 16:25:31 +01:00
|
|
|
.get(event_id.to_string().as_bytes())?
|
2020-06-09 14:13:17 +01:00
|
|
|
.map_or(Ok(None), |pdu_id| {
|
|
|
|
Ok(Some(
|
|
|
|
utils::u64_from_bytes(
|
|
|
|
&pdu_id[pdu_id.len() - mem::size_of::<u64>()..pdu_id.len()],
|
|
|
|
)
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("PDU has invalid count bytes."))?,
|
2020-06-09 14:13:17 +01:00
|
|
|
))
|
|
|
|
})
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the json of a pdu.
|
|
|
|
pub fn get_pdu_json(&self, event_id: &EventId) -> Result<Option<serde_json::Value>> {
|
|
|
|
self.eventid_pduid
|
|
|
|
.get(event_id.to_string().as_bytes())?
|
|
|
|
.map_or(Ok(None), |pdu_id| {
|
2020-06-09 14:13:17 +01:00
|
|
|
Ok(Some(
|
2020-06-11 09:03:08 +01:00
|
|
|
serde_json::from_slice(&self.pduid_pdu.get(pdu_id)?.ok_or_else(|| {
|
|
|
|
Error::bad_database("eventid_pduid points to nonexistent pdu.")
|
|
|
|
})?)
|
|
|
|
.map_err(|_| Error::bad_database("Invalid PDU in db."))?,
|
2020-06-09 14:13:17 +01:00
|
|
|
))
|
2020-05-03 16:25:31 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-05-26 09:27:51 +01:00
|
|
|
/// Returns the pdu's id.
|
|
|
|
pub fn get_pdu_id(&self, event_id: &EventId) -> Result<Option<IVec>> {
|
|
|
|
self.eventid_pduid
|
|
|
|
.get(event_id.to_string().as_bytes())?
|
|
|
|
.map_or(Ok(None), |pdu_id| Ok(Some(pdu_id)))
|
|
|
|
}
|
|
|
|
|
2020-05-24 17:25:52 +01:00
|
|
|
/// Returns the pdu.
|
|
|
|
pub fn get_pdu(&self, event_id: &EventId) -> Result<Option<PduEvent>> {
|
|
|
|
self.eventid_pduid
|
|
|
|
.get(event_id.to_string().as_bytes())?
|
|
|
|
.map_or(Ok(None), |pdu_id| {
|
2020-06-09 14:13:17 +01:00
|
|
|
Ok(Some(
|
2020-06-11 09:03:08 +01:00
|
|
|
serde_json::from_slice(&self.pduid_pdu.get(pdu_id)?.ok_or_else(|| {
|
|
|
|
Error::bad_database("eventid_pduid points to nonexistent pdu.")
|
|
|
|
})?)
|
|
|
|
.map_err(|_| Error::bad_database("Invalid PDU in db."))?,
|
2020-06-09 14:13:17 +01:00
|
|
|
))
|
2020-05-24 17:25:52 +01:00
|
|
|
})
|
|
|
|
}
|
2020-05-26 09:27:51 +01:00
|
|
|
/// Returns the pdu.
|
|
|
|
pub fn get_pdu_from_id(&self, pdu_id: &IVec) -> Result<Option<PduEvent>> {
|
2020-06-09 14:13:17 +01:00
|
|
|
self.pduid_pdu.get(pdu_id)?.map_or(Ok(None), |pdu| {
|
|
|
|
Ok(Some(
|
|
|
|
serde_json::from_slice(&pdu)
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("Invalid PDU in db."))?,
|
2020-06-09 14:13:17 +01:00
|
|
|
))
|
|
|
|
})
|
2020-05-26 09:27:51 +01:00
|
|
|
}
|
|
|
|
|
2020-06-09 14:13:17 +01:00
|
|
|
/// Removes a pdu and creates a new one with the same id.
|
|
|
|
fn replace_pdu(&self, pdu_id: &IVec, pdu: &PduEvent) -> Result<()> {
|
2020-05-26 09:27:51 +01:00
|
|
|
if self.pduid_pdu.get(&pdu_id)?.is_some() {
|
2020-06-09 14:13:17 +01:00
|
|
|
self.pduid_pdu.insert(
|
|
|
|
&pdu_id,
|
|
|
|
&*serde_json::to_string(pdu).expect("PduEvent::to_string always works"),
|
|
|
|
)?;
|
2020-05-26 09:27:51 +01:00
|
|
|
Ok(())
|
|
|
|
} else {
|
2020-06-09 14:13:17 +01:00
|
|
|
Err(Error::BadRequest(
|
|
|
|
ErrorKind::NotFound,
|
|
|
|
"PDU does not exist.",
|
|
|
|
))
|
2020-05-26 09:27:51 +01:00
|
|
|
}
|
|
|
|
}
|
2020-05-24 17:25:52 +01:00
|
|
|
|
2020-05-03 16:25:31 +01:00
|
|
|
/// Returns the leaf pdus of a room.
|
|
|
|
pub fn get_pdu_leaves(&self, room_id: &RoomId) -> Result<Vec<EventId>> {
|
|
|
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
|
|
|
prefix.push(0xff);
|
|
|
|
|
|
|
|
let mut events = Vec::new();
|
|
|
|
|
|
|
|
for event in self
|
|
|
|
.roomid_pduleaves
|
|
|
|
.scan_prefix(prefix)
|
|
|
|
.values()
|
2020-06-09 14:13:17 +01:00
|
|
|
.map(|bytes| {
|
|
|
|
Ok::<_, Error>(
|
2020-06-11 09:03:08 +01:00
|
|
|
EventId::try_from(utils::string_from_bytes(&bytes?).map_err(|_| {
|
|
|
|
Error::bad_database("EventID in roomid_pduleaves is invalid unicode.")
|
|
|
|
})?)
|
|
|
|
.map_err(|_| Error::bad_database("EventId in roomid_pduleaves is invalid."))?,
|
2020-06-09 14:13:17 +01:00
|
|
|
)
|
|
|
|
})
|
2020-05-03 16:25:31 +01:00
|
|
|
{
|
|
|
|
events.push(event?);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(events)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Replace the leaves of a room with a new event.
|
|
|
|
pub fn replace_pdu_leaves(&self, room_id: &RoomId, event_id: &EventId) -> Result<()> {
|
|
|
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
|
|
|
prefix.push(0xff);
|
|
|
|
|
|
|
|
for key in self.roomid_pduleaves.scan_prefix(&prefix).keys() {
|
|
|
|
self.roomid_pduleaves.remove(key?)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
prefix.extend_from_slice(event_id.to_string().as_bytes());
|
|
|
|
self.roomid_pduleaves
|
|
|
|
.insert(&prefix, &*event_id.to_string())?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a new persisted data unit and adds it to a room.
|
|
|
|
pub fn append_pdu(
|
|
|
|
&self,
|
|
|
|
room_id: RoomId,
|
|
|
|
sender: UserId,
|
|
|
|
event_type: EventType,
|
|
|
|
content: serde_json::Value,
|
|
|
|
unsigned: Option<serde_json::Map<String, serde_json::Value>>,
|
|
|
|
state_key: Option<String>,
|
2020-05-26 09:27:51 +01:00
|
|
|
redacts: Option<EventId>,
|
2020-05-03 16:25:31 +01:00
|
|
|
globals: &super::globals::Globals,
|
|
|
|
) -> Result<EventId> {
|
2020-05-24 17:25:52 +01:00
|
|
|
// TODO: Make sure this isn't called twice in parallel
|
|
|
|
let prev_events = self.get_pdu_leaves(&room_id)?;
|
|
|
|
|
2020-05-03 16:25:31 +01:00
|
|
|
// Is the event authorized?
|
2020-05-24 17:25:52 +01:00
|
|
|
if let Some(state_key) = &state_key {
|
|
|
|
let power_levels = self
|
2020-06-12 12:18:25 +01:00
|
|
|
.room_state_get(&room_id, &EventType::RoomPowerLevels, "")?
|
2020-05-24 17:25:52 +01:00
|
|
|
.map_or_else(
|
|
|
|
|| {
|
|
|
|
Ok::<_, Error>(power_levels::PowerLevelsEventContent {
|
|
|
|
ban: 50.into(),
|
|
|
|
events: BTreeMap::new(),
|
|
|
|
events_default: 0.into(),
|
|
|
|
invite: 50.into(),
|
|
|
|
kick: 50.into(),
|
|
|
|
redact: 50.into(),
|
|
|
|
state_default: 0.into(),
|
|
|
|
users: BTreeMap::new(),
|
|
|
|
users_default: 0.into(),
|
|
|
|
notifications:
|
2020-06-05 17:19:26 +01:00
|
|
|
ruma::events::room::power_levels::NotificationPowerLevels {
|
2020-05-24 17:25:52 +01:00
|
|
|
room: 50.into(),
|
|
|
|
},
|
|
|
|
})
|
|
|
|
},
|
|
|
|
|power_levels| {
|
2020-07-26 14:41:28 +01:00
|
|
|
Ok(serde_json::from_value::<Raw<PowerLevelsEventContent>>(
|
|
|
|
power_levels.content.clone(),
|
2020-05-24 17:25:52 +01:00
|
|
|
)
|
2020-07-26 14:41:28 +01:00
|
|
|
.expect("Raw::from_value always works.")
|
|
|
|
.deserialize()
|
|
|
|
.map_err(|_| Error::bad_database("Invalid PowerLevels event in db."))?)
|
2020-05-24 17:25:52 +01:00
|
|
|
},
|
|
|
|
)?;
|
2020-06-09 14:13:17 +01:00
|
|
|
let sender_membership = self
|
2020-06-12 12:18:25 +01:00
|
|
|
.room_state_get(&room_id, &EventType::RoomMember, &sender.to_string())?
|
2020-06-09 14:13:17 +01:00
|
|
|
.map_or(Ok::<_, Error>(member::MembershipState::Leave), |pdu| {
|
2020-07-26 14:41:28 +01:00
|
|
|
Ok(serde_json::from_value::<Raw<member::MemberEventContent>>(
|
|
|
|
pdu.content.clone(),
|
2020-06-09 14:13:17 +01:00
|
|
|
)
|
2020-07-26 14:41:28 +01:00
|
|
|
.expect("Raw::from_value always works.")
|
|
|
|
.deserialize()
|
|
|
|
.map_err(|_| Error::bad_database("Invalid Member event in db."))?
|
|
|
|
.membership)
|
2020-06-09 14:13:17 +01:00
|
|
|
})?;
|
|
|
|
|
|
|
|
let sender_power = power_levels.users.get(&sender).map_or_else(
|
|
|
|
|| {
|
|
|
|
if sender_membership != member::MembershipState::Join {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(&power_levels.users_default)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// If it's okay, wrap with Some(_)
|
|
|
|
Some,
|
|
|
|
);
|
|
|
|
|
2020-07-26 19:41:10 +01:00
|
|
|
// Is the event allowed?
|
2020-06-09 14:13:17 +01:00
|
|
|
if !match event_type {
|
2020-07-26 19:41:10 +01:00
|
|
|
EventType::RoomEncryption => {
|
|
|
|
// Don't allow encryption events when it's disabled
|
|
|
|
!globals.encryption_disabled()
|
|
|
|
}
|
2020-06-09 14:13:17 +01:00
|
|
|
EventType::RoomMember => {
|
|
|
|
let target_user_id = UserId::try_from(&**state_key).map_err(|_| {
|
|
|
|
Error::BadRequest(
|
|
|
|
ErrorKind::InvalidParam,
|
|
|
|
"State key of member event does not contain user id.",
|
2020-05-24 17:25:52 +01:00
|
|
|
)
|
|
|
|
})?;
|
|
|
|
|
2020-06-09 14:13:17 +01:00
|
|
|
let current_membership = self
|
2020-06-12 12:18:25 +01:00
|
|
|
.room_state_get(
|
|
|
|
&room_id,
|
|
|
|
&EventType::RoomMember,
|
|
|
|
&target_user_id.to_string(),
|
|
|
|
)?
|
2020-06-09 14:13:17 +01:00
|
|
|
.map_or(Ok::<_, Error>(member::MembershipState::Leave), |pdu| {
|
2020-07-26 14:41:28 +01:00
|
|
|
Ok(serde_json::from_value::<Raw<member::MemberEventContent>>(
|
|
|
|
pdu.content.clone(),
|
2020-06-09 14:13:17 +01:00
|
|
|
)
|
2020-07-26 14:41:28 +01:00
|
|
|
.expect("Raw::from_value always works.")
|
|
|
|
.deserialize()
|
|
|
|
.map_err(|_| Error::bad_database("Invalid Member event in db."))?
|
|
|
|
.membership)
|
2020-06-09 14:13:17 +01:00
|
|
|
})?;
|
|
|
|
|
2020-07-26 14:41:28 +01:00
|
|
|
let target_membership =
|
|
|
|
serde_json::from_value::<Raw<member::MemberEventContent>>(content.clone())
|
|
|
|
.expect("Raw::from_value always works.")
|
|
|
|
.deserialize()
|
|
|
|
.map_err(|_| Error::bad_database("Invalid Member event in db."))?
|
|
|
|
.membership;
|
2020-06-09 14:13:17 +01:00
|
|
|
|
|
|
|
let target_power = power_levels.users.get(&target_user_id).map_or_else(
|
|
|
|
|| {
|
|
|
|
if target_membership != member::MembershipState::Join {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(&power_levels.users_default)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// If it's okay, wrap with Some(_)
|
|
|
|
Some,
|
|
|
|
);
|
2020-05-24 17:25:52 +01:00
|
|
|
|
2020-06-09 14:13:17 +01:00
|
|
|
let join_rules =
|
2020-06-12 12:18:25 +01:00
|
|
|
self.room_state_get(&room_id, &EventType::RoomJoinRules, "")?
|
2020-06-09 14:13:17 +01:00
|
|
|
.map_or(Ok::<_, Error>(join_rules::JoinRule::Public), |pdu| {
|
|
|
|
Ok(serde_json::from_value::<
|
2020-07-26 14:41:28 +01:00
|
|
|
Raw<join_rules::JoinRulesEventContent>,
|
2020-05-24 17:25:52 +01:00
|
|
|
>(pdu.content.clone())
|
2020-07-26 14:41:28 +01:00
|
|
|
.expect("Raw::from_value always works.")
|
2020-05-24 17:25:52 +01:00
|
|
|
.deserialize()
|
2020-06-09 14:13:17 +01:00
|
|
|
.map_err(|_| {
|
2020-06-11 09:03:08 +01:00
|
|
|
Error::bad_database("Database contains invalid JoinRules event")
|
2020-06-09 14:13:17 +01:00
|
|
|
})?
|
|
|
|
.join_rule)
|
|
|
|
})?;
|
|
|
|
|
|
|
|
let authorized = if target_membership == member::MembershipState::Join {
|
|
|
|
let mut prev_events = prev_events.iter();
|
|
|
|
let prev_event = self
|
|
|
|
.get_pdu(prev_events.next().ok_or(Error::BadRequest(
|
|
|
|
ErrorKind::Unknown,
|
|
|
|
"Membership can't be the first event",
|
|
|
|
))?)?
|
2020-06-11 09:03:08 +01:00
|
|
|
.ok_or_else(|| {
|
|
|
|
Error::bad_database("PDU leaf points to invalid event!")
|
|
|
|
})?;
|
2020-06-09 14:13:17 +01:00
|
|
|
if prev_event.kind == EventType::RoomCreate
|
|
|
|
&& prev_event.prev_events.is_empty()
|
|
|
|
{
|
|
|
|
true
|
|
|
|
} else if sender != target_user_id {
|
|
|
|
false
|
|
|
|
} else if let member::MembershipState::Ban = current_membership {
|
|
|
|
false
|
|
|
|
} else {
|
|
|
|
join_rules == join_rules::JoinRule::Invite
|
|
|
|
&& (current_membership == member::MembershipState::Join
|
|
|
|
|| current_membership == member::MembershipState::Invite)
|
|
|
|
|| join_rules == join_rules::JoinRule::Public
|
|
|
|
}
|
|
|
|
} else if target_membership == member::MembershipState::Invite {
|
|
|
|
if let Some(third_party_invite_json) = content.get("third_party_invite") {
|
|
|
|
if current_membership == member::MembershipState::Ban {
|
2020-05-24 17:25:52 +01:00
|
|
|
false
|
|
|
|
} else {
|
2020-06-09 14:13:17 +01:00
|
|
|
let _third_party_invite =
|
|
|
|
serde_json::from_value::<member::ThirdPartyInvite>(
|
|
|
|
third_party_invite_json.clone(),
|
|
|
|
)
|
|
|
|
.map_err(|_| {
|
|
|
|
Error::BadRequest(
|
|
|
|
ErrorKind::InvalidParam,
|
|
|
|
"ThirdPartyInvite is invalid",
|
|
|
|
)
|
|
|
|
})?;
|
|
|
|
todo!("handle third party invites");
|
2020-05-24 17:25:52 +01:00
|
|
|
}
|
2020-06-09 14:13:17 +01:00
|
|
|
} else if sender_membership != member::MembershipState::Join
|
|
|
|
|| current_membership == member::MembershipState::Join
|
|
|
|
|| current_membership == member::MembershipState::Ban
|
|
|
|
{
|
|
|
|
false
|
2020-05-24 17:25:52 +01:00
|
|
|
} else {
|
2020-06-09 14:13:17 +01:00
|
|
|
sender_power
|
|
|
|
.filter(|&p| p >= &power_levels.invite)
|
|
|
|
.is_some()
|
|
|
|
}
|
|
|
|
} else if target_membership == member::MembershipState::Leave {
|
|
|
|
if sender == target_user_id {
|
|
|
|
current_membership == member::MembershipState::Join
|
|
|
|
|| current_membership == member::MembershipState::Invite
|
|
|
|
} else if sender_membership != member::MembershipState::Join
|
|
|
|
|| current_membership == member::MembershipState::Ban
|
|
|
|
&& sender_power.filter(|&p| p < &power_levels.ban).is_some()
|
|
|
|
{
|
2020-05-24 17:25:52 +01:00
|
|
|
false
|
2020-06-09 14:13:17 +01:00
|
|
|
} else {
|
|
|
|
sender_power.filter(|&p| p >= &power_levels.kick).is_some()
|
|
|
|
&& target_power < sender_power
|
|
|
|
}
|
|
|
|
} else if target_membership == member::MembershipState::Ban {
|
|
|
|
if sender_membership != member::MembershipState::Join {
|
|
|
|
false
|
|
|
|
} else {
|
|
|
|
sender_power.filter(|&p| p >= &power_levels.ban).is_some()
|
|
|
|
&& target_power < sender_power
|
2020-05-24 17:25:52 +01:00
|
|
|
}
|
2020-06-09 14:13:17 +01:00
|
|
|
} else {
|
|
|
|
false
|
|
|
|
};
|
2020-05-25 22:24:13 +01:00
|
|
|
|
2020-06-09 14:13:17 +01:00
|
|
|
if authorized {
|
|
|
|
// Update our membership info
|
|
|
|
self.update_membership(&room_id, &target_user_id, &target_membership)?;
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
2020-06-09 14:13:17 +01:00
|
|
|
|
|
|
|
authorized
|
|
|
|
}
|
|
|
|
EventType::RoomCreate => prev_events.is_empty(),
|
|
|
|
// Not allow any of the following events if the sender is not joined.
|
|
|
|
_ if sender_membership != member::MembershipState::Join => false,
|
|
|
|
|
|
|
|
_ => {
|
|
|
|
// TODO
|
|
|
|
sender_power.unwrap_or(&power_levels.users_default)
|
|
|
|
>= &power_levels.state_default
|
2020-05-24 17:25:52 +01:00
|
|
|
}
|
2020-06-09 14:13:17 +01:00
|
|
|
} {
|
|
|
|
error!("Unauthorized");
|
|
|
|
// Not authorized
|
|
|
|
return Err(Error::BadRequest(
|
|
|
|
ErrorKind::Forbidden,
|
|
|
|
"Event is not authorized",
|
|
|
|
));
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
2020-05-24 17:25:52 +01:00
|
|
|
} else if !self.is_joined(&sender, &room_id)? {
|
2020-06-09 14:13:17 +01:00
|
|
|
// TODO: auth rules apply to all events, not only those with a state key
|
|
|
|
error!("Unauthorized");
|
|
|
|
return Err(Error::BadRequest(
|
|
|
|
ErrorKind::Forbidden,
|
|
|
|
"Event is not authorized",
|
|
|
|
));
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Our depth is the maximum depth of prev_events + 1
|
|
|
|
let depth = prev_events
|
|
|
|
.iter()
|
|
|
|
.filter_map(|event_id| Some(self.get_pdu_json(event_id).ok()??.get("depth")?.as_u64()?))
|
|
|
|
.max()
|
|
|
|
.unwrap_or(0_u64)
|
|
|
|
+ 1;
|
|
|
|
|
|
|
|
let mut unsigned = unsigned.unwrap_or_default();
|
|
|
|
if let Some(state_key) = &state_key {
|
2020-06-12 12:18:25 +01:00
|
|
|
if let Some(prev_pdu) = self.room_state_get(&room_id, &event_type, &state_key)? {
|
2020-05-03 16:25:31 +01:00
|
|
|
unsigned.insert("prev_content".to_owned(), prev_pdu.content.clone());
|
2020-06-26 09:07:02 +01:00
|
|
|
unsigned.insert(
|
|
|
|
"prev_sender".to_owned(),
|
|
|
|
serde_json::to_value(prev_pdu.sender).expect("UserId::to_value always works"),
|
|
|
|
);
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut pdu = PduEvent {
|
|
|
|
event_id: EventId::try_from("$thiswillbefilledinlater").expect("we know this is valid"),
|
|
|
|
room_id: room_id.clone(),
|
|
|
|
sender: sender.clone(),
|
2020-07-21 19:04:39 +01:00
|
|
|
origin: globals.server_name().to_owned(),
|
2020-05-03 16:25:31 +01:00
|
|
|
origin_server_ts: utils::millis_since_unix_epoch()
|
|
|
|
.try_into()
|
2020-06-09 14:13:17 +01:00
|
|
|
.expect("time is valid"),
|
2020-06-01 19:58:49 +01:00
|
|
|
kind: event_type.clone(),
|
|
|
|
content: content.clone(),
|
2020-05-03 16:25:31 +01:00
|
|
|
state_key,
|
|
|
|
prev_events,
|
|
|
|
depth: depth
|
|
|
|
.try_into()
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("Depth is invalid"))?,
|
2020-05-03 16:25:31 +01:00
|
|
|
auth_events: Vec::new(),
|
2020-06-01 19:58:49 +01:00
|
|
|
redacts: redacts.clone(),
|
2020-05-03 16:25:31 +01:00
|
|
|
unsigned,
|
2020-06-21 20:58:42 +01:00
|
|
|
hashes: ruma::events::pdu::EventHash {
|
2020-05-03 16:25:31 +01:00
|
|
|
sha256: "aaa".to_owned(),
|
|
|
|
},
|
|
|
|
signatures: HashMap::new(),
|
|
|
|
};
|
|
|
|
|
|
|
|
// Generate event id
|
|
|
|
pdu.event_id = EventId::try_from(&*format!(
|
|
|
|
"${}",
|
2020-06-09 14:13:17 +01:00
|
|
|
ruma::signatures::reference_hash(
|
|
|
|
&serde_json::to_value(&pdu).expect("event is valid, we just created it")
|
|
|
|
)
|
|
|
|
.expect("ruma can calculate reference hashes")
|
2020-05-03 16:25:31 +01:00
|
|
|
))
|
2020-06-09 14:13:17 +01:00
|
|
|
.expect("ruma's reference hashes are valid event ids");
|
2020-05-03 16:25:31 +01:00
|
|
|
|
2020-06-09 14:13:17 +01:00
|
|
|
let mut pdu_json = serde_json::to_value(&pdu).expect("event is valid, we just created it");
|
2020-06-05 17:19:26 +01:00
|
|
|
ruma::signatures::hash_and_sign_event(
|
2020-06-21 20:58:42 +01:00
|
|
|
globals.server_name().as_str(),
|
2020-05-06 14:36:44 +01:00
|
|
|
globals.keypair(),
|
|
|
|
&mut pdu_json,
|
|
|
|
)
|
2020-06-09 14:13:17 +01:00
|
|
|
.expect("event is valid, we just created it");
|
2020-05-03 16:25:31 +01:00
|
|
|
|
|
|
|
self.replace_pdu_leaves(&room_id, &pdu.event_id)?;
|
|
|
|
|
|
|
|
// Increment the last index and use that
|
|
|
|
// This is also the next_batch/since value
|
|
|
|
let index = globals.next_count()?;
|
|
|
|
|
|
|
|
let mut pdu_id = room_id.to_string().as_bytes().to_vec();
|
|
|
|
pdu_id.push(0xff);
|
|
|
|
pdu_id.extend_from_slice(&index.to_be_bytes());
|
|
|
|
|
|
|
|
self.pduid_pdu.insert(&pdu_id, &*pdu_json.to_string())?;
|
|
|
|
|
|
|
|
self.eventid_pduid
|
2020-05-11 11:33:25 +01:00
|
|
|
.insert(pdu.event_id.to_string(), pdu_id)?;
|
2020-05-03 16:25:31 +01:00
|
|
|
|
|
|
|
if let Some(state_key) = pdu.state_key {
|
|
|
|
let mut key = room_id.to_string().as_bytes().to_vec();
|
|
|
|
key.push(0xff);
|
|
|
|
key.extend_from_slice(pdu.kind.to_string().as_bytes());
|
|
|
|
key.push(0xff);
|
2020-05-11 11:33:25 +01:00
|
|
|
key.extend_from_slice(state_key.as_bytes());
|
2020-05-03 16:25:31 +01:00
|
|
|
self.roomstateid_pdu.insert(key, &*pdu_json.to_string())?;
|
|
|
|
}
|
|
|
|
|
2020-06-01 19:58:49 +01:00
|
|
|
match event_type {
|
|
|
|
EventType::RoomRedaction => {
|
|
|
|
if let Some(redact_id) = &redacts {
|
|
|
|
// TODO: Reason
|
2020-07-26 14:41:28 +01:00
|
|
|
let _reason =
|
|
|
|
serde_json::from_value::<Raw<redaction::RedactionEventContent>>(content)
|
|
|
|
.expect("Raw::from_value always works.")
|
|
|
|
.deserialize()
|
|
|
|
.map_err(|_| {
|
|
|
|
Error::BadRequest(
|
|
|
|
ErrorKind::InvalidParam,
|
|
|
|
"Invalid redaction event content.",
|
|
|
|
)
|
|
|
|
})?
|
|
|
|
.reason;
|
2020-06-01 19:58:49 +01:00
|
|
|
|
|
|
|
self.redact_pdu(&redact_id)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
|
2020-05-03 16:25:31 +01:00
|
|
|
self.edus.room_read_set(&room_id, &sender, index)?;
|
|
|
|
|
|
|
|
Ok(pdu.event_id)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns an iterator over all PDUs in a room.
|
2020-06-16 11:11:38 +01:00
|
|
|
pub fn all_pdus(
|
|
|
|
&self,
|
|
|
|
user_id: &UserId,
|
|
|
|
room_id: &RoomId,
|
|
|
|
) -> Result<impl Iterator<Item = Result<PduEvent>>> {
|
|
|
|
self.pdus_since(user_id, room_id, 0)
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns an iterator over all events in a room that happened after the event with id `since`.
|
|
|
|
pub fn pdus_since(
|
|
|
|
&self,
|
2020-06-16 11:11:38 +01:00
|
|
|
user_id: &UserId,
|
2020-05-03 16:25:31 +01:00
|
|
|
room_id: &RoomId,
|
|
|
|
since: u64,
|
|
|
|
) -> Result<impl Iterator<Item = Result<PduEvent>>> {
|
|
|
|
// Create the first part of the full pdu id
|
|
|
|
let mut pdu_id = room_id.to_string().as_bytes().to_vec();
|
|
|
|
pdu_id.push(0xff);
|
|
|
|
pdu_id.extend_from_slice(&(since).to_be_bytes());
|
|
|
|
|
2020-06-16 11:11:38 +01:00
|
|
|
self.pdus_since_pduid(user_id, room_id, &pdu_id)
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns an iterator over all events in a room that happened after the event with id `since`.
|
|
|
|
pub fn pdus_since_pduid(
|
|
|
|
&self,
|
2020-06-16 11:11:38 +01:00
|
|
|
user_id: &UserId,
|
2020-05-03 16:25:31 +01:00
|
|
|
room_id: &RoomId,
|
|
|
|
pdu_id: &[u8],
|
|
|
|
) -> Result<impl Iterator<Item = Result<PduEvent>>> {
|
|
|
|
// Create the first part of the full pdu id
|
|
|
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
|
|
|
prefix.push(0xff);
|
|
|
|
|
2020-06-16 11:11:38 +01:00
|
|
|
let user_id = user_id.clone();
|
2020-05-03 16:25:31 +01:00
|
|
|
Ok(self
|
|
|
|
.pduid_pdu
|
|
|
|
.range(pdu_id..)
|
|
|
|
// Skip the first pdu if it's exactly at since, because we sent that last time
|
|
|
|
.skip(if self.pduid_pdu.get(pdu_id)?.is_some() {
|
|
|
|
1
|
|
|
|
} else {
|
|
|
|
0
|
|
|
|
})
|
|
|
|
.filter_map(|r| r.ok())
|
|
|
|
.take_while(move |(k, _)| k.starts_with(&prefix))
|
2020-06-16 11:11:38 +01:00
|
|
|
.map(move |(_, v)| {
|
|
|
|
let mut pdu = serde_json::from_slice::<PduEvent>(&v)
|
|
|
|
.map_err(|_| Error::bad_database("PDU in db is invalid."))?;
|
|
|
|
if pdu.sender != user_id {
|
|
|
|
pdu.unsigned.remove("transaction_id");
|
|
|
|
}
|
|
|
|
Ok(pdu)
|
2020-06-09 14:13:17 +01:00
|
|
|
}))
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
|
|
|
|
2020-07-26 16:34:12 +01:00
|
|
|
/// Returns an iterator over all events and their tokens in a room that happened before the
|
|
|
|
/// event with id `until` in reverse-chronological order.
|
2020-05-03 16:25:31 +01:00
|
|
|
pub fn pdus_until(
|
|
|
|
&self,
|
2020-06-16 11:11:38 +01:00
|
|
|
user_id: &UserId,
|
2020-05-03 16:25:31 +01:00
|
|
|
room_id: &RoomId,
|
|
|
|
until: u64,
|
2020-07-26 16:34:12 +01:00
|
|
|
) -> impl Iterator<Item = Result<(IVec, PduEvent)>> {
|
2020-05-03 16:25:31 +01:00
|
|
|
// Create the first part of the full pdu id
|
|
|
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
|
|
|
prefix.push(0xff);
|
|
|
|
|
|
|
|
let mut current = prefix.clone();
|
|
|
|
current.extend_from_slice(&until.to_be_bytes());
|
|
|
|
|
|
|
|
let current: &[u8] = ¤t;
|
|
|
|
|
2020-06-16 11:11:38 +01:00
|
|
|
let user_id = user_id.clone();
|
2020-05-03 16:25:31 +01:00
|
|
|
self.pduid_pdu
|
|
|
|
.range(..current)
|
|
|
|
.rev()
|
|
|
|
.filter_map(|r| r.ok())
|
|
|
|
.take_while(move |(k, _)| k.starts_with(&prefix))
|
2020-07-26 16:34:12 +01:00
|
|
|
.map(move |(k, v)| {
|
2020-06-16 11:11:38 +01:00
|
|
|
let mut pdu = serde_json::from_slice::<PduEvent>(&v)
|
|
|
|
.map_err(|_| Error::bad_database("PDU in db is invalid."))?;
|
|
|
|
if pdu.sender != user_id {
|
|
|
|
pdu.unsigned.remove("transaction_id");
|
|
|
|
}
|
2020-07-26 16:34:12 +01:00
|
|
|
Ok((k, pdu))
|
2020-06-09 14:13:17 +01:00
|
|
|
})
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
|
|
|
|
2020-07-26 16:34:12 +01:00
|
|
|
/// Returns an iterator over all events and their token in a room that happened after the event
|
|
|
|
/// with id `from` in chronological order.
|
2020-06-04 12:58:55 +01:00
|
|
|
pub fn pdus_after(
|
|
|
|
&self,
|
2020-06-16 11:11:38 +01:00
|
|
|
user_id: &UserId,
|
2020-06-04 12:58:55 +01:00
|
|
|
room_id: &RoomId,
|
|
|
|
from: u64,
|
2020-07-26 16:34:12 +01:00
|
|
|
) -> impl Iterator<Item = Result<(IVec, PduEvent)>> {
|
2020-06-04 12:58:55 +01:00
|
|
|
// Create the first part of the full pdu id
|
|
|
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
|
|
|
prefix.push(0xff);
|
|
|
|
|
|
|
|
let mut current = prefix.clone();
|
|
|
|
current.extend_from_slice(&(from + 1).to_be_bytes()); // +1 so we don't send the base event
|
|
|
|
|
|
|
|
let current: &[u8] = ¤t;
|
|
|
|
|
2020-06-16 11:11:38 +01:00
|
|
|
let user_id = user_id.clone();
|
2020-06-04 12:58:55 +01:00
|
|
|
self.pduid_pdu
|
|
|
|
.range(current..)
|
|
|
|
.filter_map(|r| r.ok())
|
|
|
|
.take_while(move |(k, _)| k.starts_with(&prefix))
|
2020-07-26 16:34:12 +01:00
|
|
|
.map(move |(k, v)| {
|
2020-06-16 11:11:38 +01:00
|
|
|
let mut pdu = serde_json::from_slice::<PduEvent>(&v)
|
|
|
|
.map_err(|_| Error::bad_database("PDU in db is invalid."))?;
|
|
|
|
if pdu.sender != user_id {
|
|
|
|
pdu.unsigned.remove("transaction_id");
|
|
|
|
}
|
2020-07-26 16:34:12 +01:00
|
|
|
Ok((k, pdu))
|
2020-06-09 14:13:17 +01:00
|
|
|
})
|
2020-06-04 12:58:55 +01:00
|
|
|
}
|
|
|
|
|
2020-05-26 09:27:51 +01:00
|
|
|
/// Replace a PDU with the redacted form.
|
|
|
|
pub fn redact_pdu(&self, event_id: &EventId) -> Result<()> {
|
|
|
|
if let Some(pdu_id) = self.get_pdu_id(event_id)? {
|
|
|
|
let mut pdu = self
|
|
|
|
.get_pdu_from_id(&pdu_id)?
|
2020-06-11 09:03:08 +01:00
|
|
|
.ok_or_else(|| Error::bad_database("PDU ID points to invalid PDU."))?;
|
2020-06-09 14:13:17 +01:00
|
|
|
pdu.redact()?;
|
2020-05-26 09:27:51 +01:00
|
|
|
self.replace_pdu(&pdu_id, &pdu)?;
|
|
|
|
Ok(())
|
|
|
|
} else {
|
2020-06-09 14:13:17 +01:00
|
|
|
Err(Error::BadRequest(
|
|
|
|
ErrorKind::NotFound,
|
|
|
|
"Event ID does not exist.",
|
|
|
|
))
|
2020-05-26 09:27:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Update current membership data.
|
2020-05-24 17:25:52 +01:00
|
|
|
fn update_membership(
|
2020-05-03 16:25:31 +01:00
|
|
|
&self,
|
|
|
|
room_id: &RoomId,
|
|
|
|
user_id: &UserId,
|
2020-05-24 17:25:52 +01:00
|
|
|
membership: &member::MembershipState,
|
2020-05-03 16:25:31 +01:00
|
|
|
) -> Result<()> {
|
|
|
|
let mut userroom_id = user_id.to_string().as_bytes().to_vec();
|
|
|
|
userroom_id.push(0xff);
|
|
|
|
userroom_id.extend_from_slice(room_id.to_string().as_bytes());
|
|
|
|
|
|
|
|
let mut roomuser_id = room_id.to_string().as_bytes().to_vec();
|
|
|
|
roomuser_id.push(0xff);
|
|
|
|
roomuser_id.extend_from_slice(user_id.to_string().as_bytes());
|
|
|
|
|
2020-05-24 17:25:52 +01:00
|
|
|
match &membership {
|
|
|
|
member::MembershipState::Join => {
|
|
|
|
self.userroomid_joined.insert(&userroom_id, &[])?;
|
|
|
|
self.roomuserid_joined.insert(&roomuser_id, &[])?;
|
|
|
|
self.userroomid_invited.remove(&userroom_id)?;
|
|
|
|
self.roomuserid_invited.remove(&roomuser_id)?;
|
|
|
|
self.userroomid_left.remove(&userroom_id)?;
|
|
|
|
}
|
|
|
|
member::MembershipState::Invite => {
|
|
|
|
self.userroomid_invited.insert(&userroom_id, &[])?;
|
|
|
|
self.roomuserid_invited.insert(&roomuser_id, &[])?;
|
|
|
|
self.userroomid_joined.remove(&userroom_id)?;
|
|
|
|
self.roomuserid_joined.remove(&roomuser_id)?;
|
|
|
|
self.userroomid_left.remove(&userroom_id)?;
|
|
|
|
}
|
|
|
|
member::MembershipState::Leave | member::MembershipState::Ban => {
|
|
|
|
self.userroomid_left.insert(&userroom_id, &[])?;
|
|
|
|
self.userroomid_joined.remove(&userroom_id)?;
|
|
|
|
self.roomuserid_joined.remove(&roomuser_id)?;
|
|
|
|
self.userroomid_invited.remove(&userroom_id)?;
|
|
|
|
self.roomuserid_invited.remove(&roomuser_id)?;
|
|
|
|
}
|
|
|
|
_ => {}
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Makes a user forget a room.
|
|
|
|
pub fn forget(&self, room_id: &RoomId, user_id: &UserId) -> Result<()> {
|
|
|
|
let mut userroom_id = user_id.to_string().as_bytes().to_vec();
|
|
|
|
userroom_id.push(0xff);
|
|
|
|
userroom_id.extend_from_slice(room_id.to_string().as_bytes());
|
|
|
|
|
|
|
|
self.userroomid_left.remove(userroom_id)?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2020-05-25 22:24:13 +01:00
|
|
|
pub fn set_alias(
|
|
|
|
&self,
|
|
|
|
alias: &RoomAliasId,
|
|
|
|
room_id: Option<&RoomId>,
|
|
|
|
globals: &super::globals::Globals,
|
|
|
|
) -> Result<()> {
|
|
|
|
if let Some(room_id) = room_id {
|
2020-06-03 19:55:11 +01:00
|
|
|
// New alias
|
2020-05-25 22:24:13 +01:00
|
|
|
self.alias_roomid
|
|
|
|
.insert(alias.alias(), &*room_id.to_string())?;
|
|
|
|
let mut aliasid = room_id.to_string().as_bytes().to_vec();
|
|
|
|
aliasid.extend_from_slice(&globals.next_count()?.to_be_bytes());
|
|
|
|
self.aliasid_alias.insert(aliasid, &*alias.alias())?;
|
|
|
|
} else {
|
2020-06-03 19:55:11 +01:00
|
|
|
// room_id=None means remove alias
|
|
|
|
let room_id = self
|
|
|
|
.alias_roomid
|
|
|
|
.remove(alias.alias())?
|
2020-06-09 14:13:17 +01:00
|
|
|
.ok_or(Error::BadRequest(
|
|
|
|
ErrorKind::NotFound,
|
|
|
|
"Alias does not exist.",
|
|
|
|
))?;
|
2020-06-03 19:55:11 +01:00
|
|
|
|
|
|
|
for key in self.aliasid_alias.scan_prefix(room_id).keys() {
|
|
|
|
self.aliasid_alias.remove(key?)?;
|
2020-05-25 22:24:13 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn id_from_alias(&self, alias: &RoomAliasId) -> Result<Option<RoomId>> {
|
|
|
|
self.alias_roomid
|
|
|
|
.get(alias.alias())?
|
|
|
|
.map_or(Ok(None), |bytes| {
|
2020-06-11 09:03:08 +01:00
|
|
|
Ok(Some(
|
|
|
|
RoomId::try_from(utils::string_from_bytes(&bytes).map_err(|_| {
|
|
|
|
Error::bad_database("Room ID in alias_roomid is invalid unicode.")
|
|
|
|
})?)
|
|
|
|
.map_err(|_| Error::bad_database("Room ID in alias_roomid is invalid."))?,
|
|
|
|
))
|
2020-05-25 22:24:13 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn room_aliases(&self, room_id: &RoomId) -> impl Iterator<Item = Result<RoomAliasId>> {
|
|
|
|
let mut prefix = room_id.to_string().as_bytes().to_vec();
|
|
|
|
prefix.push(0xff);
|
|
|
|
|
|
|
|
self.aliasid_alias
|
|
|
|
.scan_prefix(prefix)
|
|
|
|
.values()
|
2020-06-09 14:13:17 +01:00
|
|
|
.map(|bytes| {
|
|
|
|
Ok(serde_json::from_slice(&bytes?)
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("Alias in aliasid_alias is invalid."))?)
|
2020-06-09 14:13:17 +01:00
|
|
|
})
|
2020-05-25 22:24:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn set_public(&self, room_id: &RoomId, public: bool) -> Result<()> {
|
|
|
|
if public {
|
|
|
|
self.publicroomids.insert(room_id.to_string(), &[])?;
|
|
|
|
} else {
|
|
|
|
self.publicroomids.remove(room_id.to_string())?;
|
2020-05-24 07:30:57 +01:00
|
|
|
}
|
|
|
|
|
2020-05-25 22:24:13 +01:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_public_room(&self, room_id: &RoomId) -> Result<bool> {
|
|
|
|
Ok(self.publicroomids.contains_key(room_id.to_string())?)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn public_rooms(&self) -> impl Iterator<Item = Result<RoomId>> {
|
2020-06-09 14:13:17 +01:00
|
|
|
self.publicroomids.iter().keys().map(|bytes| {
|
2020-06-11 09:03:08 +01:00
|
|
|
Ok(
|
|
|
|
RoomId::try_from(utils::string_from_bytes(&bytes?).map_err(|_| {
|
|
|
|
Error::bad_database("Room ID in publicroomids is invalid unicode.")
|
|
|
|
})?)
|
|
|
|
.map_err(|_| Error::bad_database("Room ID in publicroomids is invalid."))?,
|
|
|
|
)
|
2020-06-09 14:13:17 +01:00
|
|
|
})
|
2020-05-24 07:30:57 +01:00
|
|
|
}
|
|
|
|
|
2020-06-11 09:03:08 +01:00
|
|
|
/// Returns an iterator over all joined members of a room.
|
2020-05-03 16:25:31 +01:00
|
|
|
pub fn room_members(&self, room_id: &RoomId) -> impl Iterator<Item = Result<UserId>> {
|
|
|
|
self.roomuserid_joined
|
|
|
|
.scan_prefix(room_id.to_string())
|
2020-06-11 09:03:08 +01:00
|
|
|
.keys()
|
2020-05-03 16:25:31 +01:00
|
|
|
.map(|key| {
|
2020-06-11 09:03:08 +01:00
|
|
|
Ok(UserId::try_from(
|
|
|
|
utils::string_from_bytes(
|
|
|
|
&key?
|
|
|
|
.rsplit(|&b| b == 0xff)
|
|
|
|
.next()
|
|
|
|
.expect("rsplit always returns an element"),
|
|
|
|
)
|
|
|
|
.map_err(|_| {
|
|
|
|
Error::bad_database("User ID in roomuserid_joined is invalid unicode.")
|
|
|
|
})?,
|
2020-06-09 14:13:17 +01:00
|
|
|
)
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("User ID in roomuserid_joined is invalid."))?)
|
2020-05-03 16:25:31 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-06-11 09:03:08 +01:00
|
|
|
/// Returns an iterator over all invited members of a room.
|
2020-05-03 16:25:31 +01:00
|
|
|
pub fn room_members_invited(&self, room_id: &RoomId) -> impl Iterator<Item = Result<UserId>> {
|
|
|
|
self.roomuserid_invited
|
|
|
|
.scan_prefix(room_id.to_string())
|
|
|
|
.keys()
|
|
|
|
.map(|key| {
|
2020-06-11 09:03:08 +01:00
|
|
|
Ok(UserId::try_from(
|
|
|
|
utils::string_from_bytes(
|
|
|
|
&key?
|
|
|
|
.rsplit(|&b| b == 0xff)
|
|
|
|
.next()
|
|
|
|
.expect("rsplit always returns an element"),
|
|
|
|
)
|
|
|
|
.map_err(|_| {
|
|
|
|
Error::bad_database("User ID in roomuserid_invited is invalid unicode.")
|
|
|
|
})?,
|
2020-06-09 14:13:17 +01:00
|
|
|
)
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("User ID in roomuserid_invited is invalid."))?)
|
2020-05-03 16:25:31 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-06-11 09:03:08 +01:00
|
|
|
/// Returns an iterator over all left members of a room.
|
2020-05-03 16:25:31 +01:00
|
|
|
pub fn rooms_joined(&self, user_id: &UserId) -> impl Iterator<Item = Result<RoomId>> {
|
|
|
|
self.userroomid_joined
|
|
|
|
.scan_prefix(user_id.to_string())
|
|
|
|
.keys()
|
|
|
|
.map(|key| {
|
2020-06-11 09:03:08 +01:00
|
|
|
Ok(RoomId::try_from(
|
|
|
|
utils::string_from_bytes(
|
|
|
|
&key?
|
|
|
|
.rsplit(|&b| b == 0xff)
|
|
|
|
.next()
|
|
|
|
.expect("rsplit always returns an element"),
|
|
|
|
)
|
|
|
|
.map_err(|_| {
|
|
|
|
Error::bad_database("Room ID in userroomid_joined is invalid unicode.")
|
|
|
|
})?,
|
2020-06-09 14:13:17 +01:00
|
|
|
)
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("Room ID in userroomid_joined is invalid."))?)
|
2020-05-03 16:25:31 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns an iterator over all rooms a user was invited to.
|
|
|
|
pub fn rooms_invited(&self, user_id: &UserId) -> impl Iterator<Item = Result<RoomId>> {
|
|
|
|
self.userroomid_invited
|
|
|
|
.scan_prefix(&user_id.to_string())
|
|
|
|
.keys()
|
|
|
|
.map(|key| {
|
2020-06-11 09:03:08 +01:00
|
|
|
Ok(RoomId::try_from(
|
|
|
|
utils::string_from_bytes(
|
|
|
|
&key?
|
|
|
|
.rsplit(|&b| b == 0xff)
|
|
|
|
.next()
|
|
|
|
.expect("rsplit always returns an element"),
|
|
|
|
)
|
|
|
|
.map_err(|_| {
|
|
|
|
Error::bad_database("Room ID in userroomid_invited is invalid unicode.")
|
|
|
|
})?,
|
2020-06-09 14:13:17 +01:00
|
|
|
)
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("Room ID in userroomid_invited is invalid."))?)
|
2020-05-03 16:25:31 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns an iterator over all rooms a user left.
|
|
|
|
pub fn rooms_left(&self, user_id: &UserId) -> impl Iterator<Item = Result<RoomId>> {
|
|
|
|
self.userroomid_left
|
|
|
|
.scan_prefix(&user_id.to_string())
|
|
|
|
.keys()
|
|
|
|
.map(|key| {
|
2020-06-11 09:03:08 +01:00
|
|
|
Ok(RoomId::try_from(
|
|
|
|
utils::string_from_bytes(
|
|
|
|
&key?
|
|
|
|
.rsplit(|&b| b == 0xff)
|
|
|
|
.next()
|
|
|
|
.expect("rsplit always returns an element"),
|
|
|
|
)
|
|
|
|
.map_err(|_| {
|
|
|
|
Error::bad_database("Room ID in userroomid_left is invalid unicode.")
|
|
|
|
})?,
|
2020-06-09 14:13:17 +01:00
|
|
|
)
|
2020-06-11 09:03:08 +01:00
|
|
|
.map_err(|_| Error::bad_database("Room ID in userroomid_left is invalid."))?)
|
2020-05-03 16:25:31 +01:00
|
|
|
})
|
|
|
|
}
|
2020-05-24 17:25:52 +01:00
|
|
|
|
|
|
|
pub fn is_joined(&self, user_id: &UserId, room_id: &RoomId) -> Result<bool> {
|
|
|
|
let mut userroom_id = user_id.to_string().as_bytes().to_vec();
|
|
|
|
userroom_id.push(0xff);
|
|
|
|
userroom_id.extend_from_slice(room_id.to_string().as_bytes());
|
|
|
|
|
|
|
|
Ok(self.userroomid_joined.get(userroom_id)?.is_some())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_invited(&self, user_id: &UserId, room_id: &RoomId) -> Result<bool> {
|
|
|
|
let mut userroom_id = user_id.to_string().as_bytes().to_vec();
|
|
|
|
userroom_id.push(0xff);
|
|
|
|
userroom_id.extend_from_slice(room_id.to_string().as_bytes());
|
|
|
|
|
|
|
|
Ok(self.userroomid_invited.get(userroom_id)?.is_some())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_left(&self, user_id: &UserId, room_id: &RoomId) -> Result<bool> {
|
|
|
|
let mut userroom_id = user_id.to_string().as_bytes().to_vec();
|
|
|
|
userroom_id.push(0xff);
|
|
|
|
userroom_id.extend_from_slice(room_id.to_string().as_bytes());
|
|
|
|
|
|
|
|
Ok(self.userroomid_left.get(userroom_id)?.is_some())
|
|
|
|
}
|
2020-05-03 16:25:31 +01:00
|
|
|
}
|