use crate::{ activities::{ generate_activity_id, send_lemmy_activity, verify_person, verify_person_in_community, }, fetcher::user_or_community::UserOrCommunity, local_instance, objects::{community::ApubCommunity, person::ApubPerson}, protocol::activities::following::{ accept::AcceptFollow, follow::Follow, undo_follow::UndoFollow, }, ActorType, SendActivity, }; use activitypub_federation::{ core::object_id::ObjectId, data::Data, traits::{ActivityHandler, Actor}, }; use activitystreams_kinds::activity::FollowType; use lemmy_api_common::{ community::{BlockCommunity, BlockCommunityResponse}, context::LemmyContext, utils::get_local_user_view_from_jwt, }; use lemmy_db_schema::{ source::{ community::{Community, CommunityFollower, CommunityFollowerForm}, person::{PersonFollower, PersonFollowerForm}, }, traits::{Crud, Followable}, }; use lemmy_utils::error::LemmyError; use url::Url; impl Follow { pub(in crate::activities::following) fn new( actor: &ApubPerson, community: &ApubCommunity, context: &LemmyContext, ) -> Result { Ok(Follow { actor: ObjectId::new(actor.actor_id()), object: ObjectId::new(community.actor_id()), kind: FollowType::Follow, id: generate_activity_id( FollowType::Follow, &context.settings().get_protocol_and_hostname(), )?, }) } #[tracing::instrument(skip_all)] pub async fn send( actor: &ApubPerson, community: &ApubCommunity, context: &LemmyContext, ) -> Result<(), LemmyError> { let community_follower_form = CommunityFollowerForm { community_id: community.id, person_id: actor.id, pending: true, }; CommunityFollower::follow(context.pool(), &community_follower_form) .await .ok(); let follow = Follow::new(actor, community, context)?; let inbox = vec![community.shared_inbox_or_inbox()]; send_lemmy_activity(context, follow, actor, inbox, true).await } } #[async_trait::async_trait(?Send)] impl ActivityHandler for Follow { type DataType = LemmyContext; type Error = LemmyError; fn id(&self) -> &Url { &self.id } fn actor(&self) -> &Url { self.actor.inner() } #[tracing::instrument(skip_all)] async fn verify( &self, context: &Data, request_counter: &mut i32, ) -> Result<(), LemmyError> { verify_person(&self.actor, context, request_counter).await?; let object = self .object .dereference(context, local_instance(context).await, request_counter) .await?; if let UserOrCommunity::Community(c) = object { verify_person_in_community(&self.actor, &c, context, request_counter).await?; } Ok(()) } #[tracing::instrument(skip_all)] async fn receive( self, context: &Data, request_counter: &mut i32, ) -> Result<(), LemmyError> { let actor = self .actor .dereference(context, local_instance(context).await, request_counter) .await?; let object = self .object .dereference(context, local_instance(context).await, request_counter) .await?; match object { UserOrCommunity::User(u) => { let form = PersonFollowerForm { person_id: u.id, follower_id: actor.id, pending: false, }; PersonFollower::follow(context.pool(), &form).await?; } UserOrCommunity::Community(c) => { let form = CommunityFollowerForm { community_id: c.id, person_id: actor.id, pending: false, }; CommunityFollower::follow(context.pool(), &form).await?; } } AcceptFollow::send(self, context, request_counter).await } } #[async_trait::async_trait(?Send)] impl SendActivity for BlockCommunity { type Response = BlockCommunityResponse; async fn send_activity( request: &Self, _response: &Self::Response, context: &LemmyContext, ) -> Result<(), LemmyError> { let local_user_view = get_local_user_view_from_jwt(&request.auth, context.pool(), context.secret()).await?; let community = Community::read(context.pool(), request.community_id).await?; UndoFollow::send(&local_user_view.person.into(), &community.into(), context).await } }