use crate::ban_nonlocal_user_from_local_communities; use activitypub_federation::config::Data; use actix_web::web::Json; use lemmy_api_common::{ context::LemmyContext, person::{BanPerson, BanPersonResponse}, send_activity::{ActivityChannel, SendActivityData}, utils::{check_expire_time, is_admin, remove_user_data}, }; use lemmy_db_schema::{ source::{ login_token::LoginToken, moderator::{ModBan, ModBanForm}, person::{Person, PersonUpdateForm}, }, traits::Crud, }; use lemmy_db_views::structs::LocalUserView; use lemmy_db_views_actor::structs::PersonView; use lemmy_utils::{ error::{LemmyError, LemmyErrorExt, LemmyErrorType}, utils::validation::is_valid_body_field, }; #[tracing::instrument(skip(context))] pub async fn ban_from_site( data: Json, context: Data, local_user_view: LocalUserView, ) -> Result, LemmyError> { // Make sure user is an admin is_admin(&local_user_view)?; is_valid_body_field(&data.reason, false)?; let expires = check_expire_time(data.expires)?; let person = Person::update( &mut context.pool(), data.person_id, &PersonUpdateForm { banned: Some(data.ban), ban_expires: Some(expires), ..Default::default() }, ) .await .with_lemmy_type(LemmyErrorType::CouldntUpdateUser)?; // if its a local user, invalidate logins let local_user = LocalUserView::read_person(&mut context.pool(), person.id).await; if let Ok(local_user) = local_user { LoginToken::invalidate_all(&mut context.pool(), local_user.local_user.id).await?; } // Remove their data if that's desired let remove_data = data.remove_data.unwrap_or(false); if remove_data { remove_user_data(person.id, &context).await?; } // Mod tables let form = ModBanForm { mod_person_id: local_user_view.person.id, other_person_id: person.id, reason: data.reason.clone(), banned: Some(data.ban), expires, }; ModBan::create(&mut context.pool(), &form).await?; let person_view = PersonView::read(&mut context.pool(), person.id).await?; ban_nonlocal_user_from_local_communities( &local_user_view, &person, data.ban, &data.reason, &data.remove_data, &data.expires, &context, ) .await?; ActivityChannel::submit_activity( SendActivityData::BanFromSite { moderator: local_user_view.person, banned_user: person_view.person.clone(), reason: data.reason.clone(), remove_data: data.remove_data, ban: data.ban, expires: data.expires, }, &context, ) .await?; Ok(Json(BanPersonResponse { person_view, banned: data.ban, })) }