mirror of
https://github.com/LemmyNet/lemmy.git
synced 2024-05-19 10:18:05 +00:00
aa79c5131f
* Implement federated bans (fixes #1298) * mod actions should always be federated to affected user, in addition to followers * Make Undo/Block work for remote mods * clippy fix * fix federation test * vscodium doesnt auto-save changes...
118 lines
3.1 KiB
Rust
118 lines
3.1 KiB
Rust
use crate::{
|
|
activities::send::generate_activity_id,
|
|
activity_queue::send_activity_single_dest,
|
|
extensions::context::lemmy_context,
|
|
ActorType,
|
|
UserType,
|
|
};
|
|
use activitystreams::{
|
|
activity::{
|
|
kind::{FollowType, UndoType},
|
|
Follow,
|
|
Undo,
|
|
},
|
|
base::{BaseExt, ExtendsExt},
|
|
object::ObjectExt,
|
|
};
|
|
use lemmy_api_common::blocking;
|
|
use lemmy_db_queries::{ApubObject, Followable};
|
|
use lemmy_db_schema::source::{
|
|
community::{Community, CommunityFollower, CommunityFollowerForm},
|
|
person::Person,
|
|
};
|
|
use lemmy_utils::LemmyError;
|
|
use lemmy_websocket::LemmyContext;
|
|
use url::Url;
|
|
|
|
#[async_trait::async_trait(?Send)]
|
|
impl ActorType for Person {
|
|
fn is_local(&self) -> bool {
|
|
self.local
|
|
}
|
|
fn actor_id(&self) -> Url {
|
|
self.actor_id.to_owned().into_inner()
|
|
}
|
|
|
|
fn public_key(&self) -> Option<String> {
|
|
self.public_key.to_owned()
|
|
}
|
|
|
|
fn private_key(&self) -> Option<String> {
|
|
self.private_key.to_owned()
|
|
}
|
|
|
|
fn get_shared_inbox_or_inbox_url(&self) -> Url {
|
|
self
|
|
.shared_inbox_url
|
|
.clone()
|
|
.unwrap_or_else(|| self.inbox_url.to_owned())
|
|
.into()
|
|
}
|
|
}
|
|
|
|
#[async_trait::async_trait(?Send)]
|
|
impl UserType for Person {
|
|
/// As a given local person, send out a follow request to a remote community.
|
|
async fn send_follow(
|
|
&self,
|
|
follow_actor_id: &Url,
|
|
context: &LemmyContext,
|
|
) -> Result<(), LemmyError> {
|
|
let follow_actor_id = follow_actor_id.to_owned();
|
|
let community = blocking(context.pool(), move |conn| {
|
|
Community::read_from_apub_id(conn, &follow_actor_id.into())
|
|
})
|
|
.await??;
|
|
|
|
let community_follower_form = CommunityFollowerForm {
|
|
community_id: community.id,
|
|
person_id: self.id,
|
|
pending: true,
|
|
};
|
|
blocking(&context.pool(), move |conn| {
|
|
CommunityFollower::follow(conn, &community_follower_form).ok()
|
|
})
|
|
.await?;
|
|
|
|
let mut follow = Follow::new(self.actor_id(), community.actor_id());
|
|
follow
|
|
.set_many_contexts(lemmy_context()?)
|
|
.set_id(generate_activity_id(FollowType::Follow)?)
|
|
.set_to(community.actor_id());
|
|
|
|
send_activity_single_dest(follow, self, community.inbox_url.into(), context).await?;
|
|
Ok(())
|
|
}
|
|
|
|
async fn send_unfollow(
|
|
&self,
|
|
follow_actor_id: &Url,
|
|
context: &LemmyContext,
|
|
) -> Result<(), LemmyError> {
|
|
let follow_actor_id = follow_actor_id.to_owned();
|
|
let community = blocking(context.pool(), move |conn| {
|
|
Community::read_from_apub_id(conn, &follow_actor_id.into())
|
|
})
|
|
.await??;
|
|
|
|
let mut follow = Follow::new(self.actor_id(), community.actor_id());
|
|
follow
|
|
.set_many_contexts(lemmy_context()?)
|
|
.set_id(generate_activity_id(FollowType::Follow)?)
|
|
.set_to(community.actor_id());
|
|
|
|
// Undo that fake activity
|
|
let mut undo = Undo::new(
|
|
self.actor_id.to_owned().into_inner(),
|
|
follow.into_any_base()?,
|
|
);
|
|
undo
|
|
.set_many_contexts(lemmy_context()?)
|
|
.set_id(generate_activity_id(UndoType::Undo)?)
|
|
.set_to(community.actor_id());
|
|
|
|
send_activity_single_dest(undo, self, community.inbox_url.into(), context).await?;
|
|
Ok(())
|
|
}
|
|
}
|