use anyhow::{anyhow, Context, Result}; use diesel::{ prelude::*, sql_types::{Bool, Int8}, }; use diesel_async::RunQueryDsl; use lemmy_apub::{ activity_lists::SharedInboxActivities, fetcher::{site_or_community_or_user::SiteOrCommunityOrUser, user_or_community::UserOrCommunity}, }; use lemmy_db_schema::{ source::{ activity::{ActorType, SentActivity}, community::Community, person::Person, site::Site, }, traits::ApubActor, utils::{get_conn, DbPool}, }; use moka::future::Cache; use once_cell::sync::Lazy; use reqwest::Url; use serde_json::Value; use std::{ future::Future, pin::Pin, sync::{Arc, RwLock}, time::Duration, }; use tokio::{task::JoinHandle, time::sleep}; use tokio_util::sync::CancellationToken; /// Decrease the delays of the federation queue. /// Should only be used for federation tests since it significantly increases CPU and DB load of the federation queue. pub(crate) static LEMMY_TEST_FAST_FEDERATION: Lazy = Lazy::new(|| { std::env::var("LEMMY_TEST_FAST_FEDERATION") .map(|s| !s.is_empty()) .unwrap_or(false) }); /// Recheck for new federation work every n seconds. /// /// When the queue is processed faster than new activities are added and it reaches the current time with an empty batch, /// this is the delay the queue waits before it checks if new activities have been added to the sent_activities table. /// This delay is only applied if no federated activity happens during sending activities of the last batch. pub(crate) static WORK_FINISHED_RECHECK_DELAY: Lazy = Lazy::new(|| { if *LEMMY_TEST_FAST_FEDERATION { Duration::from_millis(100) } else { Duration::from_secs(30) } }); pub struct CancellableTask { f: Pin> + Send + 'static>>, ended: Arc>, } impl CancellableTask { /// spawn a task but with graceful shutdown pub fn spawn( timeout: Duration, task: impl FnOnce(CancellationToken) -> F, ) -> CancellableTask where F: Future> + Send + 'static, { let stop = CancellationToken::new(); let task = task(stop.clone()); let ended = Arc::new(RwLock::new(false)); let ended_write = ended.clone(); let task: JoinHandle> = tokio::spawn(async move { match task.await { Ok(o) => Ok(o), Err(e) => { *ended_write.write().expect("poisoned") = true; Err(e) } } }); let abort = task.abort_handle(); CancellableTask { ended, f: Box::pin(async move { stop.cancel(); tokio::select! { r = task => { Ok(r.context("could not join")??) }, _ = sleep(timeout) => { abort.abort(); tracing::warn!("Graceful shutdown timed out, aborting task"); Err(anyhow!("task aborted due to timeout")) } } }), } } /// cancel the cancel signal, wait for timeout for the task to stop gracefully, otherwise abort it pub async fn cancel(self) -> Result { self.f.await } pub fn has_ended(&self) -> bool { *self.ended.read().expect("poisoned") } } /// assuming apub priv key and ids are immutable, then we don't need to have TTL /// TODO: capacity should be configurable maybe based on memory use pub(crate) async fn get_actor_cached( pool: &mut DbPool<'_>, actor_type: ActorType, actor_apub_id: &Url, ) -> Result> { static CACHE: Lazy>> = Lazy::new(|| Cache::builder().max_capacity(10000).build()); CACHE .try_get_with(actor_apub_id.clone(), async { let url = actor_apub_id.clone().into(); let person = match actor_type { ActorType::Site => SiteOrCommunityOrUser::Site( Site::read_from_apub_id(pool, &url) .await? .context("apub site not found")? .into(), ), ActorType::Community => SiteOrCommunityOrUser::UserOrCommunity(UserOrCommunity::Community( Community::read_from_apub_id(pool, &url) .await? .context("apub community not found")? .into(), )), ActorType::Person => SiteOrCommunityOrUser::UserOrCommunity(UserOrCommunity::User( Person::read_from_apub_id(pool, &url) .await? .context("apub person not found")? .into(), )), }; Result::<_, anyhow::Error>::Ok(Arc::new(person)) }) .await .map_err(|e| anyhow::anyhow!("err getting actor {actor_type:?} {actor_apub_id}: {e:?}")) } /// this should maybe be a newtype like all the other PersonId CommunityId etc. pub(crate) type ActivityId = i64; type CachedActivityInfo = Option>; /// activities are immutable so cache does not need to have TTL /// May return None if the corresponding id does not exist or is a received activity. /// Holes in serials are expected behaviour in postgresql /// todo: cache size should probably be configurable / dependent on desired memory usage pub(crate) async fn get_activity_cached( pool: &mut DbPool<'_>, activity_id: ActivityId, ) -> Result { static ACTIVITIES: Lazy> = Lazy::new(|| Cache::builder().max_capacity(10000).build()); ACTIVITIES .try_get_with(activity_id, async { let row = SentActivity::read(pool, activity_id) .await .optional() .context("could not read activity")?; let Some(mut row) = row else { return anyhow::Result::<_, anyhow::Error>::Ok(None); }; // swap to avoid cloning let mut data = Value::Null; std::mem::swap(&mut row.data, &mut data); let activity_actual: SharedInboxActivities = serde_json::from_value(data)?; Ok(Some(Arc::new((row, activity_actual)))) }) .await .map_err(|e| anyhow::anyhow!("err getting activity: {e:?}")) } /// return the most current activity id (with 1 second cache) pub(crate) async fn get_latest_activity_id(pool: &mut DbPool<'_>) -> Result { static CACHE: Lazy> = Lazy::new(|| { Cache::builder() .time_to_live(if *LEMMY_TEST_FAST_FEDERATION { *WORK_FINISHED_RECHECK_DELAY } else { Duration::from_secs(1) }) .build() }); CACHE .try_get_with((), async { let conn = &mut get_conn(pool).await?; let seq: Sequence = diesel::sql_query("select last_value, is_called from sent_activity_id_seq") .get_result(conn) .await?; let latest_id = if seq.is_called { seq.last_value as ActivityId } else { // if a PG sequence has never been used, last_value will actually be next_value (seq.last_value - 1) as ActivityId }; anyhow::Result::<_, anyhow::Error>::Ok(latest_id as ActivityId) }) .await .map_err(|e| anyhow::anyhow!("err getting id: {e:?}")) } /// how long to sleep based on how many retries have already happened pub(crate) fn retry_sleep_duration(retry_count: i32) -> Duration { Duration::from_secs_f64(10.0 * 2.0_f64.powf(f64::from(retry_count))) } #[derive(QueryableByName)] struct Sequence { #[diesel(sql_type = Int8)] last_value: i64, // this value is bigint for some reason even if sequence is int4 #[diesel(sql_type = Bool)] is_called: bool, }