use crate::{ activities::{ community::{ announce::{AnnouncableActivities, GetCommunity}, get_community_from_moderators_url, send_to_community, }, generate_activity_id, verify_activity, verify_add_remove_moderator_target, verify_is_public, verify_mod_action, verify_person_in_community, }, context::lemmy_context, fetcher::object_id::ObjectId, generate_moderators_url, objects::{community::ApubCommunity, person::ApubPerson}, }; use activitystreams::{ activity::kind::AddType, base::AnyBase, primitives::OneOrMany, public, unparsed::Unparsed, }; use lemmy_api_common::blocking; use lemmy_apub_lib::{ data::Data, traits::{ActivityFields, ActivityHandler, ActorType}, }; use lemmy_db_schema::{ source::community::{CommunityModerator, CommunityModeratorForm}, traits::Joinable, }; use lemmy_utils::LemmyError; use lemmy_websocket::LemmyContext; use serde::{Deserialize, Serialize}; use url::Url; #[derive(Clone, Debug, Deserialize, Serialize, ActivityFields)] #[serde(rename_all = "camelCase")] pub struct AddMod { actor: ObjectId, to: Vec, object: ObjectId, target: Url, cc: [ObjectId; 1], #[serde(rename = "type")] kind: AddType, id: Url, #[serde(rename = "@context")] context: OneOrMany, #[serde(flatten)] unparsed: Unparsed, } impl AddMod { pub async fn send( community: &ApubCommunity, added_mod: &ApubPerson, actor: &ApubPerson, context: &LemmyContext, ) -> Result<(), LemmyError> { let id = generate_activity_id( AddType::Add, &context.settings().get_protocol_and_hostname(), )?; let add = AddMod { actor: ObjectId::new(actor.actor_id()), to: vec![public()], object: ObjectId::new(added_mod.actor_id()), target: generate_moderators_url(&community.actor_id)?.into(), cc: [ObjectId::new(community.actor_id())], kind: AddType::Add, id: id.clone(), context: lemmy_context(), unparsed: Default::default(), }; let activity = AnnouncableActivities::AddMod(add); let inboxes = vec![added_mod.shared_inbox_or_inbox_url()]; send_to_community(activity, &id, actor, community, inboxes, context).await } } #[async_trait::async_trait(?Send)] impl ActivityHandler for AddMod { type DataType = LemmyContext; async fn verify( &self, context: &Data, request_counter: &mut i32, ) -> Result<(), LemmyError> { verify_is_public(&self.to)?; verify_activity(self, &context.settings())?; let community = self.get_community(context, request_counter).await?; verify_person_in_community(&self.actor, &community, context, request_counter).await?; verify_mod_action(&self.actor, &self.cc[0], context, request_counter).await?; verify_add_remove_moderator_target(&self.target, &self.cc[0])?; Ok(()) } async fn receive( self, context: &Data, request_counter: &mut i32, ) -> Result<(), LemmyError> { let community = self.get_community(context, request_counter).await?; let new_mod = self.object.dereference(context, request_counter).await?; // If we had to refetch the community while parsing the activity, then the new mod has already // been added. Skip it here as it would result in a duplicate key error. let new_mod_id = new_mod.id; let moderated_communities = blocking(context.pool(), move |conn| { CommunityModerator::get_person_moderated_communities(conn, new_mod_id) }) .await??; if !moderated_communities.contains(&community.id) { let form = CommunityModeratorForm { community_id: community.id, person_id: new_mod.id, }; blocking(context.pool(), move |conn| { CommunityModerator::join(conn, &form) }) .await??; } // TODO: send websocket notification about added mod Ok(()) } } #[async_trait::async_trait(?Send)] impl GetCommunity for AddMod { async fn get_community( &self, context: &LemmyContext, request_counter: &mut i32, ) -> Result { get_community_from_moderators_url(&self.target, context, request_counter).await } }