]> Untitled Git - lemmy.git/blob - crates/apub/src/activities/community/remove_mod.rs
Add missing mod log entries for federated actions (fixes #1489) (#2198)
[lemmy.git] / crates / apub / src / activities / community / remove_mod.rs
1 use crate::{
2   activities::{
3     community::{
4       announce::GetCommunity,
5       get_community_from_moderators_url,
6       send_activity_in_community,
7     },
8     generate_activity_id,
9     verify_activity,
10     verify_add_remove_moderator_target,
11     verify_is_public,
12     verify_mod_action,
13     verify_person_in_community,
14   },
15   activity_lists::AnnouncableActivities,
16   generate_moderators_url,
17   objects::{community::ApubCommunity, person::ApubPerson},
18   protocol::activities::community::remove_mod::RemoveMod,
19 };
20 use activitystreams_kinds::{activity::RemoveType, public};
21 use lemmy_api_common::blocking;
22 use lemmy_apub_lib::{
23   data::Data,
24   object_id::ObjectId,
25   traits::{ActivityHandler, ActorType},
26 };
27 use lemmy_db_schema::{
28   source::{
29     community::{CommunityModerator, CommunityModeratorForm},
30     moderator::{ModAddCommunity, ModAddCommunityForm},
31   },
32   traits::{Crud, Joinable},
33 };
34 use lemmy_utils::LemmyError;
35 use lemmy_websocket::LemmyContext;
36
37 impl RemoveMod {
38   #[tracing::instrument(skip_all)]
39   pub async fn send(
40     community: &ApubCommunity,
41     removed_mod: &ApubPerson,
42     actor: &ApubPerson,
43     context: &LemmyContext,
44   ) -> Result<(), LemmyError> {
45     let id = generate_activity_id(
46       RemoveType::Remove,
47       &context.settings().get_protocol_and_hostname(),
48     )?;
49     let remove = RemoveMod {
50       actor: ObjectId::new(actor.actor_id()),
51       to: vec![public()],
52       object: ObjectId::new(removed_mod.actor_id()),
53       target: generate_moderators_url(&community.actor_id)?.into(),
54       id: id.clone(),
55       cc: vec![community.actor_id()],
56       kind: RemoveType::Remove,
57       unparsed: Default::default(),
58     };
59
60     let activity = AnnouncableActivities::RemoveMod(remove);
61     let inboxes = vec![removed_mod.shared_inbox_or_inbox_url()];
62     send_activity_in_community(activity, &id, actor, community, inboxes, context).await
63   }
64 }
65
66 #[async_trait::async_trait(?Send)]
67 impl ActivityHandler for RemoveMod {
68   type DataType = LemmyContext;
69
70   #[tracing::instrument(skip_all)]
71   async fn verify(
72     &self,
73     context: &Data<LemmyContext>,
74     request_counter: &mut i32,
75   ) -> Result<(), LemmyError> {
76     verify_is_public(&self.to, &self.cc)?;
77     verify_activity(&self.id, self.actor.inner(), &context.settings())?;
78     let community = self.get_community(context, request_counter).await?;
79     verify_person_in_community(&self.actor, &community, context, request_counter).await?;
80     verify_mod_action(
81       &self.actor,
82       self.object.inner(),
83       &community,
84       context,
85       request_counter,
86     )
87     .await?;
88     verify_add_remove_moderator_target(&self.target, &community)?;
89     Ok(())
90   }
91
92   #[tracing::instrument(skip_all)]
93   async fn receive(
94     self,
95     context: &Data<LemmyContext>,
96     request_counter: &mut i32,
97   ) -> Result<(), LemmyError> {
98     let community = self.get_community(context, request_counter).await?;
99     let remove_mod = self
100       .object
101       .dereference(context, context.client(), request_counter)
102       .await?;
103
104     let form = CommunityModeratorForm {
105       community_id: community.id,
106       person_id: remove_mod.id,
107     };
108     blocking(context.pool(), move |conn| {
109       CommunityModerator::leave(conn, &form)
110     })
111     .await??;
112
113     // write mod log
114     let actor = self
115       .actor
116       .dereference(context, context.client(), request_counter)
117       .await?;
118     let form = ModAddCommunityForm {
119       mod_person_id: actor.id,
120       other_person_id: remove_mod.id,
121       community_id: community.id,
122       removed: Some(true),
123     };
124     blocking(context.pool(), move |conn| {
125       ModAddCommunity::create(conn, &form)
126     })
127     .await??;
128
129     // TODO: send websocket notification about removed mod
130     Ok(())
131   }
132 }
133
134 #[async_trait::async_trait(?Send)]
135 impl GetCommunity for RemoveMod {
136   #[tracing::instrument(skip_all)]
137   async fn get_community(
138     &self,
139     context: &LemmyContext,
140     request_counter: &mut i32,
141   ) -> Result<ApubCommunity, LemmyError> {
142     get_community_from_moderators_url(&self.target, context, request_counter).await
143   }
144 }