mirror of
https://forgejo.ellis.link/continuwuation/continuwuity.git
synced 2025-07-14 18:46:32 +02:00
269 lines
8 KiB
Rust
269 lines
8 KiB
Rust
mod append;
|
|
mod backfill;
|
|
mod build;
|
|
mod create;
|
|
mod data;
|
|
mod redact;
|
|
|
|
use std::{fmt::Write, sync::Arc};
|
|
|
|
use async_trait::async_trait;
|
|
pub use conduwuit_core::matrix::pdu::{PduId, RawPduId};
|
|
use conduwuit_core::{
|
|
Result, Server, at, err,
|
|
matrix::{
|
|
event::Event,
|
|
pdu::{PduCount, PduEvent},
|
|
},
|
|
utils::{MutexMap, MutexMapGuard, future::TryExtExt, stream::TryIgnore},
|
|
warn,
|
|
};
|
|
use futures::{Future, Stream, TryStreamExt, pin_mut};
|
|
use ruma::{
|
|
CanonicalJsonObject, EventId, OwnedEventId, OwnedRoomId, RoomId, UserId,
|
|
events::room::encrypted::Relation,
|
|
};
|
|
use serde::Deserialize;
|
|
|
|
use self::data::Data;
|
|
pub use self::data::PdusIterItem;
|
|
use crate::{
|
|
Dep, account_data, admin, appservice, globals, pusher, rooms, sending, server_keys, users,
|
|
};
|
|
|
|
// Update Relationships
|
|
#[derive(Deserialize)]
|
|
struct ExtractRelatesTo {
|
|
#[serde(rename = "m.relates_to")]
|
|
relates_to: Relation,
|
|
}
|
|
|
|
#[derive(Clone, Debug, Deserialize)]
|
|
struct ExtractEventId {
|
|
event_id: OwnedEventId,
|
|
}
|
|
#[derive(Clone, Debug, Deserialize)]
|
|
struct ExtractRelatesToEventId {
|
|
#[serde(rename = "m.relates_to")]
|
|
relates_to: ExtractEventId,
|
|
}
|
|
|
|
#[derive(Deserialize)]
|
|
struct ExtractBody {
|
|
body: Option<String>,
|
|
}
|
|
|
|
pub struct Service {
|
|
services: Services,
|
|
db: Data,
|
|
pub mutex_insert: RoomMutexMap,
|
|
}
|
|
|
|
struct Services {
|
|
server: Arc<Server>,
|
|
account_data: Dep<account_data::Service>,
|
|
appservice: Dep<appservice::Service>,
|
|
admin: Dep<admin::Service>,
|
|
alias: Dep<rooms::alias::Service>,
|
|
globals: Dep<globals::Service>,
|
|
short: Dep<rooms::short::Service>,
|
|
state: Dep<rooms::state::Service>,
|
|
state_cache: Dep<rooms::state_cache::Service>,
|
|
state_accessor: Dep<rooms::state_accessor::Service>,
|
|
pdu_metadata: Dep<rooms::pdu_metadata::Service>,
|
|
read_receipt: Dep<rooms::read_receipt::Service>,
|
|
sending: Dep<sending::Service>,
|
|
server_keys: Dep<server_keys::Service>,
|
|
user: Dep<rooms::user::Service>,
|
|
users: Dep<users::Service>,
|
|
pusher: Dep<pusher::Service>,
|
|
threads: Dep<rooms::threads::Service>,
|
|
search: Dep<rooms::search::Service>,
|
|
spaces: Dep<rooms::spaces::Service>,
|
|
event_handler: Dep<rooms::event_handler::Service>,
|
|
}
|
|
|
|
type RoomMutexMap = MutexMap<OwnedRoomId, ()>;
|
|
pub type RoomMutexGuard = MutexMapGuard<OwnedRoomId, ()>;
|
|
|
|
#[async_trait]
|
|
impl crate::Service for Service {
|
|
fn build(args: crate::Args<'_>) -> Result<Arc<Self>> {
|
|
Ok(Arc::new(Self {
|
|
services: Services {
|
|
server: args.server.clone(),
|
|
account_data: args.depend::<account_data::Service>("account_data"),
|
|
appservice: args.depend::<appservice::Service>("appservice"),
|
|
admin: args.depend::<admin::Service>("admin"),
|
|
alias: args.depend::<rooms::alias::Service>("rooms::alias"),
|
|
globals: args.depend::<globals::Service>("globals"),
|
|
short: args.depend::<rooms::short::Service>("rooms::short"),
|
|
state: args.depend::<rooms::state::Service>("rooms::state"),
|
|
state_cache: args.depend::<rooms::state_cache::Service>("rooms::state_cache"),
|
|
state_accessor: args
|
|
.depend::<rooms::state_accessor::Service>("rooms::state_accessor"),
|
|
pdu_metadata: args.depend::<rooms::pdu_metadata::Service>("rooms::pdu_metadata"),
|
|
read_receipt: args.depend::<rooms::read_receipt::Service>("rooms::read_receipt"),
|
|
sending: args.depend::<sending::Service>("sending"),
|
|
server_keys: args.depend::<server_keys::Service>("server_keys"),
|
|
user: args.depend::<rooms::user::Service>("rooms::user"),
|
|
users: args.depend::<users::Service>("users"),
|
|
pusher: args.depend::<pusher::Service>("pusher"),
|
|
threads: args.depend::<rooms::threads::Service>("rooms::threads"),
|
|
search: args.depend::<rooms::search::Service>("rooms::search"),
|
|
spaces: args.depend::<rooms::spaces::Service>("rooms::spaces"),
|
|
event_handler: args
|
|
.depend::<rooms::event_handler::Service>("rooms::event_handler"),
|
|
},
|
|
db: Data::new(&args),
|
|
mutex_insert: RoomMutexMap::new(),
|
|
}))
|
|
}
|
|
|
|
async fn memory_usage(&self, out: &mut (dyn Write + Send)) -> Result {
|
|
let mutex_insert = self.mutex_insert.len();
|
|
writeln!(out, "insert_mutex: {mutex_insert}")?;
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn name(&self) -> &str { crate::service::make_name(std::module_path!()) }
|
|
}
|
|
|
|
impl Service {
|
|
#[tracing::instrument(skip(self), level = "debug")]
|
|
pub async fn first_pdu_in_room(&self, room_id: &RoomId) -> Result<impl Event> {
|
|
self.first_item_in_room(room_id).await.map(at!(1))
|
|
}
|
|
|
|
#[tracing::instrument(skip(self), level = "debug")]
|
|
pub async fn first_item_in_room(&self, room_id: &RoomId) -> Result<(PduCount, impl Event)> {
|
|
let pdus = self.pdus(None, room_id, None);
|
|
|
|
pin_mut!(pdus);
|
|
pdus.try_next()
|
|
.await?
|
|
.ok_or_else(|| err!(Request(NotFound("No PDU found in room"))))
|
|
}
|
|
|
|
#[tracing::instrument(skip(self), level = "debug")]
|
|
pub async fn latest_pdu_in_room(&self, room_id: &RoomId) -> Result<impl Event> {
|
|
self.db.latest_pdu_in_room(None, room_id).await
|
|
}
|
|
|
|
#[tracing::instrument(skip(self), level = "debug")]
|
|
pub async fn last_timeline_count(
|
|
&self,
|
|
sender_user: Option<&UserId>,
|
|
room_id: &RoomId,
|
|
) -> Result<PduCount> {
|
|
self.db.last_timeline_count(sender_user, room_id).await
|
|
}
|
|
|
|
/// Returns the `count` of this pdu's id.
|
|
pub async fn get_pdu_count(&self, event_id: &EventId) -> Result<PduCount> {
|
|
self.db.get_pdu_count(event_id).await
|
|
}
|
|
|
|
/// Returns the json of a pdu.
|
|
pub async fn get_pdu_json(&self, event_id: &EventId) -> Result<CanonicalJsonObject> {
|
|
self.db.get_pdu_json(event_id).await
|
|
}
|
|
|
|
/// Returns the json of a pdu.
|
|
#[inline]
|
|
pub async fn get_non_outlier_pdu_json(
|
|
&self,
|
|
event_id: &EventId,
|
|
) -> Result<CanonicalJsonObject> {
|
|
self.db.get_non_outlier_pdu_json(event_id).await
|
|
}
|
|
|
|
/// Returns the pdu's id.
|
|
#[inline]
|
|
pub async fn get_pdu_id(&self, event_id: &EventId) -> Result<RawPduId> {
|
|
self.db.get_pdu_id(event_id).await
|
|
}
|
|
|
|
/// Returns the pdu.
|
|
///
|
|
/// Checks the `eventid_outlierpdu` Tree if not found in the timeline.
|
|
#[inline]
|
|
pub async fn get_non_outlier_pdu(&self, event_id: &EventId) -> Result<impl Event> {
|
|
self.db.get_non_outlier_pdu(event_id).await
|
|
}
|
|
|
|
/// Returns the pdu.
|
|
///
|
|
/// Checks the `eventid_outlierpdu` Tree if not found in the timeline.
|
|
#[inline]
|
|
pub async fn get_pdu(&self, event_id: &EventId) -> Result<PduEvent> {
|
|
self.db.get_pdu(event_id).await
|
|
}
|
|
|
|
/// Returns the pdu.
|
|
///
|
|
/// This does __NOT__ check the outliers `Tree`.
|
|
#[inline]
|
|
pub async fn get_pdu_from_id(&self, pdu_id: &RawPduId) -> Result<PduEvent> {
|
|
self.db.get_pdu_from_id(pdu_id).await
|
|
}
|
|
|
|
/// Returns the pdu as a `BTreeMap<String, CanonicalJsonValue>`.
|
|
#[inline]
|
|
pub async fn get_pdu_json_from_id(&self, pdu_id: &RawPduId) -> Result<CanonicalJsonObject> {
|
|
self.db.get_pdu_json_from_id(pdu_id).await
|
|
}
|
|
|
|
/// Checks if pdu exists
|
|
///
|
|
/// Checks the `eventid_outlierpdu` Tree if not found in the timeline.
|
|
#[inline]
|
|
pub fn pdu_exists<'a>(
|
|
&'a self,
|
|
event_id: &'a EventId,
|
|
) -> impl Future<Output = bool> + Send + 'a {
|
|
self.db.pdu_exists(event_id).is_ok()
|
|
}
|
|
|
|
/// Removes a pdu and creates a new one with the same id.
|
|
#[tracing::instrument(skip(self), level = "debug")]
|
|
pub async fn replace_pdu(&self, pdu_id: &RawPduId, pdu_json: &CanonicalJsonObject) -> Result {
|
|
self.db.replace_pdu(pdu_id, pdu_json).await
|
|
}
|
|
|
|
/// Returns an iterator over all PDUs in a room. Unknown rooms produce no
|
|
/// items.
|
|
#[inline]
|
|
pub fn all_pdus<'a>(
|
|
&'a self,
|
|
user_id: &'a UserId,
|
|
room_id: &'a RoomId,
|
|
) -> impl Stream<Item = PdusIterItem> + Send + 'a {
|
|
self.pdus(Some(user_id), room_id, None).ignore_err()
|
|
}
|
|
|
|
/// Reverse iteration starting at from.
|
|
#[tracing::instrument(skip(self), level = "debug")]
|
|
pub fn pdus_rev<'a>(
|
|
&'a self,
|
|
user_id: Option<&'a UserId>,
|
|
room_id: &'a RoomId,
|
|
until: Option<PduCount>,
|
|
) -> impl Stream<Item = Result<PdusIterItem>> + Send + 'a {
|
|
self.db
|
|
.pdus_rev(user_id, room_id, until.unwrap_or_else(PduCount::max))
|
|
}
|
|
|
|
/// Forward iteration starting at from.
|
|
#[tracing::instrument(skip(self), level = "debug")]
|
|
pub fn pdus<'a>(
|
|
&'a self,
|
|
user_id: Option<&'a UserId>,
|
|
room_id: &'a RoomId,
|
|
from: Option<PduCount>,
|
|
) -> impl Stream<Item = Result<PdusIterItem>> + Send + 'a {
|
|
self.db
|
|
.pdus(user_id, room_id, from.unwrap_or_else(PduCount::min))
|
|
}
|
|
}
|