Apub inbox rewrite (#1652)
* 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
2021-07-17 16:08:46 +00:00
|
|
|
use crate::{
|
2022-11-12 13:52:57 +00:00
|
|
|
activities::{
|
|
|
|
generate_activity_id,
|
2023-09-20 10:46:25 +00:00
|
|
|
generate_announce_activity_id,
|
2022-11-12 13:52:57 +00:00
|
|
|
send_lemmy_activity,
|
|
|
|
verify_is_public,
|
|
|
|
verify_person_in_community,
|
|
|
|
},
|
2021-10-29 10:32:42 +00:00
|
|
|
activity_lists::AnnouncableActivities,
|
2023-07-14 15:17:06 +00:00
|
|
|
insert_received_activity,
|
2021-10-18 21:36:44 +00:00
|
|
|
objects::community::ApubCommunity,
|
2022-05-06 23:53:33 +00:00
|
|
|
protocol::{
|
2022-11-12 13:52:57 +00:00
|
|
|
activities::community::announce::{AnnounceActivity, RawAnnouncableActivities},
|
|
|
|
Id,
|
2022-05-06 23:53:33 +00:00
|
|
|
IdOrNestedObject,
|
2022-12-01 20:52:49 +00:00
|
|
|
InCommunity,
|
2022-05-06 23:53:33 +00:00
|
|
|
},
|
2021-08-19 21:24:33 +00:00
|
|
|
};
|
2023-03-21 15:03:05 +00:00
|
|
|
use activitypub_federation::{
|
|
|
|
config::Data,
|
|
|
|
kinds::{activity::AnnounceType, public},
|
|
|
|
traits::{ActivityHandler, Actor},
|
|
|
|
};
|
2022-11-28 14:29:33 +00:00
|
|
|
use lemmy_api_common::context::LemmyContext;
|
2023-10-16 10:03:49 +00:00
|
|
|
use lemmy_db_schema::source::{activity::ActivitySendTargets, community::CommunityFollower};
|
|
|
|
use lemmy_utils::error::{LemmyError, LemmyErrorType, LemmyResult};
|
2022-11-12 13:52:57 +00:00
|
|
|
use serde_json::Value;
|
2022-06-02 14:33:41 +00:00
|
|
|
use url::Url;
|
Apub inbox rewrite (#1652)
* 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
2021-07-17 16:08:46 +00:00
|
|
|
|
2023-03-21 15:03:05 +00:00
|
|
|
#[async_trait::async_trait]
|
2022-11-12 13:52:57 +00:00
|
|
|
impl ActivityHandler for RawAnnouncableActivities {
|
|
|
|
type DataType = LemmyContext;
|
|
|
|
type Error = LemmyError;
|
|
|
|
|
|
|
|
fn id(&self) -> &Url {
|
|
|
|
&self.id
|
|
|
|
}
|
|
|
|
|
|
|
|
fn actor(&self) -> &Url {
|
|
|
|
&self.actor
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tracing::instrument(skip_all)]
|
2023-03-21 15:03:05 +00:00
|
|
|
async fn verify(&self, _data: &Data<Self::DataType>) -> Result<(), Self::Error> {
|
2022-11-12 13:52:57 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tracing::instrument(skip_all)]
|
2023-10-16 10:03:49 +00:00
|
|
|
async fn receive(self, context: &Data<Self::DataType>) -> Result<(), Self::Error> {
|
2022-11-12 13:52:57 +00:00
|
|
|
let activity: AnnouncableActivities = self.clone().try_into()?;
|
2023-10-16 10:03:49 +00:00
|
|
|
|
2022-11-12 13:52:57 +00:00
|
|
|
// This is only for sending, not receiving so we reject it.
|
|
|
|
if let AnnouncableActivities::Page(_) = activity {
|
2023-08-31 13:01:08 +00:00
|
|
|
Err(LemmyErrorType::CannotReceivePage)?
|
2022-11-12 13:52:57 +00:00
|
|
|
}
|
|
|
|
|
2023-10-16 10:03:49 +00:00
|
|
|
// Need to treat community as optional here because `Delete/PrivateMessage` gets routed through
|
|
|
|
let community = activity.community(context).await.ok();
|
|
|
|
can_accept_activity_in_community(&community, context).await?;
|
|
|
|
|
2022-11-12 13:52:57 +00:00
|
|
|
// verify and receive activity
|
2023-10-16 10:03:49 +00:00
|
|
|
activity.verify(context).await?;
|
|
|
|
activity.clone().receive(context).await?;
|
2022-11-12 13:52:57 +00:00
|
|
|
|
2023-10-16 10:03:49 +00:00
|
|
|
// if community is local, send activity to followers
|
|
|
|
if let Some(community) = community {
|
2023-07-27 10:17:40 +00:00
|
|
|
if community.local {
|
|
|
|
let actor_id = activity.actor().clone().into();
|
2023-10-16 10:03:49 +00:00
|
|
|
verify_person_in_community(&actor_id, &community, context).await?;
|
|
|
|
AnnounceActivity::send(self, &community, context).await?;
|
2023-07-27 10:17:40 +00:00
|
|
|
}
|
2022-11-12 13:52:57 +00:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-27 22:18:50 +00:00
|
|
|
impl AnnounceActivity {
|
2021-11-18 17:04:28 +00:00
|
|
|
pub(crate) fn new(
|
2022-11-12 13:52:57 +00:00
|
|
|
object: RawAnnouncableActivities,
|
2021-10-18 21:36:44 +00:00
|
|
|
community: &ApubCommunity,
|
2023-03-21 15:03:05 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2021-11-11 19:49:15 +00:00
|
|
|
) -> Result<AnnounceActivity, LemmyError> {
|
2023-09-20 10:46:25 +00:00
|
|
|
let inner_kind = object
|
|
|
|
.other
|
|
|
|
.get("type")
|
2023-09-20 11:41:34 +00:00
|
|
|
.and_then(serde_json::Value::as_str)
|
2023-09-20 10:46:25 +00:00
|
|
|
.unwrap_or("other");
|
|
|
|
let id =
|
|
|
|
generate_announce_activity_id(inner_kind, &context.settings().get_protocol_and_hostname())?;
|
2021-11-11 19:49:15 +00:00
|
|
|
Ok(AnnounceActivity {
|
2023-03-21 15:03:05 +00:00
|
|
|
actor: community.id().into(),
|
2021-10-25 17:22:34 +00:00
|
|
|
to: vec![public()],
|
2022-05-06 23:53:33 +00:00
|
|
|
object: IdOrNestedObject::NestedObject(object),
|
2021-11-05 00:24:10 +00:00
|
|
|
cc: vec![community.followers_url.clone().into()],
|
2021-07-27 22:18:50 +00:00
|
|
|
kind: AnnounceType::Announce,
|
2023-09-20 10:46:25 +00:00
|
|
|
id,
|
2021-11-11 19:49:15 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-12-06 14:54:47 +00:00
|
|
|
#[tracing::instrument(skip_all)]
|
2021-11-11 19:49:15 +00:00
|
|
|
pub async fn send(
|
2022-11-12 13:52:57 +00:00
|
|
|
object: RawAnnouncableActivities,
|
2021-11-11 19:49:15 +00:00
|
|
|
community: &ApubCommunity,
|
2023-03-21 15:03:05 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2021-11-11 19:49:15 +00:00
|
|
|
) -> Result<(), LemmyError> {
|
|
|
|
let announce = AnnounceActivity::new(object.clone(), community, context)?;
|
2023-09-09 16:25:03 +00:00
|
|
|
let inboxes = ActivitySendTargets::to_local_community_followers(community.id);
|
2022-06-08 15:45:39 +00:00
|
|
|
send_lemmy_activity(context, announce, community, inboxes.clone(), false).await?;
|
2021-11-11 19:49:15 +00:00
|
|
|
|
2021-11-18 15:20:35 +00:00
|
|
|
// Pleroma and Mastodon can't handle activities like Announce/Create/Page. So for
|
|
|
|
// compatibility, we also send Announce/Page so that they can follow Lemmy communities.
|
2022-11-12 13:52:57 +00:00
|
|
|
let object_parsed = object.try_into()?;
|
|
|
|
if let AnnouncableActivities::CreateOrUpdatePost(c) = object_parsed {
|
|
|
|
// Hack: need to convert Page into a format which can be sent as activity, which requires
|
|
|
|
// adding actor field.
|
|
|
|
let announcable_page = RawAnnouncableActivities {
|
2023-03-21 15:03:05 +00:00
|
|
|
id: generate_activity_id(
|
|
|
|
AnnounceType::Announce,
|
|
|
|
&context.settings().get_protocol_and_hostname(),
|
|
|
|
)?,
|
2022-11-12 13:52:57 +00:00
|
|
|
actor: c.actor.clone().into_inner(),
|
2022-11-14 14:30:44 +00:00
|
|
|
other: serde_json::to_value(c.object)?
|
|
|
|
.as_object()
|
|
|
|
.expect("is object")
|
|
|
|
.clone(),
|
2022-11-12 13:52:57 +00:00
|
|
|
};
|
|
|
|
let announce_compat = AnnounceActivity::new(announcable_page, community, context)?;
|
|
|
|
send_lemmy_activity(context, announce_compat, community, inboxes, false).await?;
|
|
|
|
}
|
2021-11-11 19:49:15 +00:00
|
|
|
Ok(())
|
2021-07-27 22:18:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-21 15:03:05 +00:00
|
|
|
#[async_trait::async_trait]
|
Apub inbox rewrite (#1652)
* 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
2021-07-17 16:08:46 +00:00
|
|
|
impl ActivityHandler for AnnounceActivity {
|
2021-10-06 20:20:05 +00:00
|
|
|
type DataType = LemmyContext;
|
2022-06-02 14:33:41 +00:00
|
|
|
type Error = LemmyError;
|
|
|
|
|
|
|
|
fn id(&self) -> &Url {
|
|
|
|
&self.id
|
|
|
|
}
|
|
|
|
|
|
|
|
fn actor(&self) -> &Url {
|
|
|
|
self.actor.inner()
|
|
|
|
}
|
2021-12-06 14:54:47 +00:00
|
|
|
|
|
|
|
#[tracing::instrument(skip_all)]
|
2024-01-11 23:56:19 +00:00
|
|
|
async fn verify(&self, _context: &Data<Self::DataType>) -> Result<(), LemmyError> {
|
2021-11-06 17:44:34 +00:00
|
|
|
verify_is_public(&self.to, &self.cc)?;
|
Apub inbox rewrite (#1652)
* 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
2021-07-17 16:08:46 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2021-12-06 14:54:47 +00:00
|
|
|
#[tracing::instrument(skip_all)]
|
2023-03-21 15:03:05 +00:00
|
|
|
async fn receive(self, context: &Data<Self::DataType>) -> Result<(), LemmyError> {
|
2024-01-11 23:56:19 +00:00
|
|
|
insert_received_activity(&self.id, context).await?;
|
2023-03-21 15:03:05 +00:00
|
|
|
let object: AnnouncableActivities = self.object.object(context).await?.try_into()?;
|
2023-10-16 10:03:49 +00:00
|
|
|
|
2022-11-12 13:52:57 +00:00
|
|
|
// This is only for sending, not receiving so we reject it.
|
|
|
|
if let AnnouncableActivities::Page(_) = object {
|
2023-08-31 13:01:08 +00:00
|
|
|
Err(LemmyErrorType::CannotReceivePage)?
|
2022-11-12 13:52:57 +00:00
|
|
|
}
|
|
|
|
|
2023-10-16 10:03:49 +00:00
|
|
|
let community = object.community(context).await?;
|
|
|
|
can_accept_activity_in_community(&Some(community), context).await?;
|
|
|
|
|
2023-03-21 15:03:05 +00:00
|
|
|
// verify here in order to avoid fetching the object twice over http
|
|
|
|
object.verify(context).await?;
|
|
|
|
object.receive(context).await
|
Apub inbox rewrite (#1652)
* 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
2021-07-17 16:08:46 +00:00
|
|
|
}
|
|
|
|
}
|
2022-11-12 13:52:57 +00:00
|
|
|
|
|
|
|
impl Id for RawAnnouncableActivities {
|
|
|
|
fn object_id(&self) -> &Url {
|
|
|
|
ActivityHandler::id(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<RawAnnouncableActivities> for AnnouncableActivities {
|
|
|
|
type Error = serde_json::error::Error;
|
|
|
|
|
|
|
|
fn try_from(value: RawAnnouncableActivities) -> Result<Self, Self::Error> {
|
|
|
|
let mut map = value.other.clone();
|
|
|
|
map.insert("id".to_string(), Value::String(value.id.to_string()));
|
|
|
|
map.insert("actor".to_string(), Value::String(value.actor.to_string()));
|
|
|
|
serde_json::from_value(Value::Object(map))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<AnnouncableActivities> for RawAnnouncableActivities {
|
|
|
|
type Error = serde_json::error::Error;
|
|
|
|
|
|
|
|
fn try_from(value: AnnouncableActivities) -> Result<Self, Self::Error> {
|
|
|
|
serde_json::from_value(serde_json::to_value(value)?)
|
|
|
|
}
|
|
|
|
}
|
2023-10-16 10:03:49 +00:00
|
|
|
|
|
|
|
/// Check if an activity in the given community can be accepted. To return true, the community must
|
|
|
|
/// either be local to this instance, or it must have at least one local follower.
|
|
|
|
///
|
|
|
|
/// TODO: This means mentions dont work if the community has no local followers. Can be fixed
|
|
|
|
/// by checking if any local user is in to/cc fields of activity. Anyway this is a minor
|
|
|
|
/// problem compared to receiving unsolicited posts.
|
|
|
|
async fn can_accept_activity_in_community(
|
|
|
|
community: &Option<ApubCommunity>,
|
|
|
|
context: &Data<LemmyContext>,
|
|
|
|
) -> LemmyResult<()> {
|
|
|
|
if let Some(community) = community {
|
|
|
|
if !community.local
|
|
|
|
&& !CommunityFollower::has_local_followers(&mut context.pool(), community.id).await?
|
|
|
|
{
|
|
|
|
Err(LemmyErrorType::CommunityHasNoFollowers)?
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|