mirror of
https://github.com/LemmyNet/lemmy.git
synced 2024-11-15 00:43:59 +00:00
Merge remote-tracking branch 'yerba/main' into main
This commit is contained in:
commit
a897ec48d2
16 changed files with 1002 additions and 1119 deletions
|
@ -1,51 +0,0 @@
|
||||||
use crate::{
|
|
||||||
activities::receive::{
|
|
||||||
create::receive_create,
|
|
||||||
delete::receive_delete,
|
|
||||||
dislike::receive_dislike,
|
|
||||||
like::receive_like,
|
|
||||||
receive_unhandled_activity,
|
|
||||||
remove::receive_remove,
|
|
||||||
undo::receive_undo,
|
|
||||||
update::receive_update,
|
|
||||||
verify_activity_domains_valid,
|
|
||||||
},
|
|
||||||
check_is_apub_id_valid,
|
|
||||||
ActorType,
|
|
||||||
};
|
|
||||||
use activitystreams::{activity::*, base::AnyBase, prelude::ExtendsExt};
|
|
||||||
use actix_web::HttpResponse;
|
|
||||||
use anyhow::Context;
|
|
||||||
use lemmy_utils::{location_info, LemmyError};
|
|
||||||
use lemmy_websocket::LemmyContext;
|
|
||||||
|
|
||||||
/// Takes an announce and passes the inner activity to the appropriate handler.
|
|
||||||
pub async fn receive_announce(
|
|
||||||
context: &LemmyContext,
|
|
||||||
activity: AnyBase,
|
|
||||||
actor: &dyn ActorType,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let announce = Announce::from_any_base(activity)?.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&announce, actor.actor_id()?, false)?;
|
|
||||||
|
|
||||||
let kind = announce.object().as_single_kind_str();
|
|
||||||
let object = announce
|
|
||||||
.object()
|
|
||||||
.to_owned()
|
|
||||||
.one()
|
|
||||||
.context(location_info!())?;
|
|
||||||
|
|
||||||
let inner_id = object.id().context(location_info!())?.to_owned();
|
|
||||||
check_is_apub_id_valid(&inner_id)?;
|
|
||||||
|
|
||||||
match kind {
|
|
||||||
Some("Create") => receive_create(context, object, inner_id).await,
|
|
||||||
Some("Update") => receive_update(context, object, inner_id).await,
|
|
||||||
Some("Like") => receive_like(context, object, inner_id).await,
|
|
||||||
Some("Dislike") => receive_dislike(context, object, inner_id).await,
|
|
||||||
Some("Delete") => receive_delete(context, object, inner_id).await,
|
|
||||||
Some("Remove") => receive_remove(context, object, inner_id).await,
|
|
||||||
Some("Undo") => receive_undo(context, object, inner_id).await,
|
|
||||||
_ => receive_unhandled_activity(announce),
|
|
||||||
}
|
|
||||||
}
|
|
307
lemmy_apub/src/activities/receive/comment.rs
Normal file
307
lemmy_apub/src/activities/receive/comment.rs
Normal file
|
@ -0,0 +1,307 @@
|
||||||
|
use crate::{
|
||||||
|
activities::receive::{announce_if_community_is_local, get_actor_as_user},
|
||||||
|
fetcher::get_or_fetch_and_insert_comment,
|
||||||
|
ActorType,
|
||||||
|
FromApub,
|
||||||
|
};
|
||||||
|
use activitystreams::{
|
||||||
|
activity::{ActorAndObjectRefExt, Create, Delete, Dislike, Like, Remove, Update},
|
||||||
|
base::ExtendsExt,
|
||||||
|
object::Note,
|
||||||
|
};
|
||||||
|
use actix_web::HttpResponse;
|
||||||
|
use anyhow::Context;
|
||||||
|
use lemmy_db::{
|
||||||
|
comment::{Comment, CommentForm, CommentLike, CommentLikeForm},
|
||||||
|
comment_view::CommentView,
|
||||||
|
post::Post,
|
||||||
|
Crud,
|
||||||
|
Likeable,
|
||||||
|
};
|
||||||
|
use lemmy_structs::{blocking, comment::CommentResponse, send_local_notifs};
|
||||||
|
use lemmy_utils::{location_info, utils::scrape_text_for_mentions, LemmyError};
|
||||||
|
use lemmy_websocket::{messages::SendComment, LemmyContext, UserOperation};
|
||||||
|
|
||||||
|
pub(crate) async fn receive_create_comment(
|
||||||
|
create: Create,
|
||||||
|
context: &LemmyContext,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let user = get_actor_as_user(&create, context).await?;
|
||||||
|
let note = Note::from_any_base(create.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
|
||||||
|
let comment = CommentForm::from_apub(¬e, context, Some(user.actor_id()?)).await?;
|
||||||
|
|
||||||
|
let inserted_comment =
|
||||||
|
blocking(context.pool(), move |conn| Comment::upsert(conn, &comment)).await??;
|
||||||
|
|
||||||
|
let post_id = inserted_comment.post_id;
|
||||||
|
let post = blocking(context.pool(), move |conn| Post::read(conn, post_id)).await??;
|
||||||
|
|
||||||
|
// Note:
|
||||||
|
// Although mentions could be gotten from the post tags (they are included there), or the ccs,
|
||||||
|
// Its much easier to scrape them from the comment body, since the API has to do that
|
||||||
|
// anyway.
|
||||||
|
let mentions = scrape_text_for_mentions(&inserted_comment.content);
|
||||||
|
let recipient_ids = send_local_notifs(
|
||||||
|
mentions,
|
||||||
|
inserted_comment.clone(),
|
||||||
|
&user,
|
||||||
|
post,
|
||||||
|
context.pool(),
|
||||||
|
true,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let comment_view = blocking(context.pool(), move |conn| {
|
||||||
|
CommentView::read(conn, inserted_comment.id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let res = CommentResponse {
|
||||||
|
comment: comment_view,
|
||||||
|
recipient_ids,
|
||||||
|
form_id: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
context.chat_server().do_send(SendComment {
|
||||||
|
op: UserOperation::CreateComment,
|
||||||
|
comment: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
announce_if_community_is_local(create, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_update_comment(
|
||||||
|
update: Update,
|
||||||
|
context: &LemmyContext,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let note = Note::from_any_base(update.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
let user = get_actor_as_user(&update, context).await?;
|
||||||
|
|
||||||
|
let comment = CommentForm::from_apub(¬e, context, Some(user.actor_id()?)).await?;
|
||||||
|
|
||||||
|
let original_comment_id = get_or_fetch_and_insert_comment(&comment.get_ap_id()?, context)
|
||||||
|
.await?
|
||||||
|
.id;
|
||||||
|
|
||||||
|
let updated_comment = blocking(context.pool(), move |conn| {
|
||||||
|
Comment::update(conn, original_comment_id, &comment)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let post_id = updated_comment.post_id;
|
||||||
|
let post = blocking(context.pool(), move |conn| Post::read(conn, post_id)).await??;
|
||||||
|
|
||||||
|
let mentions = scrape_text_for_mentions(&updated_comment.content);
|
||||||
|
let recipient_ids = send_local_notifs(
|
||||||
|
mentions,
|
||||||
|
updated_comment,
|
||||||
|
&user,
|
||||||
|
post,
|
||||||
|
context.pool(),
|
||||||
|
false,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let comment_view = blocking(context.pool(), move |conn| {
|
||||||
|
CommentView::read(conn, original_comment_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let res = CommentResponse {
|
||||||
|
comment: comment_view,
|
||||||
|
recipient_ids,
|
||||||
|
form_id: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
context.chat_server().do_send(SendComment {
|
||||||
|
op: UserOperation::EditComment,
|
||||||
|
comment: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
announce_if_community_is_local(update, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_like_comment(
|
||||||
|
like: Like,
|
||||||
|
context: &LemmyContext,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let note = Note::from_any_base(like.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
let user = get_actor_as_user(&like, context).await?;
|
||||||
|
|
||||||
|
let comment = CommentForm::from_apub(¬e, context, None).await?;
|
||||||
|
|
||||||
|
let comment_id = get_or_fetch_and_insert_comment(&comment.get_ap_id()?, context)
|
||||||
|
.await?
|
||||||
|
.id;
|
||||||
|
|
||||||
|
let like_form = CommentLikeForm {
|
||||||
|
comment_id,
|
||||||
|
post_id: comment.post_id,
|
||||||
|
user_id: user.id,
|
||||||
|
score: 1,
|
||||||
|
};
|
||||||
|
let user_id = user.id;
|
||||||
|
blocking(context.pool(), move |conn| {
|
||||||
|
CommentLike::remove(conn, user_id, comment_id)?;
|
||||||
|
CommentLike::like(conn, &like_form)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let comment_view = blocking(context.pool(), move |conn| {
|
||||||
|
CommentView::read(conn, comment_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// TODO get those recipient actor ids from somewhere
|
||||||
|
let recipient_ids = vec![];
|
||||||
|
let res = CommentResponse {
|
||||||
|
comment: comment_view,
|
||||||
|
recipient_ids,
|
||||||
|
form_id: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
context.chat_server().do_send(SendComment {
|
||||||
|
op: UserOperation::CreateCommentLike,
|
||||||
|
comment: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
announce_if_community_is_local(like, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_dislike_comment(
|
||||||
|
dislike: Dislike,
|
||||||
|
context: &LemmyContext,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let note = Note::from_any_base(
|
||||||
|
dislike
|
||||||
|
.object()
|
||||||
|
.to_owned()
|
||||||
|
.one()
|
||||||
|
.context(location_info!())?,
|
||||||
|
)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
let user = get_actor_as_user(&dislike, context).await?;
|
||||||
|
|
||||||
|
let comment = CommentForm::from_apub(¬e, context, None).await?;
|
||||||
|
|
||||||
|
let comment_id = get_or_fetch_and_insert_comment(&comment.get_ap_id()?, context)
|
||||||
|
.await?
|
||||||
|
.id;
|
||||||
|
|
||||||
|
let like_form = CommentLikeForm {
|
||||||
|
comment_id,
|
||||||
|
post_id: comment.post_id,
|
||||||
|
user_id: user.id,
|
||||||
|
score: -1,
|
||||||
|
};
|
||||||
|
let user_id = user.id;
|
||||||
|
blocking(context.pool(), move |conn| {
|
||||||
|
CommentLike::remove(conn, user_id, comment_id)?;
|
||||||
|
CommentLike::like(conn, &like_form)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let comment_view = blocking(context.pool(), move |conn| {
|
||||||
|
CommentView::read(conn, comment_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// TODO get those recipient actor ids from somewhere
|
||||||
|
let recipient_ids = vec![];
|
||||||
|
let res = CommentResponse {
|
||||||
|
comment: comment_view,
|
||||||
|
recipient_ids,
|
||||||
|
form_id: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
context.chat_server().do_send(SendComment {
|
||||||
|
op: UserOperation::CreateCommentLike,
|
||||||
|
comment: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
announce_if_community_is_local(dislike, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_delete_comment(
|
||||||
|
context: &LemmyContext,
|
||||||
|
delete: Delete,
|
||||||
|
comment: Comment,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let deleted_comment = blocking(context.pool(), move |conn| {
|
||||||
|
Comment::update_deleted(conn, comment.id, true)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let comment_id = deleted_comment.id;
|
||||||
|
let comment_view = blocking(context.pool(), move |conn| {
|
||||||
|
CommentView::read(conn, comment_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// TODO get those recipient actor ids from somewhere
|
||||||
|
let recipient_ids = vec![];
|
||||||
|
let res = CommentResponse {
|
||||||
|
comment: comment_view,
|
||||||
|
recipient_ids,
|
||||||
|
form_id: None,
|
||||||
|
};
|
||||||
|
context.chat_server().do_send(SendComment {
|
||||||
|
op: UserOperation::EditComment,
|
||||||
|
comment: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
let user = get_actor_as_user(&delete, context).await?;
|
||||||
|
announce_if_community_is_local(delete, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_remove_comment(
|
||||||
|
context: &LemmyContext,
|
||||||
|
_remove: Remove,
|
||||||
|
comment: Comment,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let removed_comment = blocking(context.pool(), move |conn| {
|
||||||
|
Comment::update_removed(conn, comment.id, true)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let comment_id = removed_comment.id;
|
||||||
|
let comment_view = blocking(context.pool(), move |conn| {
|
||||||
|
CommentView::read(conn, comment_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// TODO get those recipient actor ids from somewhere
|
||||||
|
let recipient_ids = vec![];
|
||||||
|
let res = CommentResponse {
|
||||||
|
comment: comment_view,
|
||||||
|
recipient_ids,
|
||||||
|
form_id: None,
|
||||||
|
};
|
||||||
|
context.chat_server().do_send(SendComment {
|
||||||
|
op: UserOperation::EditComment,
|
||||||
|
comment: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
130
lemmy_apub/src/activities/receive/community.rs
Normal file
130
lemmy_apub/src/activities/receive/community.rs
Normal file
|
@ -0,0 +1,130 @@
|
||||||
|
use crate::activities::receive::{announce_if_community_is_local, get_actor_as_user};
|
||||||
|
use activitystreams::activity::{Delete, Remove, Undo};
|
||||||
|
use actix_web::HttpResponse;
|
||||||
|
use lemmy_db::{community::Community, community_view::CommunityView};
|
||||||
|
use lemmy_structs::{blocking, community::CommunityResponse};
|
||||||
|
use lemmy_utils::LemmyError;
|
||||||
|
use lemmy_websocket::{messages::SendCommunityRoomMessage, LemmyContext, UserOperation};
|
||||||
|
|
||||||
|
pub(crate) async fn receive_delete_community(
|
||||||
|
context: &LemmyContext,
|
||||||
|
delete: Delete,
|
||||||
|
community: Community,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let deleted_community = blocking(context.pool(), move |conn| {
|
||||||
|
Community::update_deleted(conn, community.id, true)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let community_id = deleted_community.id;
|
||||||
|
let res = CommunityResponse {
|
||||||
|
community: blocking(context.pool(), move |conn| {
|
||||||
|
CommunityView::read(conn, community_id, None)
|
||||||
|
})
|
||||||
|
.await??,
|
||||||
|
};
|
||||||
|
|
||||||
|
let community_id = res.community.id;
|
||||||
|
context.chat_server().do_send(SendCommunityRoomMessage {
|
||||||
|
op: UserOperation::EditCommunity,
|
||||||
|
response: res,
|
||||||
|
community_id,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
let user = get_actor_as_user(&delete, context).await?;
|
||||||
|
announce_if_community_is_local(delete, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_remove_community(
|
||||||
|
context: &LemmyContext,
|
||||||
|
_remove: Remove,
|
||||||
|
community: Community,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let removed_community = blocking(context.pool(), move |conn| {
|
||||||
|
Community::update_removed(conn, community.id, true)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let community_id = removed_community.id;
|
||||||
|
let res = CommunityResponse {
|
||||||
|
community: blocking(context.pool(), move |conn| {
|
||||||
|
CommunityView::read(conn, community_id, None)
|
||||||
|
})
|
||||||
|
.await??,
|
||||||
|
};
|
||||||
|
|
||||||
|
let community_id = res.community.id;
|
||||||
|
context.chat_server().do_send(SendCommunityRoomMessage {
|
||||||
|
op: UserOperation::EditCommunity,
|
||||||
|
response: res,
|
||||||
|
community_id,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_undo_delete_community(
|
||||||
|
context: &LemmyContext,
|
||||||
|
undo: Undo,
|
||||||
|
community: Community,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let deleted_community = blocking(context.pool(), move |conn| {
|
||||||
|
Community::update_deleted(conn, community.id, false)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let community_id = deleted_community.id;
|
||||||
|
let res = CommunityResponse {
|
||||||
|
community: blocking(context.pool(), move |conn| {
|
||||||
|
CommunityView::read(conn, community_id, None)
|
||||||
|
})
|
||||||
|
.await??,
|
||||||
|
};
|
||||||
|
|
||||||
|
let community_id = res.community.id;
|
||||||
|
context.chat_server().do_send(SendCommunityRoomMessage {
|
||||||
|
op: UserOperation::EditCommunity,
|
||||||
|
response: res,
|
||||||
|
community_id,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
let user = get_actor_as_user(&undo, context).await?;
|
||||||
|
announce_if_community_is_local(undo, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_undo_remove_community(
|
||||||
|
context: &LemmyContext,
|
||||||
|
undo: Undo,
|
||||||
|
community: Community,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let removed_community = blocking(context.pool(), move |conn| {
|
||||||
|
Community::update_removed(conn, community.id, false)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let community_id = removed_community.id;
|
||||||
|
let res = CommunityResponse {
|
||||||
|
community: blocking(context.pool(), move |conn| {
|
||||||
|
CommunityView::read(conn, community_id, None)
|
||||||
|
})
|
||||||
|
.await??,
|
||||||
|
};
|
||||||
|
|
||||||
|
let community_id = res.community.id;
|
||||||
|
|
||||||
|
context.chat_server().do_send(SendCommunityRoomMessage {
|
||||||
|
op: UserOperation::EditCommunity,
|
||||||
|
response: res,
|
||||||
|
community_id,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
let mod_ = get_actor_as_user(&undo, context).await?;
|
||||||
|
announce_if_community_is_local(undo, &mod_, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
|
@ -1,129 +0,0 @@
|
||||||
use crate::{
|
|
||||||
activities::receive::{
|
|
||||||
announce_if_community_is_local,
|
|
||||||
get_actor_as_user,
|
|
||||||
receive_unhandled_activity,
|
|
||||||
verify_activity_domains_valid,
|
|
||||||
},
|
|
||||||
ActorType,
|
|
||||||
FromApub,
|
|
||||||
PageExt,
|
|
||||||
};
|
|
||||||
use activitystreams::{activity::Create, base::AnyBase, object::Note, prelude::*};
|
|
||||||
use actix_web::HttpResponse;
|
|
||||||
use anyhow::Context;
|
|
||||||
use lemmy_db::{
|
|
||||||
comment::{Comment, CommentForm},
|
|
||||||
comment_view::CommentView,
|
|
||||||
post::{Post, PostForm},
|
|
||||||
post_view::PostView,
|
|
||||||
};
|
|
||||||
use lemmy_structs::{blocking, comment::CommentResponse, post::PostResponse, send_local_notifs};
|
|
||||||
use lemmy_utils::{location_info, utils::scrape_text_for_mentions, LemmyError};
|
|
||||||
use lemmy_websocket::{
|
|
||||||
messages::{SendComment, SendPost},
|
|
||||||
LemmyContext,
|
|
||||||
UserOperation,
|
|
||||||
};
|
|
||||||
use url::Url;
|
|
||||||
|
|
||||||
pub async fn receive_create(
|
|
||||||
context: &LemmyContext,
|
|
||||||
activity: AnyBase,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let create = Create::from_any_base(activity)?.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&create, expected_domain, true)?;
|
|
||||||
|
|
||||||
match create.object().as_single_kind_str() {
|
|
||||||
Some("Page") => receive_create_post(create, context).await,
|
|
||||||
Some("Note") => receive_create_comment(create, context).await,
|
|
||||||
_ => receive_unhandled_activity(create),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_create_post(
|
|
||||||
create: Create,
|
|
||||||
context: &LemmyContext,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let user = get_actor_as_user(&create, context).await?;
|
|
||||||
let page = PageExt::from_any_base(create.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
|
|
||||||
let post = PostForm::from_apub(&page, context, Some(user.actor_id()?)).await?;
|
|
||||||
|
|
||||||
// Using an upsert, since likes (which fetch the post), sometimes come in before the create
|
|
||||||
// resulting in double posts.
|
|
||||||
let inserted_post = blocking(context.pool(), move |conn| Post::upsert(conn, &post)).await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let inserted_post_id = inserted_post.id;
|
|
||||||
let post_view = blocking(context.pool(), move |conn| {
|
|
||||||
PostView::read(conn, inserted_post_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let res = PostResponse { post: post_view };
|
|
||||||
|
|
||||||
context.chat_server().do_send(SendPost {
|
|
||||||
op: UserOperation::CreatePost,
|
|
||||||
post: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
announce_if_community_is_local(create, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_create_comment(
|
|
||||||
create: Create,
|
|
||||||
context: &LemmyContext,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let user = get_actor_as_user(&create, context).await?;
|
|
||||||
let note = Note::from_any_base(create.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
|
|
||||||
let comment = CommentForm::from_apub(¬e, context, Some(user.actor_id()?)).await?;
|
|
||||||
|
|
||||||
let inserted_comment =
|
|
||||||
blocking(context.pool(), move |conn| Comment::upsert(conn, &comment)).await??;
|
|
||||||
|
|
||||||
let post_id = inserted_comment.post_id;
|
|
||||||
let post = blocking(context.pool(), move |conn| Post::read(conn, post_id)).await??;
|
|
||||||
|
|
||||||
// Note:
|
|
||||||
// Although mentions could be gotten from the post tags (they are included there), or the ccs,
|
|
||||||
// Its much easier to scrape them from the comment body, since the API has to do that
|
|
||||||
// anyway.
|
|
||||||
let mentions = scrape_text_for_mentions(&inserted_comment.content);
|
|
||||||
let recipient_ids = send_local_notifs(
|
|
||||||
mentions,
|
|
||||||
inserted_comment.clone(),
|
|
||||||
&user,
|
|
||||||
post,
|
|
||||||
context.pool(),
|
|
||||||
true,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let comment_view = blocking(context.pool(), move |conn| {
|
|
||||||
CommentView::read(conn, inserted_comment.id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let res = CommentResponse {
|
|
||||||
comment: comment_view,
|
|
||||||
recipient_ids,
|
|
||||||
form_id: None,
|
|
||||||
};
|
|
||||||
|
|
||||||
context.chat_server().do_send(SendComment {
|
|
||||||
op: UserOperation::CreateComment,
|
|
||||||
comment: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
announce_if_community_is_local(create, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
|
@ -1,149 +0,0 @@
|
||||||
use crate::activities::receive::{
|
|
||||||
announce_if_community_is_local,
|
|
||||||
find_by_id,
|
|
||||||
get_actor_as_user,
|
|
||||||
verify_activity_domains_valid,
|
|
||||||
FindResults,
|
|
||||||
};
|
|
||||||
use activitystreams::{activity::Delete, base::AnyBase, prelude::*};
|
|
||||||
use actix_web::HttpResponse;
|
|
||||||
use anyhow::Context;
|
|
||||||
use lemmy_db::{
|
|
||||||
comment::Comment,
|
|
||||||
comment_view::CommentView,
|
|
||||||
community::Community,
|
|
||||||
community_view::CommunityView,
|
|
||||||
post::Post,
|
|
||||||
post_view::PostView,
|
|
||||||
};
|
|
||||||
use lemmy_structs::{
|
|
||||||
blocking,
|
|
||||||
comment::CommentResponse,
|
|
||||||
community::CommunityResponse,
|
|
||||||
post::PostResponse,
|
|
||||||
};
|
|
||||||
use lemmy_utils::{location_info, LemmyError};
|
|
||||||
use lemmy_websocket::{
|
|
||||||
messages::{SendComment, SendCommunityRoomMessage, SendPost},
|
|
||||||
LemmyContext,
|
|
||||||
UserOperation,
|
|
||||||
};
|
|
||||||
use url::Url;
|
|
||||||
|
|
||||||
pub async fn receive_delete(
|
|
||||||
context: &LemmyContext,
|
|
||||||
activity: AnyBase,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let delete = Delete::from_any_base(activity)?.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&delete, expected_domain, true)?;
|
|
||||||
|
|
||||||
let object = delete
|
|
||||||
.object()
|
|
||||||
.to_owned()
|
|
||||||
.single_xsd_any_uri()
|
|
||||||
.context(location_info!())?;
|
|
||||||
|
|
||||||
match find_by_id(context, object).await {
|
|
||||||
Ok(FindResults::Post(p)) => receive_delete_post(context, delete, p).await,
|
|
||||||
Ok(FindResults::Comment(c)) => receive_delete_comment(context, delete, c).await,
|
|
||||||
Ok(FindResults::Community(c)) => receive_delete_community(context, delete, c).await,
|
|
||||||
// if we dont have the object, no need to do anything
|
|
||||||
Err(_) => Ok(HttpResponse::Ok().finish()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_delete_post(
|
|
||||||
context: &LemmyContext,
|
|
||||||
delete: Delete,
|
|
||||||
post: Post,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let deleted_post = blocking(context.pool(), move |conn| {
|
|
||||||
Post::update_deleted(conn, post.id, true)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let post_id = deleted_post.id;
|
|
||||||
let post_view = blocking(context.pool(), move |conn| {
|
|
||||||
PostView::read(conn, post_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let res = PostResponse { post: post_view };
|
|
||||||
context.chat_server().do_send(SendPost {
|
|
||||||
op: UserOperation::EditPost,
|
|
||||||
post: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
let user = get_actor_as_user(&delete, context).await?;
|
|
||||||
announce_if_community_is_local(delete, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_delete_comment(
|
|
||||||
context: &LemmyContext,
|
|
||||||
delete: Delete,
|
|
||||||
comment: Comment,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let deleted_comment = blocking(context.pool(), move |conn| {
|
|
||||||
Comment::update_deleted(conn, comment.id, true)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let comment_id = deleted_comment.id;
|
|
||||||
let comment_view = blocking(context.pool(), move |conn| {
|
|
||||||
CommentView::read(conn, comment_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// TODO get those recipient actor ids from somewhere
|
|
||||||
let recipient_ids = vec![];
|
|
||||||
let res = CommentResponse {
|
|
||||||
comment: comment_view,
|
|
||||||
recipient_ids,
|
|
||||||
form_id: None,
|
|
||||||
};
|
|
||||||
context.chat_server().do_send(SendComment {
|
|
||||||
op: UserOperation::EditComment,
|
|
||||||
comment: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
let user = get_actor_as_user(&delete, context).await?;
|
|
||||||
announce_if_community_is_local(delete, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_delete_community(
|
|
||||||
context: &LemmyContext,
|
|
||||||
delete: Delete,
|
|
||||||
community: Community,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let deleted_community = blocking(context.pool(), move |conn| {
|
|
||||||
Community::update_deleted(conn, community.id, true)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let community_id = deleted_community.id;
|
|
||||||
let res = CommunityResponse {
|
|
||||||
community: blocking(context.pool(), move |conn| {
|
|
||||||
CommunityView::read(conn, community_id, None)
|
|
||||||
})
|
|
||||||
.await??,
|
|
||||||
};
|
|
||||||
|
|
||||||
let community_id = res.community.id;
|
|
||||||
context.chat_server().do_send(SendCommunityRoomMessage {
|
|
||||||
op: UserOperation::EditCommunity,
|
|
||||||
response: res,
|
|
||||||
community_id,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
let user = get_actor_as_user(&delete, context).await?;
|
|
||||||
announce_if_community_is_local(delete, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
|
@ -1,161 +0,0 @@
|
||||||
use crate::{
|
|
||||||
activities::receive::{
|
|
||||||
announce_if_community_is_local,
|
|
||||||
get_actor_as_user,
|
|
||||||
receive_unhandled_activity,
|
|
||||||
verify_activity_domains_valid,
|
|
||||||
},
|
|
||||||
fetcher::{get_or_fetch_and_insert_comment, get_or_fetch_and_insert_post},
|
|
||||||
FromApub,
|
|
||||||
PageExt,
|
|
||||||
};
|
|
||||||
use activitystreams::{activity::Dislike, base::AnyBase, object::Note, prelude::*};
|
|
||||||
use actix_web::HttpResponse;
|
|
||||||
use anyhow::Context;
|
|
||||||
use lemmy_db::{
|
|
||||||
comment::{CommentForm, CommentLike, CommentLikeForm},
|
|
||||||
comment_view::CommentView,
|
|
||||||
post::{PostForm, PostLike, PostLikeForm},
|
|
||||||
post_view::PostView,
|
|
||||||
site::Site,
|
|
||||||
Crud,
|
|
||||||
Likeable,
|
|
||||||
};
|
|
||||||
use lemmy_structs::{blocking, comment::CommentResponse, post::PostResponse};
|
|
||||||
use lemmy_utils::{location_info, LemmyError};
|
|
||||||
use lemmy_websocket::{
|
|
||||||
messages::{SendComment, SendPost},
|
|
||||||
LemmyContext,
|
|
||||||
UserOperation,
|
|
||||||
};
|
|
||||||
use url::Url;
|
|
||||||
|
|
||||||
pub async fn receive_dislike(
|
|
||||||
context: &LemmyContext,
|
|
||||||
activity: AnyBase,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let enable_downvotes = blocking(context.pool(), move |conn| {
|
|
||||||
Site::read(conn, 1).map(|s| s.enable_downvotes)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
if !enable_downvotes {
|
|
||||||
return Ok(HttpResponse::Ok().finish());
|
|
||||||
}
|
|
||||||
|
|
||||||
let dislike = Dislike::from_any_base(activity)?.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&dislike, expected_domain, false)?;
|
|
||||||
|
|
||||||
match dislike.object().as_single_kind_str() {
|
|
||||||
Some("Page") => receive_dislike_post(dislike, context).await,
|
|
||||||
Some("Note") => receive_dislike_comment(dislike, context).await,
|
|
||||||
_ => receive_unhandled_activity(dislike),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_dislike_post(
|
|
||||||
dislike: Dislike,
|
|
||||||
context: &LemmyContext,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let user = get_actor_as_user(&dislike, context).await?;
|
|
||||||
let page = PageExt::from_any_base(
|
|
||||||
dislike
|
|
||||||
.object()
|
|
||||||
.to_owned()
|
|
||||||
.one()
|
|
||||||
.context(location_info!())?,
|
|
||||||
)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
|
|
||||||
let post = PostForm::from_apub(&page, context, None).await?;
|
|
||||||
|
|
||||||
let post_id = get_or_fetch_and_insert_post(&post.get_ap_id()?, context)
|
|
||||||
.await?
|
|
||||||
.id;
|
|
||||||
|
|
||||||
let like_form = PostLikeForm {
|
|
||||||
post_id,
|
|
||||||
user_id: user.id,
|
|
||||||
score: -1,
|
|
||||||
};
|
|
||||||
let user_id = user.id;
|
|
||||||
blocking(context.pool(), move |conn| {
|
|
||||||
PostLike::remove(conn, user_id, post_id)?;
|
|
||||||
PostLike::like(conn, &like_form)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let post_view = blocking(context.pool(), move |conn| {
|
|
||||||
PostView::read(conn, post_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let res = PostResponse { post: post_view };
|
|
||||||
|
|
||||||
context.chat_server().do_send(SendPost {
|
|
||||||
op: UserOperation::CreatePostLike,
|
|
||||||
post: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
announce_if_community_is_local(dislike, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_dislike_comment(
|
|
||||||
dislike: Dislike,
|
|
||||||
context: &LemmyContext,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let note = Note::from_any_base(
|
|
||||||
dislike
|
|
||||||
.object()
|
|
||||||
.to_owned()
|
|
||||||
.one()
|
|
||||||
.context(location_info!())?,
|
|
||||||
)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
let user = get_actor_as_user(&dislike, context).await?;
|
|
||||||
|
|
||||||
let comment = CommentForm::from_apub(¬e, context, None).await?;
|
|
||||||
|
|
||||||
let comment_id = get_or_fetch_and_insert_comment(&comment.get_ap_id()?, context)
|
|
||||||
.await?
|
|
||||||
.id;
|
|
||||||
|
|
||||||
let like_form = CommentLikeForm {
|
|
||||||
comment_id,
|
|
||||||
post_id: comment.post_id,
|
|
||||||
user_id: user.id,
|
|
||||||
score: -1,
|
|
||||||
};
|
|
||||||
let user_id = user.id;
|
|
||||||
blocking(context.pool(), move |conn| {
|
|
||||||
CommentLike::remove(conn, user_id, comment_id)?;
|
|
||||||
CommentLike::like(conn, &like_form)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let comment_view = blocking(context.pool(), move |conn| {
|
|
||||||
CommentView::read(conn, comment_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// TODO get those recipient actor ids from somewhere
|
|
||||||
let recipient_ids = vec![];
|
|
||||||
let res = CommentResponse {
|
|
||||||
comment: comment_view,
|
|
||||||
recipient_ids,
|
|
||||||
form_id: None,
|
|
||||||
};
|
|
||||||
|
|
||||||
context.chat_server().do_send(SendComment {
|
|
||||||
op: UserOperation::CreateCommentLike,
|
|
||||||
comment: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
announce_if_community_is_local(dislike, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
|
@ -1,136 +0,0 @@
|
||||||
use crate::{
|
|
||||||
activities::receive::{
|
|
||||||
announce_if_community_is_local,
|
|
||||||
get_actor_as_user,
|
|
||||||
receive_unhandled_activity,
|
|
||||||
verify_activity_domains_valid,
|
|
||||||
},
|
|
||||||
fetcher::{get_or_fetch_and_insert_comment, get_or_fetch_and_insert_post},
|
|
||||||
FromApub,
|
|
||||||
PageExt,
|
|
||||||
};
|
|
||||||
use activitystreams::{activity::Like, base::AnyBase, object::Note, prelude::*};
|
|
||||||
use actix_web::HttpResponse;
|
|
||||||
use anyhow::Context;
|
|
||||||
use lemmy_db::{
|
|
||||||
comment::{CommentForm, CommentLike, CommentLikeForm},
|
|
||||||
comment_view::CommentView,
|
|
||||||
post::{PostForm, PostLike, PostLikeForm},
|
|
||||||
post_view::PostView,
|
|
||||||
Likeable,
|
|
||||||
};
|
|
||||||
use lemmy_structs::{blocking, comment::CommentResponse, post::PostResponse};
|
|
||||||
use lemmy_utils::{location_info, LemmyError};
|
|
||||||
use lemmy_websocket::{
|
|
||||||
messages::{SendComment, SendPost},
|
|
||||||
LemmyContext,
|
|
||||||
UserOperation,
|
|
||||||
};
|
|
||||||
use url::Url;
|
|
||||||
|
|
||||||
pub async fn receive_like(
|
|
||||||
context: &LemmyContext,
|
|
||||||
activity: AnyBase,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let like = Like::from_any_base(activity)?.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&like, expected_domain, false)?;
|
|
||||||
|
|
||||||
match like.object().as_single_kind_str() {
|
|
||||||
Some("Page") => receive_like_post(like, context).await,
|
|
||||||
Some("Note") => receive_like_comment(like, context).await,
|
|
||||||
_ => receive_unhandled_activity(like),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_like_post(like: Like, context: &LemmyContext) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let user = get_actor_as_user(&like, context).await?;
|
|
||||||
let page = PageExt::from_any_base(like.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
|
|
||||||
let post = PostForm::from_apub(&page, context, None).await?;
|
|
||||||
|
|
||||||
let post_id = get_or_fetch_and_insert_post(&post.get_ap_id()?, context)
|
|
||||||
.await?
|
|
||||||
.id;
|
|
||||||
|
|
||||||
let like_form = PostLikeForm {
|
|
||||||
post_id,
|
|
||||||
user_id: user.id,
|
|
||||||
score: 1,
|
|
||||||
};
|
|
||||||
let user_id = user.id;
|
|
||||||
blocking(context.pool(), move |conn| {
|
|
||||||
PostLike::remove(conn, user_id, post_id)?;
|
|
||||||
PostLike::like(conn, &like_form)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let post_view = blocking(context.pool(), move |conn| {
|
|
||||||
PostView::read(conn, post_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let res = PostResponse { post: post_view };
|
|
||||||
|
|
||||||
context.chat_server().do_send(SendPost {
|
|
||||||
op: UserOperation::CreatePostLike,
|
|
||||||
post: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
announce_if_community_is_local(like, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_like_comment(
|
|
||||||
like: Like,
|
|
||||||
context: &LemmyContext,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let note = Note::from_any_base(like.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
let user = get_actor_as_user(&like, context).await?;
|
|
||||||
|
|
||||||
let comment = CommentForm::from_apub(¬e, context, None).await?;
|
|
||||||
|
|
||||||
let comment_id = get_or_fetch_and_insert_comment(&comment.get_ap_id()?, context)
|
|
||||||
.await?
|
|
||||||
.id;
|
|
||||||
|
|
||||||
let like_form = CommentLikeForm {
|
|
||||||
comment_id,
|
|
||||||
post_id: comment.post_id,
|
|
||||||
user_id: user.id,
|
|
||||||
score: 1,
|
|
||||||
};
|
|
||||||
let user_id = user.id;
|
|
||||||
blocking(context.pool(), move |conn| {
|
|
||||||
CommentLike::remove(conn, user_id, comment_id)?;
|
|
||||||
CommentLike::like(conn, &like_form)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let comment_view = blocking(context.pool(), move |conn| {
|
|
||||||
CommentView::read(conn, comment_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// TODO get those recipient actor ids from somewhere
|
|
||||||
let recipient_ids = vec![];
|
|
||||||
let res = CommentResponse {
|
|
||||||
comment: comment_view,
|
|
||||||
recipient_ids,
|
|
||||||
form_id: None,
|
|
||||||
};
|
|
||||||
|
|
||||||
context.chat_server().do_send(SendComment {
|
|
||||||
op: UserOperation::CreateCommentLike,
|
|
||||||
comment: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
announce_if_community_is_local(like, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
|
@ -20,19 +20,14 @@ use serde::Serialize;
|
||||||
use std::fmt::Debug;
|
use std::fmt::Debug;
|
||||||
use url::Url;
|
use url::Url;
|
||||||
|
|
||||||
pub mod announce;
|
pub(crate) mod comment;
|
||||||
pub mod create;
|
pub(crate) mod comment_undo;
|
||||||
pub mod delete;
|
pub(crate) mod community;
|
||||||
pub mod dislike;
|
pub(crate) mod post;
|
||||||
pub mod like;
|
pub(crate) mod post_undo;
|
||||||
pub mod remove;
|
|
||||||
pub mod undo;
|
|
||||||
mod undo_comment;
|
|
||||||
mod undo_post;
|
|
||||||
pub mod update;
|
|
||||||
|
|
||||||
/// Return HTTP 501 for unsupported activities in inbox.
|
/// Return HTTP 501 for unsupported activities in inbox.
|
||||||
fn receive_unhandled_activity<A>(activity: A) -> Result<HttpResponse, LemmyError>
|
pub(crate) fn receive_unhandled_activity<A>(activity: A) -> Result<HttpResponse, LemmyError>
|
||||||
where
|
where
|
||||||
A: Debug,
|
A: Debug,
|
||||||
{
|
{
|
||||||
|
|
242
lemmy_apub/src/activities/receive/post.rs
Normal file
242
lemmy_apub/src/activities/receive/post.rs
Normal file
|
@ -0,0 +1,242 @@
|
||||||
|
use crate::{
|
||||||
|
activities::receive::{announce_if_community_is_local, get_actor_as_user},
|
||||||
|
fetcher::get_or_fetch_and_insert_post,
|
||||||
|
ActorType,
|
||||||
|
FromApub,
|
||||||
|
PageExt,
|
||||||
|
};
|
||||||
|
use activitystreams::{
|
||||||
|
activity::{Create, Delete, Dislike, Like, Remove, Update},
|
||||||
|
prelude::*,
|
||||||
|
};
|
||||||
|
use actix_web::HttpResponse;
|
||||||
|
use anyhow::Context;
|
||||||
|
use lemmy_db::{
|
||||||
|
post::{Post, PostForm, PostLike, PostLikeForm},
|
||||||
|
post_view::PostView,
|
||||||
|
Crud,
|
||||||
|
Likeable,
|
||||||
|
};
|
||||||
|
use lemmy_structs::{blocking, post::PostResponse};
|
||||||
|
use lemmy_utils::{location_info, LemmyError};
|
||||||
|
use lemmy_websocket::{messages::SendPost, LemmyContext, UserOperation};
|
||||||
|
|
||||||
|
pub(crate) async fn receive_create_post(
|
||||||
|
create: Create,
|
||||||
|
context: &LemmyContext,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let user = get_actor_as_user(&create, context).await?;
|
||||||
|
let page = PageExt::from_any_base(create.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
|
||||||
|
let post = PostForm::from_apub(&page, context, Some(user.actor_id()?)).await?;
|
||||||
|
|
||||||
|
// Using an upsert, since likes (which fetch the post), sometimes come in before the create
|
||||||
|
// resulting in double posts.
|
||||||
|
let inserted_post = blocking(context.pool(), move |conn| Post::upsert(conn, &post)).await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let inserted_post_id = inserted_post.id;
|
||||||
|
let post_view = blocking(context.pool(), move |conn| {
|
||||||
|
PostView::read(conn, inserted_post_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let res = PostResponse { post: post_view };
|
||||||
|
|
||||||
|
context.chat_server().do_send(SendPost {
|
||||||
|
op: UserOperation::CreatePost,
|
||||||
|
post: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
announce_if_community_is_local(create, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_update_post(
|
||||||
|
update: Update,
|
||||||
|
context: &LemmyContext,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let user = get_actor_as_user(&update, context).await?;
|
||||||
|
let page = PageExt::from_any_base(update.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
|
||||||
|
let post = PostForm::from_apub(&page, context, Some(user.actor_id()?)).await?;
|
||||||
|
|
||||||
|
let original_post_id = get_or_fetch_and_insert_post(&post.get_ap_id()?, context)
|
||||||
|
.await?
|
||||||
|
.id;
|
||||||
|
|
||||||
|
blocking(context.pool(), move |conn| {
|
||||||
|
Post::update(conn, original_post_id, &post)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let post_view = blocking(context.pool(), move |conn| {
|
||||||
|
PostView::read(conn, original_post_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let res = PostResponse { post: post_view };
|
||||||
|
|
||||||
|
context.chat_server().do_send(SendPost {
|
||||||
|
op: UserOperation::EditPost,
|
||||||
|
post: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
announce_if_community_is_local(update, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_like_post(
|
||||||
|
like: Like,
|
||||||
|
context: &LemmyContext,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let user = get_actor_as_user(&like, context).await?;
|
||||||
|
let page = PageExt::from_any_base(like.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
|
||||||
|
let post = PostForm::from_apub(&page, context, None).await?;
|
||||||
|
|
||||||
|
let post_id = get_or_fetch_and_insert_post(&post.get_ap_id()?, context)
|
||||||
|
.await?
|
||||||
|
.id;
|
||||||
|
|
||||||
|
let like_form = PostLikeForm {
|
||||||
|
post_id,
|
||||||
|
user_id: user.id,
|
||||||
|
score: 1,
|
||||||
|
};
|
||||||
|
let user_id = user.id;
|
||||||
|
blocking(context.pool(), move |conn| {
|
||||||
|
PostLike::remove(conn, user_id, post_id)?;
|
||||||
|
PostLike::like(conn, &like_form)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let post_view = blocking(context.pool(), move |conn| {
|
||||||
|
PostView::read(conn, post_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let res = PostResponse { post: post_view };
|
||||||
|
|
||||||
|
context.chat_server().do_send(SendPost {
|
||||||
|
op: UserOperation::CreatePostLike,
|
||||||
|
post: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
announce_if_community_is_local(like, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_dislike_post(
|
||||||
|
dislike: Dislike,
|
||||||
|
context: &LemmyContext,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let user = get_actor_as_user(&dislike, context).await?;
|
||||||
|
let page = PageExt::from_any_base(
|
||||||
|
dislike
|
||||||
|
.object()
|
||||||
|
.to_owned()
|
||||||
|
.one()
|
||||||
|
.context(location_info!())?,
|
||||||
|
)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
|
||||||
|
let post = PostForm::from_apub(&page, context, None).await?;
|
||||||
|
|
||||||
|
let post_id = get_or_fetch_and_insert_post(&post.get_ap_id()?, context)
|
||||||
|
.await?
|
||||||
|
.id;
|
||||||
|
|
||||||
|
let like_form = PostLikeForm {
|
||||||
|
post_id,
|
||||||
|
user_id: user.id,
|
||||||
|
score: -1,
|
||||||
|
};
|
||||||
|
let user_id = user.id;
|
||||||
|
blocking(context.pool(), move |conn| {
|
||||||
|
PostLike::remove(conn, user_id, post_id)?;
|
||||||
|
PostLike::like(conn, &like_form)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let post_view = blocking(context.pool(), move |conn| {
|
||||||
|
PostView::read(conn, post_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let res = PostResponse { post: post_view };
|
||||||
|
|
||||||
|
context.chat_server().do_send(SendPost {
|
||||||
|
op: UserOperation::CreatePostLike,
|
||||||
|
post: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
announce_if_community_is_local(dislike, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_delete_post(
|
||||||
|
context: &LemmyContext,
|
||||||
|
delete: Delete,
|
||||||
|
post: Post,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let deleted_post = blocking(context.pool(), move |conn| {
|
||||||
|
Post::update_deleted(conn, post.id, true)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let post_id = deleted_post.id;
|
||||||
|
let post_view = blocking(context.pool(), move |conn| {
|
||||||
|
PostView::read(conn, post_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let res = PostResponse { post: post_view };
|
||||||
|
context.chat_server().do_send(SendPost {
|
||||||
|
op: UserOperation::EditPost,
|
||||||
|
post: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
let user = get_actor_as_user(&delete, context).await?;
|
||||||
|
announce_if_community_is_local(delete, &user, context).await?;
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn receive_remove_post(
|
||||||
|
context: &LemmyContext,
|
||||||
|
_remove: Remove,
|
||||||
|
post: Post,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let removed_post = blocking(context.pool(), move |conn| {
|
||||||
|
Post::update_removed(conn, post.id, true)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
// Refetch the view
|
||||||
|
let post_id = removed_post.id;
|
||||||
|
let post_view = blocking(context.pool(), move |conn| {
|
||||||
|
PostView::read(conn, post_id, None)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
|
||||||
|
let res = PostResponse { post: post_view };
|
||||||
|
context.chat_server().do_send(SendPost {
|
||||||
|
op: UserOperation::EditPost,
|
||||||
|
post: res,
|
||||||
|
websocket_id: None,
|
||||||
|
});
|
||||||
|
|
||||||
|
Ok(HttpResponse::Ok().finish())
|
||||||
|
}
|
|
@ -1,151 +0,0 @@
|
||||||
use crate::activities::receive::{find_by_id, verify_activity_domains_valid, FindResults};
|
|
||||||
use activitystreams::{activity::Remove, base::AnyBase, prelude::*};
|
|
||||||
use actix_web::HttpResponse;
|
|
||||||
use anyhow::Context;
|
|
||||||
use lemmy_db::{
|
|
||||||
comment::Comment,
|
|
||||||
comment_view::CommentView,
|
|
||||||
community::Community,
|
|
||||||
community_view::CommunityView,
|
|
||||||
post::Post,
|
|
||||||
post_view::PostView,
|
|
||||||
};
|
|
||||||
use lemmy_structs::{
|
|
||||||
blocking,
|
|
||||||
comment::CommentResponse,
|
|
||||||
community::CommunityResponse,
|
|
||||||
post::PostResponse,
|
|
||||||
};
|
|
||||||
use lemmy_utils::{location_info, LemmyError};
|
|
||||||
use lemmy_websocket::{
|
|
||||||
messages::{SendComment, SendCommunityRoomMessage, SendPost},
|
|
||||||
LemmyContext,
|
|
||||||
UserOperation,
|
|
||||||
};
|
|
||||||
use url::Url;
|
|
||||||
|
|
||||||
pub async fn receive_remove(
|
|
||||||
context: &LemmyContext,
|
|
||||||
activity: AnyBase,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let remove = Remove::from_any_base(activity)?.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&remove, expected_domain, false)?;
|
|
||||||
|
|
||||||
let cc = remove
|
|
||||||
.cc()
|
|
||||||
.map(|c| c.as_many())
|
|
||||||
.flatten()
|
|
||||||
.context(location_info!())?;
|
|
||||||
let community_id = cc
|
|
||||||
.first()
|
|
||||||
.map(|c| c.as_xsd_any_uri())
|
|
||||||
.flatten()
|
|
||||||
.context(location_info!())?;
|
|
||||||
|
|
||||||
let object = remove
|
|
||||||
.object()
|
|
||||||
.to_owned()
|
|
||||||
.single_xsd_any_uri()
|
|
||||||
.context(location_info!())?;
|
|
||||||
|
|
||||||
// Ensure that remove activity comes from the same domain as the community
|
|
||||||
remove.id(community_id.domain().context(location_info!())?)?;
|
|
||||||
|
|
||||||
match find_by_id(context, object).await {
|
|
||||||
Ok(FindResults::Post(p)) => receive_remove_post(context, remove, p).await,
|
|
||||||
Ok(FindResults::Comment(c)) => receive_remove_comment(context, remove, c).await,
|
|
||||||
Ok(FindResults::Community(c)) => receive_remove_community(context, remove, c).await,
|
|
||||||
// if we dont have the object, no need to do anything
|
|
||||||
Err(_) => Ok(HttpResponse::Ok().finish()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_remove_post(
|
|
||||||
context: &LemmyContext,
|
|
||||||
_remove: Remove,
|
|
||||||
post: Post,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let removed_post = blocking(context.pool(), move |conn| {
|
|
||||||
Post::update_removed(conn, post.id, true)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let post_id = removed_post.id;
|
|
||||||
let post_view = blocking(context.pool(), move |conn| {
|
|
||||||
PostView::read(conn, post_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let res = PostResponse { post: post_view };
|
|
||||||
context.chat_server().do_send(SendPost {
|
|
||||||
op: UserOperation::EditPost,
|
|
||||||
post: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_remove_comment(
|
|
||||||
context: &LemmyContext,
|
|
||||||
_remove: Remove,
|
|
||||||
comment: Comment,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let removed_comment = blocking(context.pool(), move |conn| {
|
|
||||||
Comment::update_removed(conn, comment.id, true)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let comment_id = removed_comment.id;
|
|
||||||
let comment_view = blocking(context.pool(), move |conn| {
|
|
||||||
CommentView::read(conn, comment_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// TODO get those recipient actor ids from somewhere
|
|
||||||
let recipient_ids = vec![];
|
|
||||||
let res = CommentResponse {
|
|
||||||
comment: comment_view,
|
|
||||||
recipient_ids,
|
|
||||||
form_id: None,
|
|
||||||
};
|
|
||||||
context.chat_server().do_send(SendComment {
|
|
||||||
op: UserOperation::EditComment,
|
|
||||||
comment: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_remove_community(
|
|
||||||
context: &LemmyContext,
|
|
||||||
_remove: Remove,
|
|
||||||
community: Community,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let removed_community = blocking(context.pool(), move |conn| {
|
|
||||||
Community::update_removed(conn, community.id, true)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let community_id = removed_community.id;
|
|
||||||
let res = CommunityResponse {
|
|
||||||
community: blocking(context.pool(), move |conn| {
|
|
||||||
CommunityView::read(conn, community_id, None)
|
|
||||||
})
|
|
||||||
.await??,
|
|
||||||
};
|
|
||||||
|
|
||||||
let community_id = res.community.id;
|
|
||||||
context.chat_server().do_send(SendCommunityRoomMessage {
|
|
||||||
op: UserOperation::EditCommunity,
|
|
||||||
response: res,
|
|
||||||
community_id,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
|
@ -1,184 +0,0 @@
|
||||||
use crate::activities::receive::{
|
|
||||||
announce_if_community_is_local,
|
|
||||||
find_by_id,
|
|
||||||
get_actor_as_user,
|
|
||||||
receive_unhandled_activity,
|
|
||||||
undo_comment::*,
|
|
||||||
undo_post::*,
|
|
||||||
verify_activity_domains_valid,
|
|
||||||
FindResults,
|
|
||||||
};
|
|
||||||
use activitystreams::{activity::*, base::AnyBase, prelude::*};
|
|
||||||
use actix_web::HttpResponse;
|
|
||||||
use anyhow::{anyhow, Context};
|
|
||||||
use lemmy_db::{community::Community, community_view::CommunityView};
|
|
||||||
use lemmy_structs::{blocking, community::CommunityResponse};
|
|
||||||
use lemmy_utils::{location_info, LemmyError};
|
|
||||||
use lemmy_websocket::{messages::SendCommunityRoomMessage, LemmyContext, UserOperation};
|
|
||||||
use url::Url;
|
|
||||||
|
|
||||||
pub async fn receive_undo(
|
|
||||||
context: &LemmyContext,
|
|
||||||
activity: AnyBase,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let undo = Undo::from_any_base(activity)?.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&undo, expected_domain.to_owned(), true)?;
|
|
||||||
|
|
||||||
match undo.object().as_single_kind_str() {
|
|
||||||
Some("Delete") => receive_undo_delete(context, undo, expected_domain).await,
|
|
||||||
Some("Remove") => receive_undo_remove(context, undo, expected_domain).await,
|
|
||||||
Some("Like") => receive_undo_like(context, undo, expected_domain).await,
|
|
||||||
Some("Dislike") => receive_undo_dislike(context, undo, expected_domain).await,
|
|
||||||
_ => receive_unhandled_activity(undo),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_undo_delete(
|
|
||||||
context: &LemmyContext,
|
|
||||||
undo: Undo,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let delete = Delete::from_any_base(undo.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&delete, expected_domain, true)?;
|
|
||||||
|
|
||||||
let object = delete
|
|
||||||
.object()
|
|
||||||
.to_owned()
|
|
||||||
.single_xsd_any_uri()
|
|
||||||
.context(location_info!())?;
|
|
||||||
match find_by_id(context, object).await {
|
|
||||||
Ok(FindResults::Post(p)) => receive_undo_delete_post(context, undo, p).await,
|
|
||||||
Ok(FindResults::Comment(c)) => receive_undo_delete_comment(context, undo, c).await,
|
|
||||||
Ok(FindResults::Community(c)) => receive_undo_delete_community(context, undo, c).await,
|
|
||||||
// if we dont have the object, no need to do anything
|
|
||||||
Err(_) => Ok(HttpResponse::Ok().finish()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_undo_remove(
|
|
||||||
context: &LemmyContext,
|
|
||||||
undo: Undo,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let remove = Remove::from_any_base(undo.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&remove, expected_domain, false)?;
|
|
||||||
|
|
||||||
let object = remove
|
|
||||||
.object()
|
|
||||||
.to_owned()
|
|
||||||
.single_xsd_any_uri()
|
|
||||||
.context(location_info!())?;
|
|
||||||
match find_by_id(context, object).await {
|
|
||||||
Ok(FindResults::Post(p)) => receive_undo_remove_post(context, undo, p).await,
|
|
||||||
Ok(FindResults::Comment(c)) => receive_undo_remove_comment(context, undo, c).await,
|
|
||||||
Ok(FindResults::Community(c)) => receive_undo_remove_community(context, undo, c).await,
|
|
||||||
// if we dont have the object, no need to do anything
|
|
||||||
Err(_) => Ok(HttpResponse::Ok().finish()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_undo_like(
|
|
||||||
context: &LemmyContext,
|
|
||||||
undo: Undo,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let like = Like::from_any_base(undo.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&like, expected_domain, false)?;
|
|
||||||
|
|
||||||
let type_ = like
|
|
||||||
.object()
|
|
||||||
.as_single_kind_str()
|
|
||||||
.context(location_info!())?;
|
|
||||||
match type_ {
|
|
||||||
"Note" => receive_undo_like_comment(undo, &like, context).await,
|
|
||||||
"Page" => receive_undo_like_post(undo, &like, context).await,
|
|
||||||
d => Err(anyhow!("Undo Delete type {} not supported", d).into()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_undo_dislike(
|
|
||||||
context: &LemmyContext,
|
|
||||||
undo: Undo,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let dislike = Dislike::from_any_base(undo.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&dislike, expected_domain, false)?;
|
|
||||||
|
|
||||||
let type_ = dislike
|
|
||||||
.object()
|
|
||||||
.as_single_kind_str()
|
|
||||||
.context(location_info!())?;
|
|
||||||
match type_ {
|
|
||||||
"Note" => receive_undo_dislike_comment(undo, &dislike, context).await,
|
|
||||||
"Page" => receive_undo_dislike_post(undo, &dislike, context).await,
|
|
||||||
d => Err(anyhow!("Undo Delete type {} not supported", d).into()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_undo_delete_community(
|
|
||||||
context: &LemmyContext,
|
|
||||||
undo: Undo,
|
|
||||||
community: Community,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let deleted_community = blocking(context.pool(), move |conn| {
|
|
||||||
Community::update_deleted(conn, community.id, false)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let community_id = deleted_community.id;
|
|
||||||
let res = CommunityResponse {
|
|
||||||
community: blocking(context.pool(), move |conn| {
|
|
||||||
CommunityView::read(conn, community_id, None)
|
|
||||||
})
|
|
||||||
.await??,
|
|
||||||
};
|
|
||||||
|
|
||||||
let community_id = res.community.id;
|
|
||||||
context.chat_server().do_send(SendCommunityRoomMessage {
|
|
||||||
op: UserOperation::EditCommunity,
|
|
||||||
response: res,
|
|
||||||
community_id,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
let user = get_actor_as_user(&undo, context).await?;
|
|
||||||
announce_if_community_is_local(undo, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_undo_remove_community(
|
|
||||||
context: &LemmyContext,
|
|
||||||
undo: Undo,
|
|
||||||
community: Community,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let removed_community = blocking(context.pool(), move |conn| {
|
|
||||||
Community::update_removed(conn, community.id, false)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let community_id = removed_community.id;
|
|
||||||
let res = CommunityResponse {
|
|
||||||
community: blocking(context.pool(), move |conn| {
|
|
||||||
CommunityView::read(conn, community_id, None)
|
|
||||||
})
|
|
||||||
.await??,
|
|
||||||
};
|
|
||||||
|
|
||||||
let community_id = res.community.id;
|
|
||||||
|
|
||||||
context.chat_server().do_send(SendCommunityRoomMessage {
|
|
||||||
op: UserOperation::EditCommunity,
|
|
||||||
response: res,
|
|
||||||
community_id,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
let mod_ = get_actor_as_user(&undo, context).await?;
|
|
||||||
announce_if_community_is_local(undo, &mod_, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
|
@ -1,137 +0,0 @@
|
||||||
use crate::{
|
|
||||||
activities::receive::{
|
|
||||||
announce_if_community_is_local,
|
|
||||||
get_actor_as_user,
|
|
||||||
receive_unhandled_activity,
|
|
||||||
verify_activity_domains_valid,
|
|
||||||
},
|
|
||||||
fetcher::{get_or_fetch_and_insert_comment, get_or_fetch_and_insert_post},
|
|
||||||
ActorType,
|
|
||||||
FromApub,
|
|
||||||
PageExt,
|
|
||||||
};
|
|
||||||
use activitystreams::{activity::Update, base::AnyBase, object::Note, prelude::*};
|
|
||||||
use actix_web::HttpResponse;
|
|
||||||
use anyhow::Context;
|
|
||||||
use lemmy_db::{
|
|
||||||
comment::{Comment, CommentForm},
|
|
||||||
comment_view::CommentView,
|
|
||||||
post::{Post, PostForm},
|
|
||||||
post_view::PostView,
|
|
||||||
Crud,
|
|
||||||
};
|
|
||||||
use lemmy_structs::{blocking, comment::CommentResponse, post::PostResponse, send_local_notifs};
|
|
||||||
use lemmy_utils::{location_info, utils::scrape_text_for_mentions, LemmyError};
|
|
||||||
use lemmy_websocket::{
|
|
||||||
messages::{SendComment, SendPost},
|
|
||||||
LemmyContext,
|
|
||||||
UserOperation,
|
|
||||||
};
|
|
||||||
use url::Url;
|
|
||||||
|
|
||||||
pub async fn receive_update(
|
|
||||||
context: &LemmyContext,
|
|
||||||
activity: AnyBase,
|
|
||||||
expected_domain: Url,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let update = Update::from_any_base(activity)?.context(location_info!())?;
|
|
||||||
verify_activity_domains_valid(&update, expected_domain, true)?;
|
|
||||||
|
|
||||||
match update.object().as_single_kind_str() {
|
|
||||||
Some("Page") => receive_update_post(update, context).await,
|
|
||||||
Some("Note") => receive_update_comment(update, context).await,
|
|
||||||
_ => receive_unhandled_activity(update),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_update_post(
|
|
||||||
update: Update,
|
|
||||||
context: &LemmyContext,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let user = get_actor_as_user(&update, context).await?;
|
|
||||||
let page = PageExt::from_any_base(update.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
|
|
||||||
let post = PostForm::from_apub(&page, context, Some(user.actor_id()?)).await?;
|
|
||||||
|
|
||||||
let original_post_id = get_or_fetch_and_insert_post(&post.get_ap_id()?, context)
|
|
||||||
.await?
|
|
||||||
.id;
|
|
||||||
|
|
||||||
blocking(context.pool(), move |conn| {
|
|
||||||
Post::update(conn, original_post_id, &post)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let post_view = blocking(context.pool(), move |conn| {
|
|
||||||
PostView::read(conn, original_post_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let res = PostResponse { post: post_view };
|
|
||||||
|
|
||||||
context.chat_server().do_send(SendPost {
|
|
||||||
op: UserOperation::EditPost,
|
|
||||||
post: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
announce_if_community_is_local(update, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
||||||
|
|
||||||
async fn receive_update_comment(
|
|
||||||
update: Update,
|
|
||||||
context: &LemmyContext,
|
|
||||||
) -> Result<HttpResponse, LemmyError> {
|
|
||||||
let note = Note::from_any_base(update.object().to_owned().one().context(location_info!())?)?
|
|
||||||
.context(location_info!())?;
|
|
||||||
let user = get_actor_as_user(&update, context).await?;
|
|
||||||
|
|
||||||
let comment = CommentForm::from_apub(¬e, context, Some(user.actor_id()?)).await?;
|
|
||||||
|
|
||||||
let original_comment_id = get_or_fetch_and_insert_comment(&comment.get_ap_id()?, context)
|
|
||||||
.await?
|
|
||||||
.id;
|
|
||||||
|
|
||||||
let updated_comment = blocking(context.pool(), move |conn| {
|
|
||||||
Comment::update(conn, original_comment_id, &comment)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let post_id = updated_comment.post_id;
|
|
||||||
let post = blocking(context.pool(), move |conn| Post::read(conn, post_id)).await??;
|
|
||||||
|
|
||||||
let mentions = scrape_text_for_mentions(&updated_comment.content);
|
|
||||||
let recipient_ids = send_local_notifs(
|
|
||||||
mentions,
|
|
||||||
updated_comment,
|
|
||||||
&user,
|
|
||||||
post,
|
|
||||||
context.pool(),
|
|
||||||
false,
|
|
||||||
)
|
|
||||||
.await?;
|
|
||||||
|
|
||||||
// Refetch the view
|
|
||||||
let comment_view = blocking(context.pool(), move |conn| {
|
|
||||||
CommentView::read(conn, original_comment_id, None)
|
|
||||||
})
|
|
||||||
.await??;
|
|
||||||
|
|
||||||
let res = CommentResponse {
|
|
||||||
comment: comment_view,
|
|
||||||
recipient_ids,
|
|
||||||
form_id: None,
|
|
||||||
};
|
|
||||||
|
|
||||||
context.chat_server().do_send(SendComment {
|
|
||||||
op: UserOperation::EditComment,
|
|
||||||
comment: res,
|
|
||||||
websocket_id: None,
|
|
||||||
});
|
|
||||||
|
|
||||||
announce_if_community_is_local(update, &user, context).await?;
|
|
||||||
Ok(HttpResponse::Ok().finish())
|
|
||||||
}
|
|
|
@ -1,27 +1,65 @@
|
||||||
use crate::{
|
use crate::{
|
||||||
activities::receive::{
|
activities::receive::{
|
||||||
announce::receive_announce,
|
comment::{
|
||||||
create::receive_create,
|
receive_create_comment,
|
||||||
delete::receive_delete,
|
receive_delete_comment,
|
||||||
dislike::receive_dislike,
|
receive_dislike_comment,
|
||||||
like::receive_like,
|
receive_like_comment,
|
||||||
remove::receive_remove,
|
receive_remove_comment,
|
||||||
undo::receive_undo,
|
receive_update_comment,
|
||||||
update::receive_update,
|
},
|
||||||
|
comment_undo::{
|
||||||
|
receive_undo_delete_comment,
|
||||||
|
receive_undo_dislike_comment,
|
||||||
|
receive_undo_like_comment,
|
||||||
|
receive_undo_remove_comment,
|
||||||
|
},
|
||||||
|
community::{
|
||||||
|
receive_delete_community,
|
||||||
|
receive_remove_community,
|
||||||
|
receive_undo_delete_community,
|
||||||
|
receive_undo_remove_community,
|
||||||
|
},
|
||||||
|
find_by_id,
|
||||||
|
post::{
|
||||||
|
receive_create_post,
|
||||||
|
receive_delete_post,
|
||||||
|
receive_dislike_post,
|
||||||
|
receive_like_post,
|
||||||
|
receive_remove_post,
|
||||||
|
receive_update_post,
|
||||||
|
},
|
||||||
|
post_undo::{
|
||||||
|
receive_undo_delete_post,
|
||||||
|
receive_undo_dislike_post,
|
||||||
|
receive_undo_like_post,
|
||||||
|
receive_undo_remove_post,
|
||||||
|
},
|
||||||
|
receive_unhandled_activity,
|
||||||
|
verify_activity_domains_valid,
|
||||||
|
FindResults,
|
||||||
},
|
},
|
||||||
check_is_apub_id_valid,
|
check_is_apub_id_valid,
|
||||||
extensions::signatures::verify_signature,
|
extensions::signatures::verify_signature,
|
||||||
fetcher::get_or_fetch_and_upsert_actor,
|
fetcher::get_or_fetch_and_upsert_actor,
|
||||||
insert_activity,
|
insert_activity,
|
||||||
|
ActorType,
|
||||||
|
};
|
||||||
|
use activitystreams::{
|
||||||
|
activity::{ActorAndObject, Announce, Create, Delete, Dislike, Like, Remove, Undo, Update},
|
||||||
|
base::AnyBase,
|
||||||
|
prelude::*,
|
||||||
};
|
};
|
||||||
use activitystreams::{activity::ActorAndObject, prelude::*};
|
|
||||||
use actix_web::{web, HttpRequest, HttpResponse};
|
use actix_web::{web, HttpRequest, HttpResponse};
|
||||||
use anyhow::Context;
|
use anyhow::{anyhow, Context};
|
||||||
|
use lemmy_db::{site::Site, Crud};
|
||||||
|
use lemmy_structs::blocking;
|
||||||
use lemmy_utils::{location_info, LemmyError};
|
use lemmy_utils::{location_info, LemmyError};
|
||||||
use lemmy_websocket::LemmyContext;
|
use lemmy_websocket::LemmyContext;
|
||||||
use log::debug;
|
use log::debug;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
use std::fmt::Debug;
|
use std::fmt::Debug;
|
||||||
|
use url::Url;
|
||||||
|
|
||||||
/// Allowed activity types for shared inbox.
|
/// Allowed activity types for shared inbox.
|
||||||
#[derive(Clone, Debug, Eq, Ord, PartialEq, PartialOrd, Deserialize, Serialize)]
|
#[derive(Clone, Debug, Eq, Ord, PartialEq, PartialOrd, Deserialize, Serialize)]
|
||||||
|
@ -81,3 +119,265 @@ pub async fn shared_inbox(
|
||||||
insert_activity(actor.user_id(), activity.clone(), false, context.pool()).await?;
|
insert_activity(actor.user_id(), activity.clone(), false, context.pool()).await?;
|
||||||
res
|
res
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Takes an announce and passes the inner activity to the appropriate handler.
|
||||||
|
async fn receive_announce(
|
||||||
|
context: &LemmyContext,
|
||||||
|
activity: AnyBase,
|
||||||
|
actor: &dyn ActorType,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let announce = Announce::from_any_base(activity)?.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&announce, actor.actor_id()?, false)?;
|
||||||
|
|
||||||
|
let kind = announce.object().as_single_kind_str();
|
||||||
|
let object = announce
|
||||||
|
.object()
|
||||||
|
.to_owned()
|
||||||
|
.one()
|
||||||
|
.context(location_info!())?;
|
||||||
|
|
||||||
|
let inner_id = object.id().context(location_info!())?.to_owned();
|
||||||
|
check_is_apub_id_valid(&inner_id)?;
|
||||||
|
|
||||||
|
match kind {
|
||||||
|
Some("Create") => receive_create(context, object, inner_id).await,
|
||||||
|
Some("Update") => receive_update(context, object, inner_id).await,
|
||||||
|
Some("Like") => receive_like(context, object, inner_id).await,
|
||||||
|
Some("Dislike") => receive_dislike(context, object, inner_id).await,
|
||||||
|
Some("Delete") => receive_delete(context, object, inner_id).await,
|
||||||
|
Some("Remove") => receive_remove(context, object, inner_id).await,
|
||||||
|
Some("Undo") => receive_undo(context, object, inner_id).await,
|
||||||
|
_ => receive_unhandled_activity(announce),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_create(
|
||||||
|
context: &LemmyContext,
|
||||||
|
activity: AnyBase,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let create = Create::from_any_base(activity)?.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&create, expected_domain, true)?;
|
||||||
|
|
||||||
|
match create.object().as_single_kind_str() {
|
||||||
|
Some("Page") => receive_create_post(create, context).await,
|
||||||
|
Some("Note") => receive_create_comment(create, context).await,
|
||||||
|
_ => receive_unhandled_activity(create),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_update(
|
||||||
|
context: &LemmyContext,
|
||||||
|
activity: AnyBase,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let update = Update::from_any_base(activity)?.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&update, expected_domain, true)?;
|
||||||
|
|
||||||
|
match update.object().as_single_kind_str() {
|
||||||
|
Some("Page") => receive_update_post(update, context).await,
|
||||||
|
Some("Note") => receive_update_comment(update, context).await,
|
||||||
|
_ => receive_unhandled_activity(update),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_like(
|
||||||
|
context: &LemmyContext,
|
||||||
|
activity: AnyBase,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let like = Like::from_any_base(activity)?.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&like, expected_domain, false)?;
|
||||||
|
|
||||||
|
match like.object().as_single_kind_str() {
|
||||||
|
Some("Page") => receive_like_post(like, context).await,
|
||||||
|
Some("Note") => receive_like_comment(like, context).await,
|
||||||
|
_ => receive_unhandled_activity(like),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_dislike(
|
||||||
|
context: &LemmyContext,
|
||||||
|
activity: AnyBase,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let enable_downvotes = blocking(context.pool(), move |conn| {
|
||||||
|
Site::read(conn, 1).map(|s| s.enable_downvotes)
|
||||||
|
})
|
||||||
|
.await??;
|
||||||
|
if !enable_downvotes {
|
||||||
|
return Ok(HttpResponse::Ok().finish());
|
||||||
|
}
|
||||||
|
|
||||||
|
let dislike = Dislike::from_any_base(activity)?.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&dislike, expected_domain, false)?;
|
||||||
|
|
||||||
|
match dislike.object().as_single_kind_str() {
|
||||||
|
Some("Page") => receive_dislike_post(dislike, context).await,
|
||||||
|
Some("Note") => receive_dislike_comment(dislike, context).await,
|
||||||
|
_ => receive_unhandled_activity(dislike),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn receive_delete(
|
||||||
|
context: &LemmyContext,
|
||||||
|
activity: AnyBase,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let delete = Delete::from_any_base(activity)?.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&delete, expected_domain, true)?;
|
||||||
|
|
||||||
|
let object = delete
|
||||||
|
.object()
|
||||||
|
.to_owned()
|
||||||
|
.single_xsd_any_uri()
|
||||||
|
.context(location_info!())?;
|
||||||
|
|
||||||
|
match find_by_id(context, object).await {
|
||||||
|
Ok(FindResults::Post(p)) => receive_delete_post(context, delete, p).await,
|
||||||
|
Ok(FindResults::Comment(c)) => receive_delete_comment(context, delete, c).await,
|
||||||
|
Ok(FindResults::Community(c)) => receive_delete_community(context, delete, c).await,
|
||||||
|
// if we dont have the object, no need to do anything
|
||||||
|
Err(_) => Ok(HttpResponse::Ok().finish()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_remove(
|
||||||
|
context: &LemmyContext,
|
||||||
|
activity: AnyBase,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let remove = Remove::from_any_base(activity)?.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&remove, expected_domain, false)?;
|
||||||
|
|
||||||
|
let cc = remove
|
||||||
|
.cc()
|
||||||
|
.map(|c| c.as_many())
|
||||||
|
.flatten()
|
||||||
|
.context(location_info!())?;
|
||||||
|
let community_id = cc
|
||||||
|
.first()
|
||||||
|
.map(|c| c.as_xsd_any_uri())
|
||||||
|
.flatten()
|
||||||
|
.context(location_info!())?;
|
||||||
|
|
||||||
|
let object = remove
|
||||||
|
.object()
|
||||||
|
.to_owned()
|
||||||
|
.single_xsd_any_uri()
|
||||||
|
.context(location_info!())?;
|
||||||
|
|
||||||
|
// Ensure that remove activity comes from the same domain as the community
|
||||||
|
remove.id(community_id.domain().context(location_info!())?)?;
|
||||||
|
|
||||||
|
match find_by_id(context, object).await {
|
||||||
|
Ok(FindResults::Post(p)) => receive_remove_post(context, remove, p).await,
|
||||||
|
Ok(FindResults::Comment(c)) => receive_remove_comment(context, remove, c).await,
|
||||||
|
Ok(FindResults::Community(c)) => receive_remove_community(context, remove, c).await,
|
||||||
|
// if we dont have the object, no need to do anything
|
||||||
|
Err(_) => Ok(HttpResponse::Ok().finish()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_undo(
|
||||||
|
context: &LemmyContext,
|
||||||
|
activity: AnyBase,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let undo = Undo::from_any_base(activity)?.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&undo, expected_domain.to_owned(), true)?;
|
||||||
|
|
||||||
|
match undo.object().as_single_kind_str() {
|
||||||
|
Some("Delete") => receive_undo_delete(context, undo, expected_domain).await,
|
||||||
|
Some("Remove") => receive_undo_remove(context, undo, expected_domain).await,
|
||||||
|
Some("Like") => receive_undo_like(context, undo, expected_domain).await,
|
||||||
|
Some("Dislike") => receive_undo_dislike(context, undo, expected_domain).await,
|
||||||
|
_ => receive_unhandled_activity(undo),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_undo_delete(
|
||||||
|
context: &LemmyContext,
|
||||||
|
undo: Undo,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let delete = Delete::from_any_base(undo.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&delete, expected_domain, true)?;
|
||||||
|
|
||||||
|
let object = delete
|
||||||
|
.object()
|
||||||
|
.to_owned()
|
||||||
|
.single_xsd_any_uri()
|
||||||
|
.context(location_info!())?;
|
||||||
|
match find_by_id(context, object).await {
|
||||||
|
Ok(FindResults::Post(p)) => receive_undo_delete_post(context, undo, p).await,
|
||||||
|
Ok(FindResults::Comment(c)) => receive_undo_delete_comment(context, undo, c).await,
|
||||||
|
Ok(FindResults::Community(c)) => receive_undo_delete_community(context, undo, c).await,
|
||||||
|
// if we dont have the object, no need to do anything
|
||||||
|
Err(_) => Ok(HttpResponse::Ok().finish()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_undo_remove(
|
||||||
|
context: &LemmyContext,
|
||||||
|
undo: Undo,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let remove = Remove::from_any_base(undo.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&remove, expected_domain, false)?;
|
||||||
|
|
||||||
|
let object = remove
|
||||||
|
.object()
|
||||||
|
.to_owned()
|
||||||
|
.single_xsd_any_uri()
|
||||||
|
.context(location_info!())?;
|
||||||
|
match find_by_id(context, object).await {
|
||||||
|
Ok(FindResults::Post(p)) => receive_undo_remove_post(context, undo, p).await,
|
||||||
|
Ok(FindResults::Comment(c)) => receive_undo_remove_comment(context, undo, c).await,
|
||||||
|
Ok(FindResults::Community(c)) => receive_undo_remove_community(context, undo, c).await,
|
||||||
|
// if we dont have the object, no need to do anything
|
||||||
|
Err(_) => Ok(HttpResponse::Ok().finish()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_undo_like(
|
||||||
|
context: &LemmyContext,
|
||||||
|
undo: Undo,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let like = Like::from_any_base(undo.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&like, expected_domain, false)?;
|
||||||
|
|
||||||
|
let type_ = like
|
||||||
|
.object()
|
||||||
|
.as_single_kind_str()
|
||||||
|
.context(location_info!())?;
|
||||||
|
match type_ {
|
||||||
|
"Note" => receive_undo_like_comment(undo, &like, context).await,
|
||||||
|
"Page" => receive_undo_like_post(undo, &like, context).await,
|
||||||
|
d => Err(anyhow!("Undo Delete type {} not supported", d).into()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn receive_undo_dislike(
|
||||||
|
context: &LemmyContext,
|
||||||
|
undo: Undo,
|
||||||
|
expected_domain: Url,
|
||||||
|
) -> Result<HttpResponse, LemmyError> {
|
||||||
|
let dislike = Dislike::from_any_base(undo.object().to_owned().one().context(location_info!())?)?
|
||||||
|
.context(location_info!())?;
|
||||||
|
verify_activity_domains_valid(&dislike, expected_domain, false)?;
|
||||||
|
|
||||||
|
let type_ = dislike
|
||||||
|
.object()
|
||||||
|
.as_single_kind_str()
|
||||||
|
.context(location_info!())?;
|
||||||
|
match type_ {
|
||||||
|
"Note" => receive_undo_dislike_comment(undo, &dislike, context).await,
|
||||||
|
"Page" => receive_undo_dislike_post(undo, &dislike, context).await,
|
||||||
|
d => Err(anyhow!("Undo Delete type {} not supported", d).into()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -27,6 +27,7 @@ use lemmy_structs::blocking;
|
||||||
use lemmy_utils::{location_info, settings::Settings, LemmyError};
|
use lemmy_utils::{location_info, settings::Settings, LemmyError};
|
||||||
use lemmy_websocket::LemmyContext;
|
use lemmy_websocket::LemmyContext;
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
use std::net::IpAddr;
|
||||||
use url::{ParseError, Url};
|
use url::{ParseError, Url};
|
||||||
|
|
||||||
/// Activitystreams type for community
|
/// Activitystreams type for community
|
||||||
|
@ -72,6 +73,12 @@ fn check_is_apub_id_valid(apub_id: &Url) -> Result<(), LemmyError> {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let host = apub_id.host_str().context(location_info!())?;
|
||||||
|
let host_as_ip = host.parse::<IpAddr>();
|
||||||
|
if host == "localhost" || host_as_ip.is_ok() {
|
||||||
|
return Err(anyhow!("invalid hostname: {:?}", host).into());
|
||||||
|
}
|
||||||
|
|
||||||
if apub_id.scheme() != Settings::get().get_protocol_string() {
|
if apub_id.scheme() != Settings::get().get_protocol_string() {
|
||||||
return Err(anyhow!("invalid apub id scheme: {:?}", apub_id.scheme()).into());
|
return Err(anyhow!("invalid apub id scheme: {:?}", apub_id.scheme()).into());
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue