Nutomic
c7de1fcf24
* start to implement apub inbox routing lib * got something that almost works * it compiles! * implemented some more * move library code to separate crate (most of it) * convert private message handlers * convert all comment receivers (except undo comment) * convert post receiver * add verify trait * convert community receivers * add cc field for all activities which i forgot before * convert inbox functions, add missing checks * convert undo like/dislike receivers * convert undo_delete and undo_remove receivers * move block/unblock activities * convert remaining activity receivers * reimplement http signature verification and other checks * also use actor type for routing, VerifyActivity and SendActivity traits * cleanup and restructure apub_receive code * wip: try to fix activity routing * implement a (very bad) derive macro for activityhandler * working activity routing! * rework pm verify(), fix tests and confirm manually also remove inbox username check which was broken * rework following verify(), fix tests and test manually * fix post/comment create/update, rework voting * Rewrite remove/delete post/comment, fix tests, test manually * Rework and fix (un)block user, announce, update post * some code cleanup * rework delete/remove activity receivers (still quite messy) * rewrite, test and fix add/remove mod, update community handlers * add docs for ActivityHandler derive macro * dont try to compile macro comments
86 lines
2.7 KiB
Rust
86 lines
2.7 KiB
Rust
use crate::activities::{
|
|
verify_activity,
|
|
verify_add_remove_moderator_target,
|
|
verify_mod_action,
|
|
verify_person_in_community,
|
|
};
|
|
use activitystreams::{activity::kind::AddType, base::AnyBase};
|
|
use lemmy_api_common::blocking;
|
|
use lemmy_apub::{
|
|
fetcher::{community::get_or_fetch_and_upsert_community, person::get_or_fetch_and_upsert_person},
|
|
CommunityType,
|
|
};
|
|
use lemmy_apub_lib::{ActivityCommonFields, ActivityHandler, PublicUrl};
|
|
use lemmy_db_queries::{source::community::CommunityModerator_, Joinable};
|
|
use lemmy_db_schema::source::community::{CommunityModerator, CommunityModeratorForm};
|
|
use lemmy_utils::LemmyError;
|
|
use lemmy_websocket::LemmyContext;
|
|
use url::Url;
|
|
|
|
#[derive(Clone, Debug, serde::Deserialize, serde::Serialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct AddMod {
|
|
to: PublicUrl,
|
|
object: Url,
|
|
target: Url,
|
|
cc: [Url; 1],
|
|
#[serde(rename = "type")]
|
|
kind: AddType,
|
|
#[serde(flatten)]
|
|
common: ActivityCommonFields,
|
|
}
|
|
|
|
#[async_trait::async_trait(?Send)]
|
|
impl ActivityHandler for AddMod {
|
|
async fn verify(
|
|
&self,
|
|
context: &LemmyContext,
|
|
request_counter: &mut i32,
|
|
) -> Result<(), LemmyError> {
|
|
verify_activity(self.common())?;
|
|
verify_person_in_community(&self.common.actor, &self.cc, context, request_counter).await?;
|
|
verify_mod_action(&self.common.actor, self.cc[0].clone(), context).await?;
|
|
verify_add_remove_moderator_target(&self.target, self.cc[0].clone())?;
|
|
Ok(())
|
|
}
|
|
|
|
async fn receive(
|
|
&self,
|
|
context: &LemmyContext,
|
|
request_counter: &mut i32,
|
|
) -> Result<(), LemmyError> {
|
|
let community =
|
|
get_or_fetch_and_upsert_community(&self.cc[0], context, request_counter).await?;
|
|
let new_mod = get_or_fetch_and_upsert_person(&self.object, context, request_counter).await?;
|
|
|
|
// If we had to refetch the community while parsing the activity, then the new mod has already
|
|
// been added. Skip it here as it would result in a duplicate key error.
|
|
let new_mod_id = new_mod.id;
|
|
let moderated_communities = blocking(context.pool(), move |conn| {
|
|
CommunityModerator::get_person_moderated_communities(conn, new_mod_id)
|
|
})
|
|
.await??;
|
|
if !moderated_communities.contains(&community.id) {
|
|
let form = CommunityModeratorForm {
|
|
community_id: community.id,
|
|
person_id: new_mod.id,
|
|
};
|
|
blocking(context.pool(), move |conn| {
|
|
CommunityModerator::join(conn, &form)
|
|
})
|
|
.await??;
|
|
}
|
|
if community.local {
|
|
let anybase = AnyBase::from_arbitrary_json(serde_json::to_string(self)?)?;
|
|
community
|
|
.send_announce(anybase, Some(self.object.clone()), context)
|
|
.await?;
|
|
}
|
|
// TODO: send websocket notification about added mod
|
|
Ok(())
|
|
}
|
|
|
|
fn common(&self) -> &ActivityCommonFields {
|
|
&self.common
|
|
}
|
|
}
|