use std::{
borrow::Cow,
collections::{BTreeMap, BTreeSet, HashMap},
fmt, iter,
path::Path,
sync::Arc,
};
use async_trait::async_trait;
use deadpool_sqlite::{Object as SqliteAsyncConn, Pool as SqlitePool, Runtime};
use matrix_sdk_base::{
deserialized_responses::{DisplayName, RawAnySyncOrStrippedState, SyncOrStrippedState},
store::{
migration_helpers::RoomInfoV1, ChildTransactionId, DependentQueuedRequest,
DependentQueuedRequestKind, QueueWedgeError, QueuedRequest, QueuedRequestKind,
SentRequestKey,
},
MinimalRoomMemberEvent, RoomInfo, RoomMemberships, RoomState, StateChanges, StateStore,
StateStoreDataKey, StateStoreDataValue,
};
use matrix_sdk_store_encryption::StoreCipher;
use ruma::{
canonical_json::{redact, RedactedBecause},
events::{
presence::PresenceEvent,
receipt::{Receipt, ReceiptThread, ReceiptType},
room::{
create::RoomCreateEventContent,
member::{StrippedRoomMemberEvent, SyncRoomMemberEvent},
},
AnyGlobalAccountDataEvent, AnyRoomAccountDataEvent, AnySyncStateEvent,
GlobalAccountDataEventType, RoomAccountDataEventType, StateEventType,
},
serde::Raw,
CanonicalJsonObject, EventId, MilliSecondsSinceUnixEpoch, OwnedEventId, OwnedRoomId,
OwnedTransactionId, OwnedUserId, RoomId, RoomVersionId, TransactionId, UInt, UserId,
};
use rusqlite::{OptionalExtension, Transaction};
use serde::{de::DeserializeOwned, Deserialize, Serialize};
use tokio::fs;
use tracing::{debug, warn};
use crate::{
error::{Error, Result},
utils::{
repeat_vars, Key, SqliteAsyncConnExt, SqliteKeyValueStoreAsyncConnExt,
SqliteKeyValueStoreConnExt,
},
OpenStoreError,
};
mod keys {
pub const KV_BLOB: &str = "kv_blob";
pub const ROOM_INFO: &str = "room_info";
pub const STATE_EVENT: &str = "state_event";
pub const GLOBAL_ACCOUNT_DATA: &str = "global_account_data";
pub const ROOM_ACCOUNT_DATA: &str = "room_account_data";
pub const MEMBER: &str = "member";
pub const PROFILE: &str = "profile";
pub const RECEIPT: &str = "receipt";
pub const DISPLAY_NAME: &str = "display_name";
pub const SEND_QUEUE: &str = "send_queue_events";
pub const DEPENDENTS_SEND_QUEUE: &str = "dependent_send_queue_events";
}
const DATABASE_VERSION: u8 = 11;
#[derive(Clone)]
pub struct SqliteStateStore {
store_cipher: Option<Arc<StoreCipher>>,
pool: SqlitePool,
}
#[cfg(not(tarpaulin_include))]
impl fmt::Debug for SqliteStateStore {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("SqliteStateStore").finish_non_exhaustive()
}
}
impl SqliteStateStore {
pub async fn open(
path: impl AsRef<Path>,
passphrase: Option<&str>,
) -> Result<Self, OpenStoreError> {
let pool = create_pool(path.as_ref()).await?;
Self::open_with_pool(pool, passphrase).await
}
pub async fn open_with_pool(
pool: SqlitePool,
passphrase: Option<&str>,
) -> Result<Self, OpenStoreError> {
let conn = pool.get().await?;
let mut version = conn.db_version().await?;
if version == 0 {
init(&conn).await?;
version = 1;
}
let store_cipher = match passphrase {
Some(p) => Some(Arc::new(conn.get_or_create_store_cipher(p).await?)),
None => None,
};
let this = Self { store_cipher, pool };
this.run_migrations(&conn, version, None).await?;
Ok(this)
}
async fn run_migrations(&self, conn: &SqliteAsyncConn, from: u8, to: Option<u8>) -> Result<()> {
let to = to.unwrap_or(DATABASE_VERSION);
if from < to {
debug!(version = from, new_version = to, "Upgrading database");
} else {
return Ok(());
}
if from < 2 && to >= 2 {
let this = self.clone();
conn.with_transaction(move |txn| {
txn.execute_batch(include_str!(
"../migrations/state_store/002_a_create_new_room_info.sql"
))?;
for data in txn
.prepare("SELECT data FROM room_info")?
.query_map((), |row| row.get::<_, Vec<u8>>(0))?
{
let data = data?;
let room_info: RoomInfoV1 = this.deserialize_json(&data)?;
let room_id = this.encode_key(keys::ROOM_INFO, room_info.room_id());
let state = this
.encode_key(keys::ROOM_INFO, serde_json::to_string(&room_info.state())?);
txn.prepare_cached(
"INSERT OR REPLACE INTO new_room_info (room_id, state, data)
VALUES (?, ?, ?)",
)?
.execute((room_id, state, data))?;
}
txn.execute_batch(include_str!(
"../migrations/state_store/002_b_replace_room_info.sql"
))?;
txn.set_db_version(2)?;
Result::<_, Error>::Ok(())
})
.await?;
}
if from < 3 && to >= 3 {
let this = self.clone();
conn.with_transaction(move |txn| {
for data in txn
.prepare("SELECT data FROM room_info")?
.query_map((), |row| row.get::<_, Vec<u8>>(0))?
{
let data = data?;
let room_info_v1: RoomInfoV1 = this.deserialize_json(&data)?;
let room_id = this.encode_key(keys::STATE_EVENT, room_info_v1.room_id());
let event_type =
this.encode_key(keys::STATE_EVENT, StateEventType::RoomCreate.to_string());
let create_res = txn
.prepare(
"SELECT stripped, data FROM state_event
WHERE room_id = ? AND event_type = ?",
)?
.query_row([room_id, event_type], |row| {
Ok((row.get::<_, bool>(0)?, row.get::<_, Vec<u8>>(1)?))
})
.optional()?;
let create = create_res.and_then(|(stripped, data)| {
let create = if stripped {
SyncOrStrippedState::<RoomCreateEventContent>::Stripped(
this.deserialize_json(&data).ok()?,
)
} else {
SyncOrStrippedState::Sync(this.deserialize_json(&data).ok()?)
};
Some(create)
});
let migrated_room_info = room_info_v1.migrate(create.as_ref());
let data = this.serialize_json(&migrated_room_info)?;
let room_id = this.encode_key(keys::ROOM_INFO, migrated_room_info.room_id());
txn.prepare_cached("UPDATE room_info SET data = ? WHERE room_id = ?")?
.execute((data, room_id))?;
}
txn.set_db_version(3)?;
Result::<_, Error>::Ok(())
})
.await?;
}
if from < 4 && to >= 4 {
conn.with_transaction(move |txn| {
txn.execute_batch(include_str!("../migrations/state_store/003_send_queue.sql"))?;
txn.set_db_version(4)
})
.await?;
}
if from < 5 && to >= 5 {
conn.with_transaction(move |txn| {
txn.execute_batch(include_str!(
"../migrations/state_store/004_send_queue_with_roomid_value.sql"
))?;
txn.set_db_version(4)
})
.await?;
}
if from < 6 && to >= 6 {
conn.with_transaction(move |txn| {
txn.execute_batch(include_str!(
"../migrations/state_store/005_send_queue_dependent_events.sql"
))?;
txn.set_db_version(6)
})
.await?;
}
if from < 7 && to >= 7 {
conn.with_transaction(move |txn| {
txn.execute_batch(include_str!("../migrations/state_store/006_drop_media.sql"))?;
txn.set_db_version(7)
})
.await?;
}
if from < 8 && to >= 8 {
let error = QueueWedgeError::GenericApiError {
msg: "local echo failed to send in a previous session".into(),
};
let default_err = self.serialize_value(&error)?;
conn.with_transaction(move |txn| {
txn.execute_batch(include_str!("../migrations/state_store/007_a_send_queue_wedge_reason.sql"))?;
for wedged_entries in txn
.prepare("SELECT room_id, transaction_id FROM send_queue_events WHERE wedged = 1")?
.query_map((), |row| {
Ok(
(row.get::<_, Vec<u8>>(0)?,row.get::<_, String>(1)?)
)
})? {
let (room_id, transaction_id) = wedged_entries?;
txn.prepare_cached("UPDATE send_queue_events SET wedge_reason = ? WHERE room_id = ? AND transaction_id = ?")?
.execute((default_err.clone(), room_id, transaction_id))?;
}
txn.execute_batch(include_str!("../migrations/state_store/007_b_send_queue_clean.sql"))?;
txn.set_db_version(8)
})
.await?;
}
if from < 9 && to >= 9 {
conn.with_transaction(move |txn| {
txn.execute_batch(include_str!("../migrations/state_store/008_send_queue.sql"))?;
txn.set_db_version(9)
})
.await?;
}
if from < 10 && to >= 10 {
conn.with_transaction(move |txn| {
txn.execute_batch(include_str!(
"../migrations/state_store/009_send_queue_priority.sql"
))?;
txn.set_db_version(10)
})
.await?;
}
if from < 11 && to >= 11 {
conn.with_transaction(move |txn| {
txn.execute_batch(include_str!(
"../migrations/state_store/010_send_queue_enqueue_time.sql"
))?;
txn.set_db_version(11)
})
.await?;
}
Ok(())
}
fn encode_value(&self, value: Vec<u8>) -> Result<Vec<u8>> {
if let Some(key) = &self.store_cipher {
let encrypted = key.encrypt_value_data(value)?;
Ok(rmp_serde::to_vec_named(&encrypted)?)
} else {
Ok(value)
}
}
fn serialize_value(&self, value: &impl Serialize) -> Result<Vec<u8>> {
let serialized = rmp_serde::to_vec_named(value)?;
self.encode_value(serialized)
}
fn serialize_json(&self, value: &impl Serialize) -> Result<Vec<u8>> {
let serialized = serde_json::to_vec(value)?;
self.encode_value(serialized)
}
fn decode_value<'a>(&self, value: &'a [u8]) -> Result<Cow<'a, [u8]>> {
if let Some(key) = &self.store_cipher {
let encrypted = rmp_serde::from_slice(value)?;
let decrypted = key.decrypt_value_data(encrypted)?;
Ok(Cow::Owned(decrypted))
} else {
Ok(Cow::Borrowed(value))
}
}
fn deserialize_json<T: DeserializeOwned>(&self, data: &[u8]) -> Result<T> {
let decoded = self.decode_value(data)?;
Ok(serde_json::from_slice(&decoded)?)
}
fn deserialize_value<T: DeserializeOwned>(&self, value: &[u8]) -> Result<T> {
let decoded = self.decode_value(value)?;
Ok(rmp_serde::from_slice(&decoded)?)
}
fn encode_key(&self, table_name: &str, key: impl AsRef<[u8]>) -> Key {
let bytes = key.as_ref();
if let Some(store_cipher) = &self.store_cipher {
Key::Hashed(store_cipher.hash_key(table_name, bytes))
} else {
Key::Plain(bytes.to_owned())
}
}
fn encode_state_store_data_key(&self, key: StateStoreDataKey<'_>) -> Key {
let key_s = match key {
StateStoreDataKey::SyncToken => Cow::Borrowed(StateStoreDataKey::SYNC_TOKEN),
StateStoreDataKey::ServerCapabilities => {
Cow::Borrowed(StateStoreDataKey::SERVER_CAPABILITIES)
}
StateStoreDataKey::Filter(f) => {
Cow::Owned(format!("{}:{f}", StateStoreDataKey::FILTER))
}
StateStoreDataKey::UserAvatarUrl(u) => {
Cow::Owned(format!("{}:{u}", StateStoreDataKey::USER_AVATAR_URL))
}
StateStoreDataKey::RecentlyVisitedRooms(b) => {
Cow::Owned(format!("{}:{b}", StateStoreDataKey::RECENTLY_VISITED_ROOMS))
}
StateStoreDataKey::UtdHookManagerData => {
Cow::Borrowed(StateStoreDataKey::UTD_HOOK_MANAGER_DATA)
}
StateStoreDataKey::ComposerDraft(room_id) => {
Cow::Owned(format!("{}:{room_id}", StateStoreDataKey::COMPOSER_DRAFT))
}
StateStoreDataKey::SeenKnockRequests(room_id) => {
Cow::Owned(format!("{}:{room_id}", StateStoreDataKey::SEEN_KNOCK_REQUESTS))
}
};
self.encode_key(keys::KV_BLOB, &*key_s)
}
fn encode_presence_key(&self, user_id: &UserId) -> Key {
self.encode_key(keys::KV_BLOB, format!("presence:{user_id}"))
}
fn encode_custom_key(&self, key: &[u8]) -> Key {
let mut full_key = b"custom:".to_vec();
full_key.extend(key);
self.encode_key(keys::KV_BLOB, full_key)
}
async fn acquire(&self) -> Result<SqliteAsyncConn> {
Ok(self.pool.get().await?)
}
fn remove_maybe_stripped_room_data(
&self,
txn: &Transaction<'_>,
room_id: &RoomId,
stripped: bool,
) -> rusqlite::Result<()> {
let state_event_room_id = self.encode_key(keys::STATE_EVENT, room_id);
txn.remove_room_state_events(&state_event_room_id, Some(stripped))?;
let member_room_id = self.encode_key(keys::MEMBER, room_id);
txn.remove_room_members(&member_room_id, Some(stripped))
}
}
async fn create_pool(path: &Path) -> Result<SqlitePool, OpenStoreError> {
fs::create_dir_all(path).await.map_err(OpenStoreError::CreateDir)?;
let cfg = deadpool_sqlite::Config::new(path.join("matrix-sdk-state.sqlite3"));
Ok(cfg.create_pool(Runtime::Tokio1)?)
}
async fn init(conn: &SqliteAsyncConn) -> Result<()> {
conn.execute_batch("PRAGMA journal_mode = wal;").await?;
conn.with_transaction(|txn| {
txn.execute_batch(include_str!("../migrations/state_store/001_init.sql"))?;
txn.set_db_version(1)?;
Ok(())
})
.await
}
trait SqliteConnectionStateStoreExt {
fn set_kv_blob(&self, key: &[u8], value: &[u8]) -> rusqlite::Result<()>;
fn set_global_account_data(&self, event_type: &[u8], data: &[u8]) -> rusqlite::Result<()>;
fn set_room_account_data(
&self,
room_id: &[u8],
event_type: &[u8],
data: &[u8],
) -> rusqlite::Result<()>;
fn remove_room_account_data(&self, room_id: &[u8]) -> rusqlite::Result<()>;
fn set_room_info(&self, room_id: &[u8], state: &[u8], data: &[u8]) -> rusqlite::Result<()>;
fn get_room_info(&self, room_id: &[u8]) -> rusqlite::Result<Option<Vec<u8>>>;
fn remove_room_info(&self, room_id: &[u8]) -> rusqlite::Result<()>;
fn set_state_event(
&self,
room_id: &[u8],
event_type: &[u8],
state_key: &[u8],
stripped: bool,
event_id: Option<&[u8]>,
data: &[u8],
) -> rusqlite::Result<()>;
fn get_state_event_by_id(
&self,
room_id: &[u8],
event_id: &[u8],
) -> rusqlite::Result<Option<Vec<u8>>>;
fn remove_room_state_events(
&self,
room_id: &[u8],
stripped: Option<bool>,
) -> rusqlite::Result<()>;
fn set_member(
&self,
room_id: &[u8],
user_id: &[u8],
membership: &[u8],
stripped: bool,
data: &[u8],
) -> rusqlite::Result<()>;
fn remove_room_members(&self, room_id: &[u8], stripped: Option<bool>) -> rusqlite::Result<()>;
fn set_profile(&self, room_id: &[u8], user_id: &[u8], data: &[u8]) -> rusqlite::Result<()>;
fn remove_room_profiles(&self, room_id: &[u8]) -> rusqlite::Result<()>;
fn remove_room_profile(&self, room_id: &[u8], user_id: &[u8]) -> rusqlite::Result<()>;
fn set_receipt(
&self,
room_id: &[u8],
user_id: &[u8],
receipt_type: &[u8],
thread_id: &[u8],
event_id: &[u8],
data: &[u8],
) -> rusqlite::Result<()>;
fn remove_room_receipts(&self, room_id: &[u8]) -> rusqlite::Result<()>;
fn set_display_name(&self, room_id: &[u8], name: &[u8], data: &[u8]) -> rusqlite::Result<()>;
fn remove_display_name(&self, room_id: &[u8], name: &[u8]) -> rusqlite::Result<()>;
fn remove_room_display_names(&self, room_id: &[u8]) -> rusqlite::Result<()>;
fn remove_room_send_queue(&self, room_id: &[u8]) -> rusqlite::Result<()>;
fn remove_room_dependent_send_queue(&self, room_id: &[u8]) -> rusqlite::Result<()>;
}
impl SqliteConnectionStateStoreExt for rusqlite::Connection {
fn set_kv_blob(&self, key: &[u8], value: &[u8]) -> rusqlite::Result<()> {
self.execute("INSERT OR REPLACE INTO kv_blob VALUES (?, ?)", (key, value))?;
Ok(())
}
fn set_global_account_data(&self, event_type: &[u8], data: &[u8]) -> rusqlite::Result<()> {
self.prepare_cached(
"INSERT OR REPLACE INTO global_account_data (event_type, data)
VALUES (?, ?)",
)?
.execute((event_type, data))?;
Ok(())
}
fn set_room_account_data(
&self,
room_id: &[u8],
event_type: &[u8],
data: &[u8],
) -> rusqlite::Result<()> {
self.prepare_cached(
"INSERT OR REPLACE INTO room_account_data (room_id, event_type, data)
VALUES (?, ?, ?)",
)?
.execute((room_id, event_type, data))?;
Ok(())
}
fn remove_room_account_data(&self, room_id: &[u8]) -> rusqlite::Result<()> {
self.prepare(
"DELETE FROM room_account_data
WHERE room_id = ?",
)?
.execute((room_id,))?;
Ok(())
}
fn set_room_info(&self, room_id: &[u8], state: &[u8], data: &[u8]) -> rusqlite::Result<()> {
self.prepare_cached(
"INSERT OR REPLACE INTO room_info (room_id, state, data)
VALUES (?, ?, ?)",
)?
.execute((room_id, state, data))?;
Ok(())
}
fn get_room_info(&self, room_id: &[u8]) -> rusqlite::Result<Option<Vec<u8>>> {
self.query_row("SELECT data FROM room_info WHERE room_id = ?", (room_id,), |row| row.get(0))
.optional()
}
fn remove_room_info(&self, room_id: &[u8]) -> rusqlite::Result<()> {
self.prepare_cached("DELETE FROM room_info WHERE room_id = ?")?.execute((room_id,))?;
Ok(())
}
fn set_state_event(
&self,
room_id: &[u8],
event_type: &[u8],
state_key: &[u8],
stripped: bool,
event_id: Option<&[u8]>,
data: &[u8],
) -> rusqlite::Result<()> {
self.prepare_cached(
"INSERT OR REPLACE
INTO state_event (room_id, event_type, state_key, stripped, event_id, data)
VALUES (?, ?, ?, ?, ?, ?)",
)?
.execute((room_id, event_type, state_key, stripped, event_id, data))?;
Ok(())
}
fn get_state_event_by_id(
&self,
room_id: &[u8],
event_id: &[u8],
) -> rusqlite::Result<Option<Vec<u8>>> {
self.query_row(
"SELECT data FROM state_event WHERE room_id = ? AND event_id = ?",
(room_id, event_id),
|row| row.get(0),
)
.optional()
}
fn remove_room_state_events(
&self,
room_id: &[u8],
stripped: Option<bool>,
) -> rusqlite::Result<()> {
if let Some(stripped) = stripped {
self.prepare_cached("DELETE FROM state_event WHERE room_id = ? AND stripped = ?")?
.execute((room_id, stripped))?;
} else {
self.prepare_cached("DELETE FROM state_event WHERE room_id = ?")?
.execute((room_id,))?;
}
Ok(())
}
fn set_member(
&self,
room_id: &[u8],
user_id: &[u8],
membership: &[u8],
stripped: bool,
data: &[u8],
) -> rusqlite::Result<()> {
self.prepare_cached(
"INSERT OR REPLACE
INTO member (room_id, user_id, membership, stripped, data)
VALUES (?, ?, ?, ?, ?)",
)?
.execute((room_id, user_id, membership, stripped, data))?;
Ok(())
}
fn remove_room_members(&self, room_id: &[u8], stripped: Option<bool>) -> rusqlite::Result<()> {
if let Some(stripped) = stripped {
self.prepare_cached("DELETE FROM member WHERE room_id = ? AND stripped = ?")?
.execute((room_id, stripped))?;
} else {
self.prepare_cached("DELETE FROM member WHERE room_id = ?")?.execute((room_id,))?;
}
Ok(())
}
fn set_profile(&self, room_id: &[u8], user_id: &[u8], data: &[u8]) -> rusqlite::Result<()> {
self.prepare_cached(
"INSERT OR REPLACE
INTO profile (room_id, user_id, data)
VALUES (?, ?, ?)",
)?
.execute((room_id, user_id, data))?;
Ok(())
}
fn remove_room_profiles(&self, room_id: &[u8]) -> rusqlite::Result<()> {
self.prepare("DELETE FROM profile WHERE room_id = ?")?.execute((room_id,))?;
Ok(())
}
fn remove_room_profile(&self, room_id: &[u8], user_id: &[u8]) -> rusqlite::Result<()> {
self.prepare("DELETE FROM profile WHERE room_id = ? AND user_id = ?")?
.execute((room_id, user_id))?;
Ok(())
}
fn set_receipt(
&self,
room_id: &[u8],
user_id: &[u8],
receipt_type: &[u8],
thread: &[u8],
event_id: &[u8],
data: &[u8],
) -> rusqlite::Result<()> {
self.prepare_cached(
"INSERT OR REPLACE
INTO receipt (room_id, user_id, receipt_type, thread, event_id, data)
VALUES (?, ?, ?, ?, ?, ?)",
)?
.execute((room_id, user_id, receipt_type, thread, event_id, data))?;
Ok(())
}
fn remove_room_receipts(&self, room_id: &[u8]) -> rusqlite::Result<()> {
self.prepare("DELETE FROM receipt WHERE room_id = ?")?.execute((room_id,))?;
Ok(())
}
fn set_display_name(&self, room_id: &[u8], name: &[u8], data: &[u8]) -> rusqlite::Result<()> {
self.prepare_cached(
"INSERT OR REPLACE
INTO display_name (room_id, name, data)
VALUES (?, ?, ?)",
)?
.execute((room_id, name, data))?;
Ok(())
}
fn remove_display_name(&self, room_id: &[u8], name: &[u8]) -> rusqlite::Result<()> {
self.prepare("DELETE FROM display_name WHERE room_id = ? AND name = ?")?
.execute((room_id, name))?;
Ok(())
}
fn remove_room_display_names(&self, room_id: &[u8]) -> rusqlite::Result<()> {
self.prepare("DELETE FROM display_name WHERE room_id = ?")?.execute((room_id,))?;
Ok(())
}
fn remove_room_send_queue(&self, room_id: &[u8]) -> rusqlite::Result<()> {
self.prepare("DELETE FROM send_queue_events WHERE room_id = ?")?.execute((room_id,))?;
Ok(())
}
fn remove_room_dependent_send_queue(&self, room_id: &[u8]) -> rusqlite::Result<()> {
self.prepare("DELETE FROM dependent_send_queue_events WHERE room_id = ?")?
.execute((room_id,))?;
Ok(())
}
}
#[async_trait]
trait SqliteObjectStateStoreExt: SqliteAsyncConnExt {
async fn get_kv_blob(&self, key: Key) -> Result<Option<Vec<u8>>> {
Ok(self
.query_row("SELECT value FROM kv_blob WHERE key = ?", (key,), |row| row.get(0))
.await
.optional()?)
}
async fn get_kv_blobs(&self, keys: Vec<Key>) -> Result<Vec<Vec<u8>>> {
let keys_length = keys.len();
self.chunk_large_query_over(keys, Some(keys_length), |txn, keys| {
let sql_params = repeat_vars(keys.len());
let sql = format!("SELECT value FROM kv_blob WHERE key IN ({sql_params})");
let params = rusqlite::params_from_iter(keys);
Ok(txn
.prepare(&sql)?
.query(params)?
.mapped(|row| row.get(0))
.collect::<Result<_, _>>()?)
})
.await
}
async fn set_kv_blob(&self, key: Key, value: Vec<u8>) -> Result<()>;
async fn delete_kv_blob(&self, key: Key) -> Result<()> {
self.execute("DELETE FROM kv_blob WHERE key = ?", (key,)).await?;
Ok(())
}
async fn get_room_infos(&self) -> Result<Vec<Vec<u8>>> {
Ok(self
.prepare("SELECT data FROM room_info", move |mut stmt| {
stmt.query_map((), |row| row.get(0))?.collect()
})
.await?)
}
async fn get_maybe_stripped_state_events_for_keys(
&self,
room_id: Key,
event_type: Key,
state_keys: Vec<Key>,
) -> Result<Vec<(bool, Vec<u8>)>> {
self.chunk_large_query_over(state_keys, None, move |txn, state_keys: Vec<Key>| {
let sql_params = repeat_vars(state_keys.len());
let sql = format!(
"SELECT stripped, data FROM state_event
WHERE room_id = ? AND event_type = ? AND state_key IN ({sql_params})"
);
let params = rusqlite::params_from_iter(
[room_id.clone(), event_type.clone()].into_iter().chain(state_keys),
);
Ok(txn
.prepare(&sql)?
.query(params)?
.mapped(|row| Ok((row.get(0)?, row.get(1)?)))
.collect::<Result<_, _>>()?)
})
.await
}
async fn get_maybe_stripped_state_events(
&self,
room_id: Key,
event_type: Key,
) -> Result<Vec<(bool, Vec<u8>)>> {
Ok(self
.prepare(
"SELECT stripped, data FROM state_event
WHERE room_id = ? AND event_type = ?",
|mut stmt| {
stmt.query((room_id, event_type))?
.mapped(|row| Ok((row.get(0)?, row.get(1)?)))
.collect()
},
)
.await?)
}
async fn get_profiles(
&self,
room_id: Key,
user_ids: Vec<Key>,
) -> Result<Vec<(Vec<u8>, Vec<u8>)>> {
let user_ids_length = user_ids.len();
self.chunk_large_query_over(user_ids, Some(user_ids_length), move |txn, user_ids| {
let sql_params = repeat_vars(user_ids.len());
let sql = format!(
"SELECT user_id, data FROM profile WHERE room_id = ? AND user_id IN ({sql_params})"
);
let params = rusqlite::params_from_iter(iter::once(room_id.clone()).chain(user_ids));
Ok(txn
.prepare(&sql)?
.query(params)?
.mapped(|row| Ok((row.get(0)?, row.get(1)?)))
.collect::<Result<_, _>>()?)
})
.await
}
async fn get_user_ids(&self, room_id: Key, memberships: Vec<Key>) -> Result<Vec<Vec<u8>>> {
let res = if memberships.is_empty() {
self.prepare("SELECT data FROM member WHERE room_id = ?", |mut stmt| {
stmt.query((room_id,))?.mapped(|row| row.get(0)).collect()
})
.await?
} else {
self.chunk_large_query_over(memberships, None, move |txn, memberships| {
let sql_params = repeat_vars(memberships.len());
let sql = format!(
"SELECT data FROM member WHERE room_id = ? AND membership IN ({sql_params})"
);
let params =
rusqlite::params_from_iter(iter::once(room_id.clone()).chain(memberships));
Ok(txn
.prepare(&sql)?
.query(params)?
.mapped(|row| row.get(0))
.collect::<Result<_, _>>()?)
})
.await?
};
Ok(res)
}
async fn get_global_account_data(&self, event_type: Key) -> Result<Option<Vec<u8>>> {
Ok(self
.query_row(
"SELECT data FROM global_account_data WHERE event_type = ?",
(event_type,),
|row| row.get(0),
)
.await
.optional()?)
}
async fn get_room_account_data(
&self,
room_id: Key,
event_type: Key,
) -> Result<Option<Vec<u8>>> {
Ok(self
.query_row(
"SELECT data FROM room_account_data WHERE room_id = ? AND event_type = ?",
(room_id, event_type),
|row| row.get(0),
)
.await
.optional()?)
}
async fn get_display_names(
&self,
room_id: Key,
names: Vec<Key>,
) -> Result<Vec<(Vec<u8>, Vec<u8>)>> {
let names_length = names.len();
self.chunk_large_query_over(names, Some(names_length), move |txn, names| {
let sql_params = repeat_vars(names.len());
let sql = format!(
"SELECT name, data FROM display_name WHERE room_id = ? AND name IN ({sql_params})"
);
let params = rusqlite::params_from_iter(iter::once(room_id.clone()).chain(names));
Ok(txn
.prepare(&sql)?
.query(params)?
.mapped(|row| Ok((row.get(0)?, row.get(1)?)))
.collect::<Result<_, _>>()?)
})
.await
}
async fn get_user_receipt(
&self,
room_id: Key,
receipt_type: Key,
thread: Key,
user_id: Key,
) -> Result<Option<Vec<u8>>> {
Ok(self
.query_row(
"SELECT data FROM receipt
WHERE room_id = ? AND receipt_type = ? AND thread = ? and user_id = ?",
(room_id, receipt_type, thread, user_id),
|row| row.get(0),
)
.await
.optional()?)
}
async fn get_event_receipts(
&self,
room_id: Key,
receipt_type: Key,
thread: Key,
event_id: Key,
) -> Result<Vec<Vec<u8>>> {
Ok(self
.prepare(
"SELECT data FROM receipt
WHERE room_id = ? AND receipt_type = ? AND thread = ? and event_id = ?",
|mut stmt| {
stmt.query((room_id, receipt_type, thread, event_id))?
.mapped(|row| row.get(0))
.collect()
},
)
.await?)
}
}
#[async_trait]
impl SqliteObjectStateStoreExt for SqliteAsyncConn {
async fn set_kv_blob(&self, key: Key, value: Vec<u8>) -> Result<()> {
Ok(self.interact(move |conn| conn.set_kv_blob(&key, &value)).await.unwrap()?)
}
}
#[async_trait]
impl StateStore for SqliteStateStore {
type Error = Error;
async fn get_kv_data(&self, key: StateStoreDataKey<'_>) -> Result<Option<StateStoreDataValue>> {
self.acquire()
.await?
.get_kv_blob(self.encode_state_store_data_key(key))
.await?
.map(|data| {
Ok(match key {
StateStoreDataKey::SyncToken => {
StateStoreDataValue::SyncToken(self.deserialize_value(&data)?)
}
StateStoreDataKey::ServerCapabilities => {
StateStoreDataValue::ServerCapabilities(self.deserialize_value(&data)?)
}
StateStoreDataKey::Filter(_) => {
StateStoreDataValue::Filter(self.deserialize_value(&data)?)
}
StateStoreDataKey::UserAvatarUrl(_) => {
StateStoreDataValue::UserAvatarUrl(self.deserialize_value(&data)?)
}
StateStoreDataKey::RecentlyVisitedRooms(_) => {
StateStoreDataValue::RecentlyVisitedRooms(self.deserialize_value(&data)?)
}
StateStoreDataKey::UtdHookManagerData => {
StateStoreDataValue::UtdHookManagerData(self.deserialize_value(&data)?)
}
StateStoreDataKey::ComposerDraft(_) => {
StateStoreDataValue::ComposerDraft(self.deserialize_value(&data)?)
}
StateStoreDataKey::SeenKnockRequests(_) => {
StateStoreDataValue::SeenKnockRequests(self.deserialize_value(&data)?)
}
})
})
.transpose()
}
async fn set_kv_data(
&self,
key: StateStoreDataKey<'_>,
value: StateStoreDataValue,
) -> Result<()> {
let serialized_value = match key {
StateStoreDataKey::SyncToken => self.serialize_value(
&value.into_sync_token().expect("Session data not a sync token"),
)?,
StateStoreDataKey::ServerCapabilities => self.serialize_value(
&value
.into_server_capabilities()
.expect("Session data not containing server capabilities"),
)?,
StateStoreDataKey::Filter(_) => {
self.serialize_value(&value.into_filter().expect("Session data not a filter"))?
}
StateStoreDataKey::UserAvatarUrl(_) => self.serialize_value(
&value.into_user_avatar_url().expect("Session data not an user avatar url"),
)?,
StateStoreDataKey::RecentlyVisitedRooms(_) => self.serialize_value(
&value.into_recently_visited_rooms().expect("Session data not breadcrumbs"),
)?,
StateStoreDataKey::UtdHookManagerData => self.serialize_value(
&value.into_utd_hook_manager_data().expect("Session data not UtdHookManagerData"),
)?,
StateStoreDataKey::ComposerDraft(_) => self.serialize_value(
&value.into_composer_draft().expect("Session data not a composer draft"),
)?,
StateStoreDataKey::SeenKnockRequests(_) => self.serialize_value(
&value
.into_seen_knock_requests()
.expect("Session data is not a set of seen knock request ids"),
)?,
};
self.acquire()
.await?
.set_kv_blob(self.encode_state_store_data_key(key), serialized_value)
.await
}
async fn remove_kv_data(&self, key: StateStoreDataKey<'_>) -> Result<()> {
self.acquire().await?.delete_kv_blob(self.encode_state_store_data_key(key)).await
}
async fn save_changes(&self, changes: &StateChanges) -> Result<()> {
let changes = changes.to_owned();
let this = self.clone();
self.acquire()
.await?
.with_transaction(move |txn| {
let StateChanges {
sync_token,
account_data,
presence,
profiles,
profiles_to_delete,
state,
room_account_data,
room_infos,
receipts,
redactions,
stripped_state,
ambiguity_maps,
} = changes;
if let Some(sync_token) = sync_token {
let key = this.encode_state_store_data_key(StateStoreDataKey::SyncToken);
let value = this.serialize_value(&sync_token)?;
txn.set_kv_blob(&key, &value)?;
}
for (event_type, event) in account_data {
let event_type =
this.encode_key(keys::GLOBAL_ACCOUNT_DATA, event_type.to_string());
let data = this.serialize_json(&event)?;
txn.set_global_account_data(&event_type, &data)?;
}
for (room_id, events) in room_account_data {
let room_id = this.encode_key(keys::ROOM_ACCOUNT_DATA, room_id);
for (event_type, event) in events {
let event_type =
this.encode_key(keys::ROOM_ACCOUNT_DATA, event_type.to_string());
let data = this.serialize_json(&event)?;
txn.set_room_account_data(&room_id, &event_type, &data)?;
}
}
for (user_id, event) in presence {
let key = this.encode_presence_key(&user_id);
let value = this.serialize_json(&event)?;
txn.set_kv_blob(&key, &value)?;
}
for (room_id, room_info) in room_infos {
let stripped = room_info.state() == RoomState::Invited;
this.remove_maybe_stripped_room_data(txn, &room_id, !stripped)?;
let room_id = this.encode_key(keys::ROOM_INFO, room_id);
let state = this
.encode_key(keys::ROOM_INFO, serde_json::to_string(&room_info.state())?);
let data = this.serialize_json(&room_info)?;
txn.set_room_info(&room_id, &state, &data)?;
}
for (room_id, user_ids) in profiles_to_delete {
let room_id = this.encode_key(keys::PROFILE, room_id);
for user_id in user_ids {
let user_id = this.encode_key(keys::PROFILE, user_id);
txn.remove_room_profile(&room_id, &user_id)?;
}
}
for (room_id, state_event_types) in state {
let profiles = profiles.get(&room_id);
let encoded_room_id = this.encode_key(keys::STATE_EVENT, &room_id);
for (event_type, state_events) in state_event_types {
let encoded_event_type =
this.encode_key(keys::STATE_EVENT, event_type.to_string());
for (state_key, raw_state_event) in state_events {
let encoded_state_key = this.encode_key(keys::STATE_EVENT, &state_key);
let data = this.serialize_json(&raw_state_event)?;
let event_id: Option<String> =
raw_state_event.get_field("event_id").ok().flatten();
let encoded_event_id =
event_id.as_ref().map(|e| this.encode_key(keys::STATE_EVENT, e));
txn.set_state_event(
&encoded_room_id,
&encoded_event_type,
&encoded_state_key,
false,
encoded_event_id.as_deref(),
&data,
)?;
if event_type == StateEventType::RoomMember {
let member_event = match raw_state_event
.deserialize_as::<SyncRoomMemberEvent>()
{
Ok(ev) => ev,
Err(e) => {
debug!(event_id, "Failed to deserialize member event: {e}");
continue;
}
};
let encoded_room_id = this.encode_key(keys::MEMBER, &room_id);
let user_id = this.encode_key(keys::MEMBER, &state_key);
let membership = this
.encode_key(keys::MEMBER, member_event.membership().as_str());
let data = this.serialize_value(&state_key)?;
txn.set_member(
&encoded_room_id,
&user_id,
&membership,
false,
&data,
)?;
if let Some(profile) =
profiles.and_then(|p| p.get(member_event.state_key()))
{
let room_id = this.encode_key(keys::PROFILE, &room_id);
let user_id = this.encode_key(keys::PROFILE, &state_key);
let data = this.serialize_json(&profile)?;
txn.set_profile(&room_id, &user_id, &data)?;
}
}
}
}
}
for (room_id, stripped_state_event_types) in stripped_state {
let encoded_room_id = this.encode_key(keys::STATE_EVENT, &room_id);
for (event_type, stripped_state_events) in stripped_state_event_types {
let encoded_event_type =
this.encode_key(keys::STATE_EVENT, event_type.to_string());
for (state_key, raw_stripped_state_event) in stripped_state_events {
let encoded_state_key = this.encode_key(keys::STATE_EVENT, &state_key);
let data = this.serialize_json(&raw_stripped_state_event)?;
txn.set_state_event(
&encoded_room_id,
&encoded_event_type,
&encoded_state_key,
true,
None,
&data,
)?;
if event_type == StateEventType::RoomMember {
let member_event = match raw_stripped_state_event
.deserialize_as::<StrippedRoomMemberEvent>(
) {
Ok(ev) => ev,
Err(e) => {
debug!("Failed to deserialize stripped member event: {e}");
continue;
}
};
let room_id = this.encode_key(keys::MEMBER, &room_id);
let user_id = this.encode_key(keys::MEMBER, &state_key);
let membership = this.encode_key(
keys::MEMBER,
member_event.content.membership.as_str(),
);
let data = this.serialize_value(&state_key)?;
txn.set_member(&room_id, &user_id, &membership, true, &data)?;
}
}
}
}
for (room_id, receipt_event) in receipts {
let room_id = this.encode_key(keys::RECEIPT, room_id);
for (event_id, receipt_types) in receipt_event {
let encoded_event_id = this.encode_key(keys::RECEIPT, &event_id);
for (receipt_type, receipt_users) in receipt_types {
let receipt_type =
this.encode_key(keys::RECEIPT, receipt_type.as_str());
for (user_id, receipt) in receipt_users {
let encoded_user_id = this.encode_key(keys::RECEIPT, &user_id);
let thread = this.encode_key(
keys::RECEIPT,
rmp_serde::to_vec_named(&receipt.thread)?,
);
let data = this.serialize_json(&ReceiptData {
receipt,
event_id: event_id.clone(),
user_id,
})?;
txn.set_receipt(
&room_id,
&encoded_user_id,
&receipt_type,
&thread,
&encoded_event_id,
&data,
)?;
}
}
}
}
for (room_id, redactions) in redactions {
let make_room_version = || {
let encoded_room_id = this.encode_key(keys::ROOM_INFO, &room_id);
txn.get_room_info(&encoded_room_id)
.ok()
.flatten()
.and_then(|v| this.deserialize_json::<RoomInfo>(&v).ok())
.and_then(|info| info.room_version().cloned())
.unwrap_or_else(|| {
warn!(
?room_id,
"Unable to find the room version, assume version 9"
);
RoomVersionId::V9
})
};
let encoded_room_id = this.encode_key(keys::STATE_EVENT, &room_id);
let mut room_version = None;
for (event_id, redaction) in redactions {
let event_id = this.encode_key(keys::STATE_EVENT, event_id);
if let Some(Ok(raw_event)) = txn
.get_state_event_by_id(&encoded_room_id, &event_id)?
.map(|value| this.deserialize_json::<Raw<AnySyncStateEvent>>(&value))
{
let event = raw_event.deserialize()?;
let redacted = redact(
raw_event.deserialize_as::<CanonicalJsonObject>()?,
room_version.get_or_insert_with(make_room_version),
Some(RedactedBecause::from_raw_event(&redaction)?),
)
.map_err(Error::Redaction)?;
let data = this.serialize_json(&redacted)?;
let event_type =
this.encode_key(keys::STATE_EVENT, event.event_type().to_string());
let state_key = this.encode_key(keys::STATE_EVENT, event.state_key());
txn.set_state_event(
&encoded_room_id,
&event_type,
&state_key,
false,
Some(&event_id),
&data,
)?;
}
}
}
for (room_id, display_names) in ambiguity_maps {
let room_id = this.encode_key(keys::DISPLAY_NAME, room_id);
for (name, user_ids) in display_names {
let encoded_name = this.encode_key(
keys::DISPLAY_NAME,
name.as_normalized_str().unwrap_or_else(|| name.as_raw_str()),
);
let data = this.serialize_json(&user_ids)?;
if user_ids.is_empty() {
txn.remove_display_name(&room_id, &encoded_name)?;
let raw_name = this.encode_key(keys::DISPLAY_NAME, name.as_raw_str());
txn.remove_display_name(&room_id, &raw_name)?;
} else {
txn.set_display_name(&room_id, &encoded_name, &data)?;
}
}
}
Ok::<_, Error>(())
})
.await?;
Ok(())
}
async fn get_presence_event(&self, user_id: &UserId) -> Result<Option<Raw<PresenceEvent>>> {
self.acquire()
.await?
.get_kv_blob(self.encode_presence_key(user_id))
.await?
.map(|data| self.deserialize_json(&data))
.transpose()
}
async fn get_presence_events(
&self,
user_ids: &[OwnedUserId],
) -> Result<Vec<Raw<PresenceEvent>>> {
if user_ids.is_empty() {
return Ok(Vec::new());
}
let user_ids = user_ids.iter().map(|u| self.encode_presence_key(u)).collect();
self.acquire()
.await?
.get_kv_blobs(user_ids)
.await?
.into_iter()
.map(|data| self.deserialize_json(&data))
.collect()
}
async fn get_state_event(
&self,
room_id: &RoomId,
event_type: StateEventType,
state_key: &str,
) -> Result<Option<RawAnySyncOrStrippedState>> {
Ok(self
.get_state_events_for_keys(room_id, event_type, &[state_key])
.await?
.into_iter()
.next())
}
async fn get_state_events(
&self,
room_id: &RoomId,
event_type: StateEventType,
) -> Result<Vec<RawAnySyncOrStrippedState>> {
let room_id = self.encode_key(keys::STATE_EVENT, room_id);
let event_type = self.encode_key(keys::STATE_EVENT, event_type.to_string());
self.acquire()
.await?
.get_maybe_stripped_state_events(room_id, event_type)
.await?
.into_iter()
.map(|(stripped, data)| {
let ev = if stripped {
RawAnySyncOrStrippedState::Stripped(self.deserialize_json(&data)?)
} else {
RawAnySyncOrStrippedState::Sync(self.deserialize_json(&data)?)
};
Ok(ev)
})
.collect()
}
async fn get_state_events_for_keys(
&self,
room_id: &RoomId,
event_type: StateEventType,
state_keys: &[&str],
) -> Result<Vec<RawAnySyncOrStrippedState>, Self::Error> {
if state_keys.is_empty() {
return Ok(Vec::new());
}
let room_id = self.encode_key(keys::STATE_EVENT, room_id);
let event_type = self.encode_key(keys::STATE_EVENT, event_type.to_string());
let state_keys = state_keys.iter().map(|k| self.encode_key(keys::STATE_EVENT, k)).collect();
self.acquire()
.await?
.get_maybe_stripped_state_events_for_keys(room_id, event_type, state_keys)
.await?
.into_iter()
.map(|(stripped, data)| {
let ev = if stripped {
RawAnySyncOrStrippedState::Stripped(self.deserialize_json(&data)?)
} else {
RawAnySyncOrStrippedState::Sync(self.deserialize_json(&data)?)
};
Ok(ev)
})
.collect()
}
async fn get_profile(
&self,
room_id: &RoomId,
user_id: &UserId,
) -> Result<Option<MinimalRoomMemberEvent>> {
let room_id = self.encode_key(keys::PROFILE, room_id);
let user_ids = vec![self.encode_key(keys::PROFILE, user_id)];
self.acquire()
.await?
.get_profiles(room_id, user_ids)
.await?
.into_iter()
.next()
.map(|(_, data)| self.deserialize_json(&data))
.transpose()
}
async fn get_profiles<'a>(
&self,
room_id: &RoomId,
user_ids: &'a [OwnedUserId],
) -> Result<BTreeMap<&'a UserId, MinimalRoomMemberEvent>> {
if user_ids.is_empty() {
return Ok(BTreeMap::new());
}
let room_id = self.encode_key(keys::PROFILE, room_id);
let mut user_ids_map = user_ids
.iter()
.map(|u| (self.encode_key(keys::PROFILE, u), u.as_ref()))
.collect::<BTreeMap<_, _>>();
let user_ids = user_ids_map.keys().cloned().collect();
self.acquire()
.await?
.get_profiles(room_id, user_ids)
.await?
.into_iter()
.map(|(user_id, data)| {
Ok((
user_ids_map
.remove(user_id.as_slice())
.expect("returned user IDs were requested"),
self.deserialize_json(&data)?,
))
})
.collect()
}
async fn get_user_ids(
&self,
room_id: &RoomId,
membership: RoomMemberships,
) -> Result<Vec<OwnedUserId>> {
let room_id = self.encode_key(keys::MEMBER, room_id);
let memberships = membership
.as_vec()
.into_iter()
.map(|m| self.encode_key(keys::MEMBER, m.as_str()))
.collect();
self.acquire()
.await?
.get_user_ids(room_id, memberships)
.await?
.iter()
.map(|data| self.deserialize_value(data))
.collect()
}
async fn get_room_infos(&self) -> Result<Vec<RoomInfo>> {
self.acquire()
.await?
.get_room_infos()
.await?
.into_iter()
.map(|data| self.deserialize_json(&data))
.collect()
}
async fn get_users_with_display_name(
&self,
room_id: &RoomId,
display_name: &DisplayName,
) -> Result<BTreeSet<OwnedUserId>> {
let room_id = self.encode_key(keys::DISPLAY_NAME, room_id);
let names = vec![self.encode_key(
keys::DISPLAY_NAME,
display_name.as_normalized_str().unwrap_or_else(|| display_name.as_raw_str()),
)];
Ok(self
.acquire()
.await?
.get_display_names(room_id, names)
.await?
.into_iter()
.next()
.map(|(_, data)| self.deserialize_json(&data))
.transpose()?
.unwrap_or_default())
}
async fn get_users_with_display_names<'a>(
&self,
room_id: &RoomId,
display_names: &'a [DisplayName],
) -> Result<HashMap<&'a DisplayName, BTreeSet<OwnedUserId>>> {
let mut result = HashMap::new();
if display_names.is_empty() {
return Ok(result);
}
let room_id = self.encode_key(keys::DISPLAY_NAME, room_id);
let mut names_map = display_names
.iter()
.flat_map(|display_name| {
let raw =
(self.encode_key(keys::DISPLAY_NAME, display_name.as_raw_str()), display_name);
let normalized = display_name.as_normalized_str().map(|normalized| {
(self.encode_key(keys::DISPLAY_NAME, normalized), display_name)
});
iter::once(raw).chain(normalized.into_iter())
})
.collect::<BTreeMap<_, _>>();
let names = names_map.keys().cloned().collect();
for (name, data) in
self.acquire().await?.get_display_names(room_id, names).await?.into_iter()
{
let display_name =
names_map.remove(name.as_slice()).expect("returned display names were requested");
let user_ids: BTreeSet<_> = self.deserialize_json(&data)?;
result.entry(display_name).or_insert_with(BTreeSet::new).extend(user_ids);
}
Ok(result)
}
async fn get_account_data_event(
&self,
event_type: GlobalAccountDataEventType,
) -> Result<Option<Raw<AnyGlobalAccountDataEvent>>> {
let event_type = self.encode_key(keys::GLOBAL_ACCOUNT_DATA, event_type.to_string());
self.acquire()
.await?
.get_global_account_data(event_type)
.await?
.map(|value| self.deserialize_json(&value))
.transpose()
}
async fn get_room_account_data_event(
&self,
room_id: &RoomId,
event_type: RoomAccountDataEventType,
) -> Result<Option<Raw<AnyRoomAccountDataEvent>>> {
let room_id = self.encode_key(keys::ROOM_ACCOUNT_DATA, room_id);
let event_type = self.encode_key(keys::ROOM_ACCOUNT_DATA, event_type.to_string());
self.acquire()
.await?
.get_room_account_data(room_id, event_type)
.await?
.map(|value| self.deserialize_json(&value))
.transpose()
}
async fn get_user_room_receipt_event(
&self,
room_id: &RoomId,
receipt_type: ReceiptType,
thread: ReceiptThread,
user_id: &UserId,
) -> Result<Option<(OwnedEventId, Receipt)>> {
let room_id = self.encode_key(keys::RECEIPT, room_id);
let receipt_type = self.encode_key(keys::RECEIPT, receipt_type.to_string());
let thread = self.encode_key(keys::RECEIPT, rmp_serde::to_vec_named(&thread)?);
let user_id = self.encode_key(keys::RECEIPT, user_id);
self.acquire()
.await?
.get_user_receipt(room_id, receipt_type, thread, user_id)
.await?
.map(|value| {
self.deserialize_json::<ReceiptData>(&value).map(|d| (d.event_id, d.receipt))
})
.transpose()
}
async fn get_event_room_receipt_events(
&self,
room_id: &RoomId,
receipt_type: ReceiptType,
thread: ReceiptThread,
event_id: &EventId,
) -> Result<Vec<(OwnedUserId, Receipt)>> {
let room_id = self.encode_key(keys::RECEIPT, room_id);
let receipt_type = self.encode_key(keys::RECEIPT, receipt_type.to_string());
let thread = self.encode_key(keys::RECEIPT, rmp_serde::to_vec_named(&thread)?);
let event_id = self.encode_key(keys::RECEIPT, event_id);
self.acquire()
.await?
.get_event_receipts(room_id, receipt_type, thread, event_id)
.await?
.iter()
.map(|value| {
self.deserialize_json::<ReceiptData>(value).map(|d| (d.user_id, d.receipt))
})
.collect()
}
async fn get_custom_value(&self, key: &[u8]) -> Result<Option<Vec<u8>>> {
self.acquire().await?.get_kv_blob(self.encode_custom_key(key)).await
}
async fn set_custom_value_no_read(&self, key: &[u8], value: Vec<u8>) -> Result<()> {
let conn = self.acquire().await?;
let key = self.encode_custom_key(key);
conn.set_kv_blob(key, value).await?;
Ok(())
}
async fn set_custom_value(&self, key: &[u8], value: Vec<u8>) -> Result<Option<Vec<u8>>> {
let conn = self.acquire().await?;
let key = self.encode_custom_key(key);
let previous = conn.get_kv_blob(key.clone()).await?;
conn.set_kv_blob(key, value).await?;
Ok(previous)
}
async fn remove_custom_value(&self, key: &[u8]) -> Result<Option<Vec<u8>>> {
let conn = self.acquire().await?;
let key = self.encode_custom_key(key);
let previous = conn.get_kv_blob(key.clone()).await?;
if previous.is_some() {
conn.delete_kv_blob(key).await?;
}
Ok(previous)
}
async fn remove_room(&self, room_id: &RoomId) -> Result<()> {
let this = self.clone();
let room_id = room_id.to_owned();
self.acquire()
.await?
.with_transaction(move |txn| {
let room_info_room_id = this.encode_key(keys::ROOM_INFO, &room_id);
txn.remove_room_info(&room_info_room_id)?;
let state_event_room_id = this.encode_key(keys::STATE_EVENT, &room_id);
txn.remove_room_state_events(&state_event_room_id, None)?;
let member_room_id = this.encode_key(keys::MEMBER, &room_id);
txn.remove_room_members(&member_room_id, None)?;
let profile_room_id = this.encode_key(keys::PROFILE, &room_id);
txn.remove_room_profiles(&profile_room_id)?;
let room_account_data_room_id = this.encode_key(keys::ROOM_ACCOUNT_DATA, &room_id);
txn.remove_room_account_data(&room_account_data_room_id)?;
let receipt_room_id = this.encode_key(keys::RECEIPT, &room_id);
txn.remove_room_receipts(&receipt_room_id)?;
let display_name_room_id = this.encode_key(keys::DISPLAY_NAME, &room_id);
txn.remove_room_display_names(&display_name_room_id)?;
let send_queue_room_id = this.encode_key(keys::SEND_QUEUE, &room_id);
txn.remove_room_send_queue(&send_queue_room_id)?;
let dependent_send_queue_room_id =
this.encode_key(keys::DEPENDENTS_SEND_QUEUE, &room_id);
txn.remove_room_dependent_send_queue(&dependent_send_queue_room_id)?;
Ok(())
})
.await
}
async fn save_send_queue_request(
&self,
room_id: &RoomId,
transaction_id: OwnedTransactionId,
created_at: MilliSecondsSinceUnixEpoch,
content: QueuedRequestKind,
priority: usize,
) -> Result<(), Self::Error> {
let room_id_key = self.encode_key(keys::SEND_QUEUE, room_id);
let room_id_value = self.serialize_value(&room_id.to_owned())?;
let content = self.serialize_json(&content)?;
let created_at_ts: u64 = created_at.0.into();
self.acquire()
.await?
.with_transaction(move |txn| {
txn.prepare_cached("INSERT INTO send_queue_events (room_id, room_id_val, transaction_id, content, priority, created_at) VALUES (?, ?, ?, ?, ?, ?)")?.execute((room_id_key, room_id_value, transaction_id.to_string(), content, priority, created_at_ts))?;
Ok(())
})
.await
}
async fn update_send_queue_request(
&self,
room_id: &RoomId,
transaction_id: &TransactionId,
content: QueuedRequestKind,
) -> Result<bool, Self::Error> {
let room_id = self.encode_key(keys::SEND_QUEUE, room_id);
let content = self.serialize_json(&content)?;
let transaction_id = transaction_id.to_string();
let num_updated = self.acquire()
.await?
.with_transaction(move |txn| {
txn.prepare_cached("UPDATE send_queue_events SET wedge_reason = NULL, content = ? WHERE room_id = ? AND transaction_id = ?")?.execute((content, room_id, transaction_id))
})
.await?;
Ok(num_updated > 0)
}
async fn remove_send_queue_request(
&self,
room_id: &RoomId,
transaction_id: &TransactionId,
) -> Result<bool, Self::Error> {
let room_id = self.encode_key(keys::SEND_QUEUE, room_id);
let transaction_id = transaction_id.to_string();
let num_deleted = self
.acquire()
.await?
.with_transaction(move |txn| {
txn.prepare_cached(
"DELETE FROM send_queue_events WHERE room_id = ? AND transaction_id = ?",
)?
.execute((room_id, &transaction_id))
})
.await?;
Ok(num_deleted > 0)
}
async fn load_send_queue_requests(
&self,
room_id: &RoomId,
) -> Result<Vec<QueuedRequest>, Self::Error> {
let room_id = self.encode_key(keys::SEND_QUEUE, room_id);
let res: Vec<(String, Vec<u8>, Option<Vec<u8>>, usize, Option<u64>)> = self
.acquire()
.await?
.prepare(
"SELECT transaction_id, content, wedge_reason, priority, created_at FROM send_queue_events WHERE room_id = ? ORDER BY priority DESC, ROWID",
|mut stmt| {
stmt.query((room_id,))?
.mapped(|row| Ok((row.get(0)?, row.get(1)?, row.get(2)?, row.get(3)?, row.get(4)?)))
.collect()
},
)
.await?;
let mut requests = Vec::with_capacity(res.len());
for entry in res {
let created_at = entry
.4
.and_then(UInt::new)
.map_or_else(MilliSecondsSinceUnixEpoch::now, MilliSecondsSinceUnixEpoch);
requests.push(QueuedRequest {
transaction_id: entry.0.into(),
kind: self.deserialize_json(&entry.1)?,
error: entry.2.map(|v| self.deserialize_value(&v)).transpose()?,
priority: entry.3,
created_at,
});
}
Ok(requests)
}
async fn update_send_queue_request_status(
&self,
room_id: &RoomId,
transaction_id: &TransactionId,
error: Option<QueueWedgeError>,
) -> Result<(), Self::Error> {
let room_id = self.encode_key(keys::SEND_QUEUE, room_id);
let transaction_id = transaction_id.to_string();
let error_value = error.map(|e| self.serialize_value(&e)).transpose()?;
self.acquire()
.await?
.with_transaction(move |txn| {
txn.prepare_cached("UPDATE send_queue_events SET wedge_reason = ? WHERE room_id = ? AND transaction_id = ?")?.execute((error_value, room_id, transaction_id))?;
Ok(())
})
.await
}
async fn load_rooms_with_unsent_requests(&self) -> Result<Vec<OwnedRoomId>, Self::Error> {
let res: Vec<Vec<u8>> = self
.acquire()
.await?
.prepare("SELECT room_id_val FROM send_queue_events", |mut stmt| {
stmt.query(())?.mapped(|row| row.get(0)).collect()
})
.await?;
Ok(res
.into_iter()
.map(|entry| self.deserialize_value(&entry))
.collect::<Result<BTreeSet<OwnedRoomId>, _>>()?
.into_iter()
.collect())
}
async fn save_dependent_queued_request(
&self,
room_id: &RoomId,
parent_txn_id: &TransactionId,
own_txn_id: ChildTransactionId,
created_at: MilliSecondsSinceUnixEpoch,
content: DependentQueuedRequestKind,
) -> Result<()> {
let room_id = self.encode_key(keys::DEPENDENTS_SEND_QUEUE, room_id);
let content = self.serialize_json(&content)?;
let parent_txn_id = parent_txn_id.to_string();
let own_txn_id = own_txn_id.to_string();
let created_at_ts: u64 = created_at.0.into();
self.acquire()
.await?
.with_transaction(move |txn| {
txn.prepare_cached(
r#"INSERT INTO dependent_send_queue_events
(room_id, parent_transaction_id, own_transaction_id, content, created_at)
VALUES (?, ?, ?, ?, ?)"#,
)?
.execute((
room_id,
parent_txn_id,
own_txn_id,
content,
created_at_ts,
))?;
Ok(())
})
.await
}
async fn update_dependent_queued_request(
&self,
room_id: &RoomId,
own_transaction_id: &ChildTransactionId,
new_content: DependentQueuedRequestKind,
) -> Result<bool> {
let room_id = self.encode_key(keys::DEPENDENTS_SEND_QUEUE, room_id);
let content = self.serialize_json(&new_content)?;
let own_txn_id = own_transaction_id.to_string();
let num_updated = self
.acquire()
.await?
.with_transaction(move |txn| {
txn.prepare_cached(
r#"UPDATE dependent_send_queue_events
SET content = ?
WHERE own_transaction_id = ?
AND room_id = ?"#,
)?
.execute((content, own_txn_id, room_id))
})
.await?;
if num_updated > 1 {
return Err(Error::InconsistentUpdate);
}
Ok(num_updated == 1)
}
async fn mark_dependent_queued_requests_as_ready(
&self,
room_id: &RoomId,
parent_txn_id: &TransactionId,
parent_key: SentRequestKey,
) -> Result<usize> {
let room_id = self.encode_key(keys::DEPENDENTS_SEND_QUEUE, room_id);
let parent_key = self.serialize_value(&parent_key)?;
let parent_txn_id = parent_txn_id.to_string();
self.acquire()
.await?
.with_transaction(move |txn| {
Ok(txn.prepare_cached(
"UPDATE dependent_send_queue_events SET parent_key = ? WHERE parent_transaction_id = ? and room_id = ?",
)?
.execute((parent_key, parent_txn_id, room_id))?)
})
.await
}
async fn remove_dependent_queued_request(
&self,
room_id: &RoomId,
txn_id: &ChildTransactionId,
) -> Result<bool> {
let room_id = self.encode_key(keys::DEPENDENTS_SEND_QUEUE, room_id);
let txn_id = txn_id.to_string();
let num_deleted = self
.acquire()
.await?
.with_transaction(move |txn| {
txn.prepare_cached(
"DELETE FROM dependent_send_queue_events WHERE own_transaction_id = ? AND room_id = ?",
)?
.execute((txn_id, room_id))
})
.await?;
Ok(num_deleted > 0)
}
async fn load_dependent_queued_requests(
&self,
room_id: &RoomId,
) -> Result<Vec<DependentQueuedRequest>> {
let room_id = self.encode_key(keys::DEPENDENTS_SEND_QUEUE, room_id);
let res: Vec<(String, String, Option<Vec<u8>>, Vec<u8>, Option<u64>)> = self
.acquire()
.await?
.prepare(
"SELECT own_transaction_id, parent_transaction_id, parent_key, content, created_at FROM dependent_send_queue_events WHERE room_id = ? ORDER BY ROWID",
|mut stmt| {
stmt.query((room_id,))?
.mapped(|row| Ok((row.get(0)?, row.get(1)?, row.get(2)?, row.get(3)?, row.get(4)?)))
.collect()
},
)
.await?;
let mut dependent_events = Vec::with_capacity(res.len());
for entry in res {
let created_at = entry
.4
.and_then(UInt::new)
.map_or_else(MilliSecondsSinceUnixEpoch::now, MilliSecondsSinceUnixEpoch);
dependent_events.push(DependentQueuedRequest {
own_transaction_id: entry.0.into(),
parent_transaction_id: entry.1.into(),
parent_key: entry.2.map(|bytes| self.deserialize_value(&bytes)).transpose()?,
kind: self.deserialize_json(&entry.3)?,
created_at,
});
}
Ok(dependent_events)
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
struct ReceiptData {
receipt: Receipt,
event_id: OwnedEventId,
user_id: OwnedUserId,
}
#[cfg(test)]
mod tests {
use std::sync::atomic::{AtomicU32, Ordering::SeqCst};
use matrix_sdk_base::{statestore_integration_tests, StateStore, StoreError};
use once_cell::sync::Lazy;
use tempfile::{tempdir, TempDir};
use super::SqliteStateStore;
static TMP_DIR: Lazy<TempDir> = Lazy::new(|| tempdir().unwrap());
static NUM: AtomicU32 = AtomicU32::new(0);
async fn get_store() -> Result<impl StateStore, StoreError> {
let name = NUM.fetch_add(1, SeqCst).to_string();
let tmpdir_path = TMP_DIR.path().join(name);
tracing::info!("using store @ {}", tmpdir_path.to_str().unwrap());
Ok(SqliteStateStore::open(tmpdir_path.to_str().unwrap(), None).await.unwrap())
}
statestore_integration_tests!();
}
#[cfg(test)]
mod encrypted_tests {
use std::sync::atomic::{AtomicU32, Ordering::SeqCst};
use matrix_sdk_base::{statestore_integration_tests, StateStore, StoreError};
use once_cell::sync::Lazy;
use tempfile::{tempdir, TempDir};
use super::SqliteStateStore;
static TMP_DIR: Lazy<TempDir> = Lazy::new(|| tempdir().unwrap());
static NUM: AtomicU32 = AtomicU32::new(0);
async fn get_store() -> Result<impl StateStore, StoreError> {
let name = NUM.fetch_add(1, SeqCst).to_string();
let tmpdir_path = TMP_DIR.path().join(name);
tracing::info!("using store @ {}", tmpdir_path.to_str().unwrap());
Ok(SqliteStateStore::open(tmpdir_path.to_str().unwrap(), Some("default_test_password"))
.await
.unwrap())
}
statestore_integration_tests!();
}
#[cfg(test)]
mod migration_tests {
use std::{
path::{Path, PathBuf},
sync::{
atomic::{AtomicU32, Ordering::SeqCst},
Arc,
},
};
use matrix_sdk_base::{
store::{ChildTransactionId, DependentQueuedRequestKind, SerializableEventContent},
sync::UnreadNotificationsCount,
RoomState, StateStore,
};
use matrix_sdk_test::async_test;
use once_cell::sync::Lazy;
use ruma::{
events::{
room::{create::RoomCreateEventContent, message::RoomMessageEventContent},
StateEventType,
},
room_id, server_name, user_id, EventId, MilliSecondsSinceUnixEpoch, RoomId, TransactionId,
UserId,
};
use rusqlite::Transaction;
use serde_json::json;
use tempfile::{tempdir, TempDir};
use super::{create_pool, init, keys, SqliteStateStore};
use crate::{
error::{Error, Result},
utils::{SqliteAsyncConnExt, SqliteKeyValueStoreAsyncConnExt},
};
static TMP_DIR: Lazy<TempDir> = Lazy::new(|| tempdir().unwrap());
static NUM: AtomicU32 = AtomicU32::new(0);
const SECRET: &str = "secret";
fn new_path() -> PathBuf {
let name = NUM.fetch_add(1, SeqCst).to_string();
TMP_DIR.path().join(name)
}
async fn create_fake_db(path: &Path, version: u8) -> Result<SqliteStateStore> {
let pool = create_pool(path).await.unwrap();
let conn = pool.get().await?;
init(&conn).await?;
let store_cipher = Some(Arc::new(conn.get_or_create_store_cipher(SECRET).await.unwrap()));
let this = SqliteStateStore { store_cipher, pool };
this.run_migrations(&conn, 1, Some(version)).await?;
Ok(this)
}
fn room_info_v1_json(
room_id: &RoomId,
state: RoomState,
name: Option<&str>,
creator: Option<&UserId>,
) -> serde_json::Value {
let name_content = match name {
Some(name) => json!({ "name": name }),
None => json!({ "name": null }),
};
let create_content = match creator {
Some(creator) => RoomCreateEventContent::new_v1(creator.to_owned()),
None => RoomCreateEventContent::new_v11(),
};
json!({
"room_id": room_id,
"room_type": state,
"notification_counts": UnreadNotificationsCount::default(),
"summary": {
"heroes": [],
"joined_member_count": 0,
"invited_member_count": 0,
},
"members_synced": false,
"base_info": {
"dm_targets": [],
"max_power_level": 100,
"name": {
"Original": {
"content": name_content,
},
},
"create": {
"Original": {
"content": create_content,
}
}
},
})
}
#[async_test]
pub async fn test_migrating_v1_to_v2() {
let path = new_path();
{
let db = create_fake_db(&path, 1).await.unwrap();
let conn = db.pool.get().await.unwrap();
let this = db.clone();
conn.with_transaction(move |txn| {
for i in 0..5 {
let room_id = RoomId::parse(format!("!room_{i}:localhost")).unwrap();
let (state, stripped) =
if i < 3 { (RoomState::Joined, false) } else { (RoomState::Invited, true) };
let info = room_info_v1_json(&room_id, state, None, None);
let room_id = this.encode_key(keys::ROOM_INFO, room_id);
let data = this.serialize_json(&info)?;
txn.prepare_cached(
"INSERT INTO room_info (room_id, stripped, data)
VALUES (?, ?, ?)",
)?
.execute((room_id, stripped, data))?;
}
Result::<_, Error>::Ok(())
})
.await
.unwrap();
}
let store = SqliteStateStore::open(path, Some(SECRET)).await.unwrap();
assert_eq!(store.get_room_infos().await.unwrap().len(), 5);
}
fn add_room_v2(
this: &SqliteStateStore,
txn: &Transaction<'_>,
room_id: &RoomId,
name: Option<&str>,
create_creator: Option<&UserId>,
create_sender: Option<&UserId>,
) -> Result<(), Error> {
let room_info_json = room_info_v1_json(room_id, RoomState::Joined, name, create_creator);
let encoded_room_id = this.encode_key(keys::ROOM_INFO, room_id);
let encoded_state =
this.encode_key(keys::ROOM_INFO, serde_json::to_string(&RoomState::Joined)?);
let data = this.serialize_json(&room_info_json)?;
txn.prepare_cached(
"INSERT INTO room_info (room_id, state, data)
VALUES (?, ?, ?)",
)?
.execute((encoded_room_id, encoded_state, data))?;
let Some(create_sender) = create_sender else {
return Ok(());
};
let create_content = match create_creator {
Some(creator) => RoomCreateEventContent::new_v1(creator.to_owned()),
None => RoomCreateEventContent::new_v11(),
};
let event_id = EventId::new(server_name!("dummy.local"));
let create_event = json!({
"content": create_content,
"event_id": event_id,
"sender": create_sender.to_owned(),
"origin_server_ts": MilliSecondsSinceUnixEpoch::now(),
"state_key": "",
"type": "m.room.create",
"unsigned": {},
});
let encoded_room_id = this.encode_key(keys::STATE_EVENT, room_id);
let encoded_event_type =
this.encode_key(keys::STATE_EVENT, StateEventType::RoomCreate.to_string());
let encoded_state_key = this.encode_key(keys::STATE_EVENT, "");
let stripped = false;
let encoded_event_id = this.encode_key(keys::STATE_EVENT, event_id);
let data = this.serialize_json(&create_event)?;
txn.prepare_cached(
"INSERT
INTO state_event (room_id, event_type, state_key, stripped, event_id, data)
VALUES (?, ?, ?, ?, ?, ?)",
)?
.execute((
encoded_room_id,
encoded_event_type,
encoded_state_key,
stripped,
encoded_event_id,
data,
))?;
Ok(())
}
#[async_test]
pub async fn test_migrating_v2_to_v3() {
let path = new_path();
let room_a_id = room_id!("!room_a:dummy.local");
let room_a_name = "Room A";
let room_a_creator = user_id!("@creator:dummy.local");
let room_a_create_sender = user_id!("@sender:dummy.local");
let room_b_id = room_id!("!room_b:dummy.local");
let room_c_id = room_id!("!room_c:dummy.local");
let room_c_create_sender = user_id!("@creator:dummy.local");
{
let db = create_fake_db(&path, 2).await.unwrap();
let conn = db.pool.get().await.unwrap();
let this = db.clone();
conn.with_transaction(move |txn| {
add_room_v2(
&this,
txn,
room_a_id,
Some(room_a_name),
Some(room_a_creator),
Some(room_a_create_sender),
)?;
add_room_v2(&this, txn, room_b_id, None, None, None)?;
add_room_v2(&this, txn, room_c_id, None, None, Some(room_c_create_sender))?;
Result::<_, Error>::Ok(())
})
.await
.unwrap();
}
let store = SqliteStateStore::open(path, Some(SECRET)).await.unwrap();
let room_infos = store.get_room_infos().await.unwrap();
assert_eq!(room_infos.len(), 3);
let room_a = room_infos.iter().find(|r| r.room_id() == room_a_id).unwrap();
assert_eq!(room_a.name(), Some(room_a_name));
assert_eq!(room_a.creator(), Some(room_a_create_sender));
let room_b = room_infos.iter().find(|r| r.room_id() == room_b_id).unwrap();
assert_eq!(room_b.name(), None);
assert_eq!(room_b.creator(), None);
let room_c = room_infos.iter().find(|r| r.room_id() == room_c_id).unwrap();
assert_eq!(room_c.name(), None);
assert_eq!(room_c.creator(), Some(room_c_create_sender));
}
#[async_test]
pub async fn test_migrating_v7_to_v9() {
let path = new_path();
let room_id = room_id!("!room_a:dummy.local");
let wedged_event_transaction_id = TransactionId::new();
let local_event_transaction_id = TransactionId::new();
{
let db = create_fake_db(&path, 7).await.unwrap();
let conn = db.pool.get().await.unwrap();
let wedge_tx = wedged_event_transaction_id.clone();
let local_tx = local_event_transaction_id.clone();
conn.with_transaction(move |txn| {
add_dependent_send_queue_event_v7(
&db,
txn,
room_id,
&local_tx,
ChildTransactionId::new(),
DependentQueuedRequestKind::RedactEvent,
)?;
add_send_queue_event_v7(&db, txn, &wedge_tx, room_id, true)?;
add_send_queue_event_v7(&db, txn, &local_tx, room_id, false)?;
Result::<_, Error>::Ok(())
})
.await
.unwrap();
}
let store = SqliteStateStore::open(path, Some(SECRET)).await.unwrap();
let requests = store.load_send_queue_requests(room_id).await.unwrap();
assert!(requests.is_empty());
let dependent_requests = store.load_dependent_queued_requests(room_id).await.unwrap();
assert!(dependent_requests.is_empty());
}
fn add_send_queue_event_v7(
this: &SqliteStateStore,
txn: &Transaction<'_>,
transaction_id: &TransactionId,
room_id: &RoomId,
is_wedged: bool,
) -> Result<(), Error> {
let content =
SerializableEventContent::new(&RoomMessageEventContent::text_plain("Hello").into())?;
let room_id_key = this.encode_key(keys::SEND_QUEUE, room_id);
let room_id_value = this.serialize_value(&room_id.to_owned())?;
let content = this.serialize_json(&content)?;
txn.prepare_cached("INSERT INTO send_queue_events (room_id, room_id_val, transaction_id, content, wedged) VALUES (?, ?, ?, ?, ?)")?
.execute((room_id_key, room_id_value, transaction_id.to_string(), content, is_wedged))?;
Ok(())
}
fn add_dependent_send_queue_event_v7(
this: &SqliteStateStore,
txn: &Transaction<'_>,
room_id: &RoomId,
parent_txn_id: &TransactionId,
own_txn_id: ChildTransactionId,
content: DependentQueuedRequestKind,
) -> Result<(), Error> {
let room_id_value = this.serialize_value(&room_id.to_owned())?;
let parent_txn_id = parent_txn_id.to_string();
let own_txn_id = own_txn_id.to_string();
let content = this.serialize_json(&content)?;
txn.prepare_cached(
"INSERT INTO dependent_send_queue_events
(room_id, parent_transaction_id, own_transaction_id, content)
VALUES (?, ?, ?, ?)",
)?
.execute((room_id_value, parent_txn_id, own_txn_id, content))?;
Ok(())
}
}