use anyhow::Context;
use async_trait::async_trait;
use mas_storage::{
compat::CompatSessionFilter,
oauth2::OAuth2SessionFilter,
queue::{DeactivateUserJob, ReactivateUserJob},
user::{BrowserSessionFilter, UserRepository},
RepositoryAccess,
};
use tracing::info;
use crate::{
new_queue::{JobContext, JobError, RunnableJob},
State,
};
#[async_trait]
impl RunnableJob for DeactivateUserJob {
#[tracing::instrument(
name = "job.deactivate_user"
fields(user.id = %self.user_id(), erase = %self.hs_erase()),
skip_all,
err,
)]
async fn run(&self, state: &State, _context: JobContext) -> Result<(), JobError> {
let clock = state.clock();
let matrix = state.matrix_connection();
let mut repo = state.repository().await.map_err(JobError::retry)?;
let user = repo
.user()
.lookup(self.user_id())
.await
.map_err(JobError::retry)?
.context("User not found")
.map_err(JobError::fail)?;
let user = repo
.user()
.lock(&clock, user)
.await
.context("Failed to lock user")
.map_err(JobError::retry)?;
let n = repo
.browser_session()
.finish_bulk(
&clock,
BrowserSessionFilter::new().for_user(&user).active_only(),
)
.await
.map_err(JobError::retry)?;
info!(affected = n, "Killed all browser sessions for user");
let n = repo
.oauth2_session()
.finish_bulk(
&clock,
OAuth2SessionFilter::new().for_user(&user).active_only(),
)
.await
.map_err(JobError::retry)?;
info!(affected = n, "Killed all OAuth 2.0 sessions for user");
let n = repo
.compat_session()
.finish_bulk(
&clock,
CompatSessionFilter::new().for_user(&user).active_only(),
)
.await
.map_err(JobError::retry)?;
info!(affected = n, "Killed all compatibility sessions for user");
repo.save().await.map_err(JobError::retry)?;
let mxid = matrix.mxid(&user.username);
info!("Deactivating user {} on homeserver", mxid);
matrix
.delete_user(&mxid, self.hs_erase())
.await
.map_err(JobError::retry)?;
Ok(())
}
}
#[async_trait]
impl RunnableJob for ReactivateUserJob {
#[tracing::instrument(
name = "job.reactivate_user",
fields(user.id = %self.user_id()),
skip_all,
err,
)]
async fn run(&self, state: &State, _context: JobContext) -> Result<(), JobError> {
let matrix = state.matrix_connection();
let mut repo = state.repository().await.map_err(JobError::retry)?;
let user = repo
.user()
.lookup(self.user_id())
.await
.map_err(JobError::retry)?
.context("User not found")
.map_err(JobError::fail)?;
let mxid = matrix.mxid(&user.username);
info!("Reactivating user {} on homeserver", mxid);
matrix
.reactivate_user(&mxid)
.await
.map_err(JobError::retry)?;
let _user = repo.user().unlock(user).await.map_err(JobError::retry)?;
repo.save().await.map_err(JobError::retry)?;
Ok(())
}
}