4 announce::GetCommunity,
5 get_community_from_moderators_url,
6 send_activity_in_community,
9 verify_add_remove_moderator_target,
12 verify_person_in_community,
14 activity_lists::AnnouncableActivities,
15 generate_moderators_url,
17 objects::{community::ApubCommunity, person::ApubPerson},
18 protocol::activities::community::add_mod::AddMod,
21 use activitypub_federation::{
22 core::object_id::ObjectId,
24 traits::{ActivityHandler, Actor},
26 use activitystreams_kinds::{activity::AddType, public};
27 use lemmy_db_schema::{
29 community::{CommunityModerator, CommunityModeratorForm},
30 moderator::{ModAddCommunity, ModAddCommunityForm},
32 traits::{Crud, Joinable},
34 use lemmy_utils::error::LemmyError;
35 use lemmy_websocket::LemmyContext;
39 #[tracing::instrument(skip_all)]
41 community: &ApubCommunity,
42 added_mod: &ApubPerson,
44 context: &LemmyContext,
45 ) -> Result<(), LemmyError> {
46 let id = generate_activity_id(
48 &context.settings().get_protocol_and_hostname(),
51 actor: ObjectId::new(actor.actor_id()),
53 object: ObjectId::new(added_mod.actor_id()),
54 target: generate_moderators_url(&community.actor_id)?.into(),
55 cc: vec![community.actor_id()],
58 unparsed: Default::default(),
61 let activity = AnnouncableActivities::AddMod(add);
62 let inboxes = vec![added_mod.shared_inbox_or_inbox()];
63 send_activity_in_community(activity, actor, community, inboxes, context).await
67 #[async_trait::async_trait(?Send)]
68 impl ActivityHandler for AddMod {
69 type DataType = LemmyContext;
70 type Error = LemmyError;
72 fn id(&self) -> &Url {
76 fn actor(&self) -> &Url {
80 #[tracing::instrument(skip_all)]
83 context: &Data<LemmyContext>,
84 request_counter: &mut i32,
85 ) -> Result<(), LemmyError> {
86 verify_is_public(&self.to, &self.cc)?;
87 let community = self.get_community(context, request_counter).await?;
88 verify_person_in_community(&self.actor, &community, context, request_counter).await?;
97 verify_add_remove_moderator_target(&self.target, &community)?;
101 #[tracing::instrument(skip_all)]
104 context: &Data<LemmyContext>,
105 request_counter: &mut i32,
106 ) -> Result<(), LemmyError> {
107 let community = self.get_community(context, request_counter).await?;
110 .dereference(context, local_instance(context).await, request_counter)
113 // If we had to refetch the community while parsing the activity, then the new mod has already
114 // been added. Skip it here as it would result in a duplicate key error.
115 let new_mod_id = new_mod.id;
116 let moderated_communities =
117 CommunityModerator::get_person_moderated_communities(context.pool(), new_mod_id).await?;
118 if !moderated_communities.contains(&community.id) {
119 let form = CommunityModeratorForm {
120 community_id: community.id,
121 person_id: new_mod.id,
123 CommunityModerator::join(context.pool(), &form).await?;
128 .dereference(context, local_instance(context).await, request_counter)
130 let form = ModAddCommunityForm {
131 mod_person_id: actor.id,
132 other_person_id: new_mod.id,
133 community_id: community.id,
134 removed: Some(false),
136 ModAddCommunity::create(context.pool(), &form).await?;
138 // TODO: send websocket notification about added mod
143 #[async_trait::async_trait(?Send)]
144 impl GetCommunity for AddMod {
145 #[tracing::instrument(skip_all)]
146 async fn get_community(
148 context: &LemmyContext,
149 request_counter: &mut i32,
150 ) -> Result<ApubCommunity, LemmyError> {
151 get_community_from_moderators_url(&self.target, context, request_counter).await