make worker a struct for readability

This commit is contained in:
phiresky 2023-08-03 12:58:46 +00:00
parent 7f82bd07fe
commit 7e72ad87fe
2 changed files with 212 additions and 146 deletions

View file

@ -1,11 +1,12 @@
use crate::{ use crate::{
util::{retry_sleep_duration, CancellableTask}, util::{retry_sleep_duration, CancellableTask},
worker::instance_worker, worker::InstanceWorker,
}; };
use activitypub_federation::config::FederationConfig; use activitypub_federation::config::FederationConfig;
use chrono::{Local, Timelike}; use chrono::{Local, Timelike};
use clap::Parser; use clap::Parser;
use federation_queue_state::FederationQueueState; use federation_queue_state::FederationQueueState;
use lemmy_api_common::context::LemmyContext;
use lemmy_db_schema::{ use lemmy_db_schema::{
source::instance::Instance, source::instance::Instance,
utils::{ActualDbPool, DbPool}, utils::{ActualDbPool, DbPool},
@ -36,10 +37,10 @@ pub struct Opts {
pub process_index: i32, pub process_index: i32,
} }
async fn start_stop_federation_workers<T: Clone + Send + Sync + 'static>( async fn start_stop_federation_workers(
opts: Opts, opts: Opts,
pool: ActualDbPool, pool: ActualDbPool,
federation_config: FederationConfig<T>, federation_config: FederationConfig<LemmyContext>,
cancel: CancellationToken, cancel: CancellationToken,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
let mut workers = HashMap::new(); let mut workers = HashMap::new();
@ -68,16 +69,20 @@ async fn start_stop_federation_workers<T: Clone + Send + Sync + 'static>(
let should_federate = allowed && !is_dead; let should_federate = allowed && !is_dead;
if !workers.contains_key(&instance.id) && should_federate { if !workers.contains_key(&instance.id) && should_federate {
let stats_sender = stats_sender.clone(); let stats_sender = stats_sender.clone();
let context = federation_config.to_request_data();
let pool = pool.clone();
workers.insert( workers.insert(
instance.id, instance.id,
CancellableTask::spawn(WORKER_EXIT_TIMEOUT, |stop| { CancellableTask::spawn(WORKER_EXIT_TIMEOUT, |stop| async move {
instance_worker( InstanceWorker::init_and_loop(
pool.clone(),
instance, instance,
federation_config.to_request_data(), context,
&mut DbPool::Pool(&pool),
stop, stop,
stats_sender, stats_sender,
) )
.await?;
Ok(())
}), }),
); );
} else if !should_federate { } else if !should_federate {
@ -112,7 +117,7 @@ async fn start_stop_federation_workers<T: Clone + Send + Sync + 'static>(
pub fn start_stop_federation_workers_cancellable( pub fn start_stop_federation_workers_cancellable(
opts: Opts, opts: Opts,
pool: ActualDbPool, pool: ActualDbPool,
config: FederationConfig<impl Clone + Send + Sync + 'static>, config: FederationConfig<LemmyContext>,
) -> CancellableTask<()> { ) -> CancellableTask<()> {
CancellableTask::spawn(WORKER_EXIT_TIMEOUT, move |c| { CancellableTask::spawn(WORKER_EXIT_TIMEOUT, move |c| {
start_stop_federation_workers(opts, pool, config, c) start_stop_federation_workers(opts, pool, config, c)

View file

@ -8,6 +8,8 @@ use activitypub_federation::{
}; };
use anyhow::Result; use anyhow::Result;
use chrono::{DateTime, TimeZone, Utc}; use chrono::{DateTime, TimeZone, Utc};
use lemmy_api_common::context::LemmyContext;
use lemmy_apub::activity_lists::SharedInboxActivities;
use lemmy_db_schema::{ use lemmy_db_schema::{
newtypes::{CommunityId, InstanceId}, newtypes::{CommunityId, InstanceId},
source::{activity::SentActivity, instance::Instance, site::Site}, source::{activity::SentActivity, instance::Instance, site::Site},
@ -26,124 +28,162 @@ use tokio_util::sync::CancellationToken;
static CHECK_SAVE_STATE_EVERY_IT: i64 = 100; static CHECK_SAVE_STATE_EVERY_IT: i64 = 100;
static SAVE_STATE_EVERY_TIME: Duration = Duration::from_secs(10); static SAVE_STATE_EVERY_TIME: Duration = Duration::from_secs(10);
/// loop fetch new activities from db and send them to the inboxes of the given instances pub(crate) struct InstanceWorker {
/// this worker only returns if (a) there is an internal error or (b) the cancellation token is cancelled (graceful exit)
pub async fn instance_worker(
pool: ActualDbPool,
instance: Instance, instance: Instance,
data: Data<impl Clone>, site: Option<Site>,
followed_communities: HashMap<CommunityId, HashSet<Url>>,
stop: CancellationToken,
context: Data<LemmyContext>,
stats_sender: UnboundedSender<FederationQueueState>,
last_full_communities_fetch: DateTime<Utc>,
last_incremental_communities_fetch: DateTime<Utc>,
state: FederationQueueState,
last_state_insert: DateTime<Utc>,
}
impl InstanceWorker {
pub(crate) async fn init_and_loop(
instance: Instance,
context: Data<LemmyContext>,
pool: &mut DbPool<'_>, // in theory there's a ref to the pool in context, but i couldn't get that to work wrt lifetimes
stop: CancellationToken, stop: CancellationToken,
stats_sender: UnboundedSender<FederationQueueState>, stats_sender: UnboundedSender<FederationQueueState>,
) -> Result<(), anyhow::Error> { ) -> Result<(), anyhow::Error> {
let pool = &mut DbPool::Pool(&pool);
let mut last_full_communities_fetch = Utc.timestamp_nanos(0);
let mut last_incremental_communities_fetch = Utc.timestamp_nanos(0);
let mut last_state_insert = Utc.timestamp_nanos(0);
let mut followed_communities: HashMap<CommunityId, HashSet<Url>> =
get_communities(pool, instance.id, &mut last_incremental_communities_fetch).await?;
let site = Site::read_from_instance_id(pool, instance.id).await?; let site = Site::read_from_instance_id(pool, instance.id).await?;
let state = FederationQueueState::load(pool, &instance.domain).await?;
let mut worker = InstanceWorker {
instance,
site,
followed_communities: HashMap::new(),
stop,
context,
stats_sender,
last_full_communities_fetch: Utc.timestamp_nanos(0),
last_incremental_communities_fetch: Utc.timestamp_nanos(0),
state,
last_state_insert: Utc.timestamp_nanos(0),
};
worker.loop_until_stopped(pool).await
}
/// loop fetch new activities from db and send them to the inboxes of the given instances
/// this worker only returns if (a) there is an internal error or (b) the cancellation token is cancelled (graceful exit)
pub(crate) async fn loop_until_stopped(
&mut self,
pool: &mut DbPool<'_>,
) -> Result<(), anyhow::Error> {
self.update_communities(pool).await;
self.initial_fail_sleep().await;
while !self.stop.is_cancelled() {
self.loop_batch(pool).await?;
if self.stop.is_cancelled() {
break;
}
if Utc::now() - self.last_state_insert
> chrono::Duration::from_std(SAVE_STATE_EVERY_TIME).expect("not negative")
{
self.last_state_insert = Utc::now();
FederationQueueState::upsert(pool, &self.state).await?;
self.stats_sender.send(self.state.clone())?;
}
self.update_communities(pool).await;
}
// final update of state in db
FederationQueueState::upsert(pool, &self.state).await?;
Ok(())
}
let mut state = FederationQueueState::load(pool, &instance.domain).await?; async fn initial_fail_sleep(&mut self) -> Result<()> {
if state.fail_count > 0 { // before starting queue, sleep remaining duration if last request failed
// before starting queue, sleep remaining duration if self.state.fail_count > 0 {
let elapsed = (Utc::now() - state.last_retry).to_std()?; let elapsed = (Utc::now() - self.state.last_retry).to_std()?;
let remaining = retry_sleep_duration(state.fail_count) - elapsed; let remaining = retry_sleep_duration(self.state.fail_count) - elapsed;
tokio::select! { tokio::select! {
() = sleep(remaining) => {}, () = sleep(remaining) => {},
() = stop.cancelled() => { return Ok(()); } () = self.stop.cancelled() => {}
} }
} }
while !stop.is_cancelled() { Ok(())
}
async fn loop_batch(&mut self, pool: &mut DbPool<'_>) -> Result<()> {
let latest_id = get_latest_activity_id(pool).await?; let latest_id = get_latest_activity_id(pool).await?;
let mut id = state.last_successful_id; let mut id = self.state.last_successful_id;
if id == latest_id { if id == latest_id {
// no more work to be done, wait before rechecking // no more work to be done, wait before rechecking
tokio::select! { tokio::select! {
() = sleep(Duration::from_secs(10)) => { continue; }, () = sleep(Duration::from_secs(10)) => {},
() = stop.cancelled() => { return Ok(()); } () = self.stop.cancelled() => {}
} }
return Ok(());
} }
let mut processed_activities = 0; let mut processed_activities = 0;
'batch: while id < latest_id while id < latest_id
&& processed_activities < CHECK_SAVE_STATE_EVERY_IT && processed_activities < CHECK_SAVE_STATE_EVERY_IT
&& !stop.is_cancelled() && !self.stop.is_cancelled()
{ {
id += 1; id += 1;
processed_activities += 1; processed_activities += 1;
let Some(ele) = get_activity_cached(pool, id).await? else { let Some(ele) = get_activity_cached(pool, id).await? else {
state.last_successful_id = id; self.state.last_successful_id = id;
continue; continue;
}; };
let (activity, object) = (&ele.0, &ele.1); self.send_retry_loop(pool, &ele.0, &ele.1).await?;
let inbox_urls = get_inbox_urls(&instance, &site, &followed_communities, activity); if self.stop.is_cancelled() {
return Ok(());
}
// send success!
self.state.last_successful_id = id;
self.state.fail_count = 0;
}
Ok(())
}
/** this function will only return if (a) send succeeded or (b) worker cancelled */
async fn send_retry_loop(
&mut self,
pool: &mut DbPool<'_>,
activity: &SentActivity,
object: &SharedInboxActivities,
) -> Result<()> {
let inbox_urls = self.get_inbox_urls(activity);
if inbox_urls.is_empty() { if inbox_urls.is_empty() {
state.last_successful_id = id; self.state.last_successful_id = activity.id;
continue; return Ok(());
} }
let Some(actor_apub_id) = &activity.actor_apub_id else { let Some(actor_apub_id) = &activity.actor_apub_id else {
continue; // activity was inserted before persistent queue was activated return Ok(()); // activity was inserted before persistent queue was activated
}; };
let actor = get_actor_cached(pool, activity.actor_type, actor_apub_id).await?; let actor = get_actor_cached(pool, activity.actor_type, actor_apub_id).await?;
let inbox_urls = inbox_urls.into_iter().collect(); let inbox_urls = inbox_urls.into_iter().collect();
let requests = prepare_raw(object, actor.as_ref(), inbox_urls, &data) let requests = prepare_raw(object, actor.as_ref(), inbox_urls, &self.context)
.await .await
.into_anyhow()?; .into_anyhow()?;
for task in requests { for task in requests {
// usually only one due to shared inbox // usually only one due to shared inbox
let mut req = sign_raw(&task, &data, REQWEST_TIMEOUT).await?; let mut req = sign_raw(&task, &self.context, REQWEST_TIMEOUT).await?;
tracing::info!("sending out {}", task); tracing::info!("sending out {}", task);
while let Err(e) = send_raw(&task, &data, req).await { while let Err(e) = send_raw(&task, &self.context, req).await {
state.fail_count += 1; self.state.fail_count += 1;
state.last_retry = Utc::now(); self.state.last_retry = Utc::now();
let retry_delay: Duration = retry_sleep_duration(state.fail_count); let retry_delay: Duration = retry_sleep_duration(self.state.fail_count);
tracing::info!( tracing::info!(
"{}: retrying {id} attempt {} with delay {retry_delay:.2?}. ({e})", "{}: retrying {} attempt {} with delay {retry_delay:.2?}. ({e})",
instance.domain, self.instance.domain,
state.fail_count activity.id,
self.state.fail_count
); );
stats_sender.send(state.clone())?; self.stats_sender.send(self.state.clone())?;
FederationQueueState::upsert(pool, &state).await?; FederationQueueState::upsert(pool, &self.state).await?;
req = sign_raw(&task, &data, REQWEST_TIMEOUT).await?; // resign request req = sign_raw(&task, &self.context, REQWEST_TIMEOUT).await?; // resign request
tokio::select! { tokio::select! {
() = sleep(retry_delay) => {}, () = sleep(retry_delay) => {},
() = stop.cancelled() => { () = self.stop.cancelled() => {
// save state to db and exit // save state to db and exit
break 'batch; return Ok(());
} }
} }
} }
} }
// send success!
state.last_successful_id = id;
state.fail_count = 0;
}
if Utc::now() - last_state_insert
> chrono::Duration::from_std(SAVE_STATE_EVERY_TIME).expect("not negative")
{
last_state_insert = Utc::now();
FederationQueueState::upsert(pool, &state).await?;
stats_sender.send(state.clone())?;
}
{
// update communities
if (Utc::now() - last_incremental_communities_fetch) > chrono::Duration::seconds(10) {
// process additions every 10s
followed_communities.extend(
get_communities(pool, instance.id, &mut last_incremental_communities_fetch).await?,
);
}
if (Utc::now() - last_full_communities_fetch) > chrono::Duration::seconds(300) {
// process removals every 5min
last_full_communities_fetch = Utc.timestamp_nanos(0);
followed_communities =
get_communities(pool, instance.id, &mut last_full_communities_fetch).await?;
last_incremental_communities_fetch = last_full_communities_fetch;
}
}
}
Ok(()) Ok(())
} }
@ -151,27 +191,22 @@ pub async fn instance_worker(
/// most often this will return 0 values (if instance doesn't care about the activity) /// most often this will return 0 values (if instance doesn't care about the activity)
/// or 1 value (the shared inbox) /// or 1 value (the shared inbox)
/// > 1 values only happens for non-lemmy software /// > 1 values only happens for non-lemmy software
fn get_inbox_urls( fn get_inbox_urls(&self, activity: &SentActivity) -> HashSet<Url> {
instance: &Instance,
site: &Option<Site>,
followed_communities: &HashMap<CommunityId, HashSet<Url>>,
activity: &SentActivity,
) -> HashSet<Url> {
let mut inbox_urls: HashSet<Url> = HashSet::new(); let mut inbox_urls: HashSet<Url> = HashSet::new();
if activity.send_all_instances { if activity.send_all_instances {
if let Some(site) = &site { if let Some(site) = &self.site {
// Nutomic: Most non-lemmy software wont have a site row. That means it cant handle these activities. So handling it like this is fine. // Nutomic: Most non-lemmy software wont have a site row. That means it cant handle these activities. So handling it like this is fine.
inbox_urls.insert(site.inbox_url.inner().clone()); inbox_urls.insert(site.inbox_url.inner().clone());
} }
} }
for t in &activity.send_community_followers_of { for t in &activity.send_community_followers_of {
if let Some(urls) = followed_communities.get(t) { if let Some(urls) = self.followed_communities.get(t) {
inbox_urls.extend(urls.iter().map(std::clone::Clone::clone)); inbox_urls.extend(urls.iter().map(std::clone::Clone::clone));
} }
} }
for inbox in &activity.send_inboxes { for inbox in &activity.send_inboxes {
if inbox.domain() != Some(&instance.domain) { if inbox.domain() != Some(&self.instance.domain) {
continue; continue;
} }
inbox_urls.insert(inbox.inner().clone()); inbox_urls.insert(inbox.inner().clone());
@ -179,21 +214,47 @@ fn get_inbox_urls(
inbox_urls inbox_urls
} }
async fn update_communities(&mut self, pool: &mut DbPool<'_>) -> Result<()> {
if (Utc::now() - self.last_full_communities_fetch) > chrono::Duration::seconds(300) {
// process removals every 5min
self.last_full_communities_fetch = Utc.timestamp_nanos(0);
(self.followed_communities, self.last_full_communities_fetch) = self
.get_communities(pool, self.instance.id, self.last_full_communities_fetch)
.await?;
self.last_incremental_communities_fetch = self.last_full_communities_fetch;
}
if (Utc::now() - self.last_incremental_communities_fetch) > chrono::Duration::seconds(10) {
let (news, time) = self
.get_communities(
pool,
self.instance.id,
self.last_incremental_communities_fetch,
)
.await?;
// process additions every 10s
self.followed_communities.extend(news);
self.last_incremental_communities_fetch = time;
}
Ok(())
}
/// get a list of local communities with the remote inboxes on the given instance that cares about them /// get a list of local communities with the remote inboxes on the given instance that cares about them
async fn get_communities( async fn get_communities(
&mut self,
pool: &mut DbPool<'_>, pool: &mut DbPool<'_>,
instance_id: InstanceId, instance_id: InstanceId,
last_fetch: &mut DateTime<Utc>, last_fetch: DateTime<Utc>,
) -> Result<HashMap<CommunityId, HashSet<Url>>> { ) -> Result<(HashMap<CommunityId, HashSet<Url>>, DateTime<Utc>)> {
let e = *last_fetch; let new_last_fetch = Utc::now(); // update to time before fetch to ensure overlap
*last_fetch = Utc::now(); // update to time before fetch to ensure overlap Ok((
Ok( CommunityFollowerView::get_instance_followed_community_inboxes(pool, instance_id, last_fetch)
CommunityFollowerView::get_instance_followed_community_inboxes(pool, instance_id, e)
.await? .await?
.into_iter() .into_iter()
.fold(HashMap::new(), |mut map, (c, u)| { .fold(HashMap::new(), |mut map, (c, u)| {
map.entry(c).or_insert_with(HashSet::new).insert(u.into()); map.entry(c).or_insert_with(HashSet::new).insert(u.into());
map map
}), }),
) new_last_fetch,
))
}
} }