]> Untitled Git - lemmy.git/blob - crates/apub/src/activities/community/add_mod.rs
Add missing mod log entries for federated actions (fixes #1489) (#2198)
[lemmy.git] / crates / apub / src / activities / community / add_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::add_mod::AddMod,
19 };
20 use activitystreams_kinds::{activity::AddType, 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 AddMod {
38   #[tracing::instrument(skip_all)]
39   pub async fn send(
40     community: &ApubCommunity,
41     added_mod: &ApubPerson,
42     actor: &ApubPerson,
43     context: &LemmyContext,
44   ) -> Result<(), LemmyError> {
45     let id = generate_activity_id(
46       AddType::Add,
47       &context.settings().get_protocol_and_hostname(),
48     )?;
49     let add = AddMod {
50       actor: ObjectId::new(actor.actor_id()),
51       to: vec![public()],
52       object: ObjectId::new(added_mod.actor_id()),
53       target: generate_moderators_url(&community.actor_id)?.into(),
54       cc: vec![community.actor_id()],
55       kind: AddType::Add,
56       id: id.clone(),
57       unparsed: Default::default(),
58     };
59
60     let activity = AnnouncableActivities::AddMod(add);
61     let inboxes = vec![added_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 AddMod {
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 new_mod = self
100       .object
101       .dereference(context, context.client(), request_counter)
102       .await?;
103
104     // If we had to refetch the community while parsing the activity, then the new mod has already
105     // been added. Skip it here as it would result in a duplicate key error.
106     let new_mod_id = new_mod.id;
107     let moderated_communities = blocking(context.pool(), move |conn| {
108       CommunityModerator::get_person_moderated_communities(conn, new_mod_id)
109     })
110     .await??;
111     if !moderated_communities.contains(&community.id) {
112       let form = CommunityModeratorForm {
113         community_id: community.id,
114         person_id: new_mod.id,
115       };
116       blocking(context.pool(), move |conn| {
117         CommunityModerator::join(conn, &form)
118       })
119       .await??;
120
121       // write mod log
122       let actor = self
123         .actor
124         .dereference(context, context.client(), request_counter)
125         .await?;
126       let form = ModAddCommunityForm {
127         mod_person_id: actor.id,
128         other_person_id: new_mod.id,
129         community_id: community.id,
130         removed: Some(false),
131       };
132       blocking(context.pool(), move |conn| {
133         ModAddCommunity::create(conn, &form)
134       })
135       .await??;
136     }
137     // TODO: send websocket notification about added mod
138     Ok(())
139   }
140 }
141
142 #[async_trait::async_trait(?Send)]
143 impl GetCommunity for AddMod {
144   #[tracing::instrument(skip_all)]
145   async fn get_community(
146     &self,
147     context: &LemmyContext,
148     request_counter: &mut i32,
149   ) -> Result<ApubCommunity, LemmyError> {
150     get_community_from_moderators_url(&self.target, context, request_counter).await
151   }
152 }