conduwuit/src/database/key_value/rooms/short.rs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

209 lines
5.9 KiB
Rust
Raw Normal View History

2022-10-05 15:33:57 +02:00
use std::sync::Arc;
2022-10-05 20:34:31 +02:00
use ruma::{events::StateEventType, EventId, RoomId};
use tracing::warn;
2022-09-07 13:25:51 +02:00
2022-10-05 20:34:31 +02:00
use crate::{database::KeyValueDatabase, service, services, utils, Error, Result};
2022-10-05 18:36:12 +02:00
impl service::rooms::short::Data for KeyValueDatabase {
2022-10-05 20:34:31 +02:00
fn get_or_create_shorteventid(&self, event_id: &EventId) -> Result<u64> {
2022-10-05 18:36:12 +02:00
if let Some(short) = self.eventidshort_cache.lock().unwrap().get_mut(event_id) {
return Ok(*short);
}
2022-10-05 18:36:12 +02:00
let short = match self.eventid_shorteventid.get(event_id.as_bytes())? {
Some(shorteventid) => {
utils::u64_from_bytes(&shorteventid).map_err(|_| Error::bad_database("Invalid shorteventid in db."))?
},
None => {
let shorteventid = services().globals.next_count()?;
self.eventid_shorteventid
.insert(event_id.as_bytes(), &shorteventid.to_be_bytes())?;
self.shorteventid_eventid
.insert(&shorteventid.to_be_bytes(), event_id.as_bytes())?;
2022-10-05 18:36:12 +02:00
shorteventid
},
};
self.eventidshort_cache
.lock()
.unwrap()
.insert(event_id.to_owned(), short);
2022-10-05 18:36:12 +02:00
Ok(short)
}
2022-10-05 18:36:12 +02:00
fn get_shortstatekey(&self, event_type: &StateEventType, state_key: &str) -> Result<Option<u64>> {
if let Some(short) = self
.statekeyshort_cache
.lock()
.unwrap()
.get_mut(&(event_type.clone(), state_key.to_owned()))
2022-10-05 18:36:12 +02:00
{
return Ok(Some(*short));
}
let mut statekey_vec = event_type.to_string().as_bytes().to_vec();
statekey_vec.push(0xFF);
statekey_vec.extend_from_slice(state_key.as_bytes());
2022-10-05 18:36:12 +02:00
let short = self
.statekey_shortstatekey
.get(&statekey_vec)?
2022-10-05 18:36:12 +02:00
.map(|shortstatekey| {
utils::u64_from_bytes(&shortstatekey).map_err(|_| Error::bad_database("Invalid shortstatekey in db."))
})
.transpose()?;
2022-10-05 18:36:12 +02:00
if let Some(s) = short {
self.statekeyshort_cache
.lock()
.unwrap()
.insert((event_type.clone(), state_key.to_owned()), s);
2022-10-05 18:36:12 +02:00
}
2022-10-05 18:36:12 +02:00
Ok(short)
}
2022-10-05 18:36:12 +02:00
fn get_or_create_shortstatekey(&self, event_type: &StateEventType, state_key: &str) -> Result<u64> {
if let Some(short) = self
.statekeyshort_cache
.lock()
.unwrap()
.get_mut(&(event_type.clone(), state_key.to_owned()))
2022-10-05 18:36:12 +02:00
{
return Ok(*short);
}
let mut statekey_vec = event_type.to_string().as_bytes().to_vec();
statekey_vec.push(0xFF);
statekey_vec.extend_from_slice(state_key.as_bytes());
let short = match self.statekey_shortstatekey.get(&statekey_vec)? {
2022-10-05 18:36:12 +02:00
Some(shortstatekey) => utils::u64_from_bytes(&shortstatekey)
.map_err(|_| Error::bad_database("Invalid shortstatekey in db."))?,
None => {
let shortstatekey = services().globals.next_count()?;
self.statekey_shortstatekey
.insert(&statekey_vec, &shortstatekey.to_be_bytes())?;
self.shortstatekey_statekey
.insert(&shortstatekey.to_be_bytes(), &statekey_vec)?;
2022-10-05 18:36:12 +02:00
shortstatekey
},
};
self.statekeyshort_cache
.lock()
.unwrap()
.insert((event_type.clone(), state_key.to_owned()), short);
2022-10-05 18:36:12 +02:00
Ok(short)
}
2022-10-05 18:36:12 +02:00
fn get_eventid_from_short(&self, shorteventid: u64) -> Result<Arc<EventId>> {
if let Some(id) = self
.shorteventid_cache
.lock()
.unwrap()
.get_mut(&shorteventid)
{
2022-10-05 18:36:12 +02:00
return Ok(Arc::clone(id));
}
2022-10-05 18:36:12 +02:00
let bytes = self
.shorteventid_eventid
.get(&shorteventid.to_be_bytes())?
.ok_or_else(|| Error::bad_database("Shorteventid does not exist"))?;
2022-10-05 18:36:12 +02:00
let event_id = EventId::parse_arc(
utils::string_from_bytes(&bytes)
.map_err(|_| Error::bad_database("EventID in shorteventid_eventid is invalid unicode."))?,
)
.map_err(|_| Error::bad_database("EventId in shorteventid_eventid is invalid."))?;
self.shorteventid_cache
.lock()
.unwrap()
.insert(shorteventid, Arc::clone(&event_id));
2022-10-05 18:36:12 +02:00
Ok(event_id)
}
2022-10-05 18:36:12 +02:00
fn get_statekey_from_short(&self, shortstatekey: u64) -> Result<(StateEventType, String)> {
if let Some(id) = self
.shortstatekey_cache
.lock()
.unwrap()
.get_mut(&shortstatekey)
{
2022-10-05 18:36:12 +02:00
return Ok(id.clone());
}
2022-10-05 18:36:12 +02:00
let bytes = self
.shortstatekey_statekey
.get(&shortstatekey.to_be_bytes())?
.ok_or_else(|| Error::bad_database("Shortstatekey does not exist"))?;
2022-10-05 18:36:12 +02:00
let mut parts = bytes.splitn(2, |&b| b == 0xFF);
let eventtype_bytes = parts.next().expect("split always returns one entry");
let statekey_bytes = parts
.next()
.ok_or_else(|| Error::bad_database("Invalid statekey in shortstatekey_statekey."))?;
let event_type = StateEventType::from(utils::string_from_bytes(eventtype_bytes).map_err(|e| {
warn!("Event type in shortstatekey_statekey is invalid: {}", e);
Error::bad_database("Event type in shortstatekey_statekey is invalid.")
})?);
2022-10-05 18:36:12 +02:00
let state_key = utils::string_from_bytes(statekey_bytes)
.map_err(|_| Error::bad_database("Statekey in shortstatekey_statekey is invalid unicode."))?;
2022-10-05 18:36:12 +02:00
let result = (event_type, state_key);
self.shortstatekey_cache
.lock()
.unwrap()
.insert(shortstatekey, result.clone());
2022-10-05 18:36:12 +02:00
Ok(result)
}
2022-10-05 18:36:12 +02:00
/// Returns (shortstatehash, already_existed)
2022-10-05 20:34:31 +02:00
fn get_or_create_shortstatehash(&self, state_hash: &[u8]) -> Result<(u64, bool)> {
2022-10-05 18:36:12 +02:00
Ok(match self.statehash_shortstatehash.get(state_hash)? {
Some(shortstatehash) => (
utils::u64_from_bytes(&shortstatehash)
.map_err(|_| Error::bad_database("Invalid shortstatehash in db."))?,
true,
),
None => {
let shortstatehash = services().globals.next_count()?;
self.statehash_shortstatehash
.insert(state_hash, &shortstatehash.to_be_bytes())?;
2022-10-05 18:36:12 +02:00
(shortstatehash, false)
},
})
}
2022-10-05 18:36:12 +02:00
fn get_shortroomid(&self, room_id: &RoomId) -> Result<Option<u64>> {
self.roomid_shortroomid
.get(room_id.as_bytes())?
.map(|bytes| utils::u64_from_bytes(&bytes).map_err(|_| Error::bad_database("Invalid shortroomid in db.")))
.transpose()
}
2022-10-05 20:34:31 +02:00
fn get_or_create_shortroomid(&self, room_id: &RoomId) -> Result<u64> {
2022-10-05 18:36:12 +02:00
Ok(match self.roomid_shortroomid.get(room_id.as_bytes())? {
Some(short) => {
utils::u64_from_bytes(&short).map_err(|_| Error::bad_database("Invalid shortroomid in db."))?
},
None => {
let short = services().globals.next_count()?;
self.roomid_shortroomid
.insert(room_id.as_bytes(), &short.to_be_bytes())?;
2022-10-05 18:36:12 +02:00
short
},
})
}
2022-09-07 13:25:51 +02:00
}