use crate::{ apub::AcceptedActivities, config::UrlKind, db::Actor, error::Error, future::BoxFuture, jobs::{apub::generate_undo_follow, Deliver, JobState}, }; use activitystreams::prelude::BaseExt; use background_jobs::Job; #[derive(Clone, serde::Deserialize, serde::Serialize)] pub(crate) struct Undo { input: AcceptedActivities, actor: Actor, } impl std::fmt::Debug for Undo { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { f.debug_struct("Undo") .field("input", &self.input.id_unchecked()) .field("actor", &self.actor.id) .finish() } } impl Undo { pub(crate) fn new(input: AcceptedActivities, actor: Actor) -> Self { Undo { input, actor } } #[tracing::instrument(name = "Undo", skip(state))] async fn perform(self, state: JobState) -> Result<(), Error> { let was_following = state.state.db.is_connected(self.actor.id.clone()).await?; state.actors.remove_connection(&self.actor).await?; if was_following { let my_id = state.config.generate_url(UrlKind::Actor); let undo = generate_undo_follow(&state.config, &self.actor.id, &my_id)?; state .job_server .queue(Deliver::new(self.actor.inbox, undo)?) .await?; } Ok(()) } } impl Job for Undo { type State = JobState; type Future = BoxFuture<'static, anyhow::Result<()>>; const NAME: &'static str = "relay::jobs::apub::Undo"; const QUEUE: &'static str = "apub"; fn run(self, state: Self::State) -> Self::Future { Box::pin(async move { self.perform(state).await.map_err(Into::into) }) } }