]> Untitled Git - lemmy.git/blob - crates/apub/src/activities/community/add_mod.rs
Activity.to should always be a vec (and unspecified size for public activities)
[lemmy.git] / crates / apub / src / activities / community / add_mod.rs
1 use crate::{
2   activities::{
3     community::{announce::AnnouncableActivities, send_to_community},
4     generate_activity_id,
5     verify_activity,
6     verify_add_remove_moderator_target,
7     verify_is_public,
8     verify_mod_action,
9     verify_person_in_community,
10   },
11   context::lemmy_context,
12   fetcher::object_id::ObjectId,
13   generate_moderators_url,
14   objects::{community::ApubCommunity, person::ApubPerson},
15 };
16 use activitystreams::{
17   activity::kind::AddType,
18   base::AnyBase,
19   primitives::OneOrMany,
20   public,
21   unparsed::Unparsed,
22 };
23 use lemmy_api_common::blocking;
24 use lemmy_apub_lib::{
25   data::Data,
26   traits::{ActivityFields, ActivityHandler, ActorType},
27 };
28 use lemmy_db_schema::{
29   source::community::{CommunityModerator, CommunityModeratorForm},
30   traits::Joinable,
31 };
32 use lemmy_utils::LemmyError;
33 use lemmy_websocket::LemmyContext;
34 use serde::{Deserialize, Serialize};
35 use url::Url;
36
37 #[derive(Clone, Debug, Deserialize, Serialize, ActivityFields)]
38 #[serde(rename_all = "camelCase")]
39 pub struct AddMod {
40   actor: ObjectId<ApubPerson>,
41   to: Vec<Url>,
42   object: ObjectId<ApubPerson>,
43   target: Url,
44   cc: [ObjectId<ApubCommunity>; 1],
45   #[serde(rename = "type")]
46   kind: AddType,
47   id: Url,
48   #[serde(rename = "@context")]
49   context: OneOrMany<AnyBase>,
50   #[serde(flatten)]
51   unparsed: Unparsed,
52 }
53
54 impl AddMod {
55   pub async fn send(
56     community: &ApubCommunity,
57     added_mod: &ApubPerson,
58     actor: &ApubPerson,
59     context: &LemmyContext,
60   ) -> Result<(), LemmyError> {
61     let id = generate_activity_id(
62       AddType::Add,
63       &context.settings().get_protocol_and_hostname(),
64     )?;
65     let add = AddMod {
66       actor: ObjectId::new(actor.actor_id()),
67       to: vec![public()],
68       object: ObjectId::new(added_mod.actor_id()),
69       target: generate_moderators_url(&community.actor_id)?.into(),
70       cc: [ObjectId::new(community.actor_id())],
71       kind: AddType::Add,
72       id: id.clone(),
73       context: lemmy_context(),
74       unparsed: Default::default(),
75     };
76
77     let activity = AnnouncableActivities::AddMod(add);
78     let inboxes = vec![added_mod.shared_inbox_or_inbox_url()];
79     send_to_community(activity, &id, actor, community, inboxes, context).await
80   }
81 }
82
83 #[async_trait::async_trait(?Send)]
84 impl ActivityHandler for AddMod {
85   type DataType = LemmyContext;
86
87   async fn verify(
88     &self,
89     context: &Data<LemmyContext>,
90     request_counter: &mut i32,
91   ) -> Result<(), LemmyError> {
92     verify_is_public(&self.to)?;
93     verify_activity(self, &context.settings())?;
94     verify_person_in_community(&self.actor, &self.cc[0], context, request_counter).await?;
95     verify_mod_action(&self.actor, &self.cc[0], context, request_counter).await?;
96     verify_add_remove_moderator_target(&self.target, &self.cc[0])?;
97     Ok(())
98   }
99
100   async fn receive(
101     self,
102     context: &Data<LemmyContext>,
103     request_counter: &mut i32,
104   ) -> Result<(), LemmyError> {
105     let community = self.cc[0].dereference(context, request_counter).await?;
106     let new_mod = self.object.dereference(context, request_counter).await?;
107
108     // If we had to refetch the community while parsing the activity, then the new mod has already
109     // been added. Skip it here as it would result in a duplicate key error.
110     let new_mod_id = new_mod.id;
111     let moderated_communities = blocking(context.pool(), move |conn| {
112       CommunityModerator::get_person_moderated_communities(conn, new_mod_id)
113     })
114     .await??;
115     if !moderated_communities.contains(&community.id) {
116       let form = CommunityModeratorForm {
117         community_id: community.id,
118         person_id: new_mod.id,
119       };
120       blocking(context.pool(), move |conn| {
121         CommunityModerator::join(conn, &form)
122       })
123       .await??;
124     }
125     // TODO: send websocket notification about added mod
126     Ok(())
127   }
128 }