|
|
|
@ -13,14 +13,13 @@ use crate::{
|
|
|
|
|
use lemmy_db::{
|
|
|
|
|
comment::*,
|
|
|
|
|
comment_view::*,
|
|
|
|
|
community::Community,
|
|
|
|
|
community_view::*,
|
|
|
|
|
moderator::*,
|
|
|
|
|
naive_now,
|
|
|
|
|
post::*,
|
|
|
|
|
site_view::*,
|
|
|
|
|
user::*,
|
|
|
|
|
user_mention::*,
|
|
|
|
|
user_view::*,
|
|
|
|
|
Crud,
|
|
|
|
|
Likeable,
|
|
|
|
|
ListingType,
|
|
|
|
@ -44,7 +43,6 @@ use std::str::FromStr;
|
|
|
|
|
pub struct CreateComment {
|
|
|
|
|
content: String,
|
|
|
|
|
parent_id: Option<i32>,
|
|
|
|
|
edit_id: Option<i32>, // TODO this isn't used
|
|
|
|
|
pub post_id: i32,
|
|
|
|
|
auth: String,
|
|
|
|
|
}
|
|
|
|
@ -52,14 +50,29 @@ pub struct CreateComment {
|
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
|
pub struct EditComment {
|
|
|
|
|
content: String,
|
|
|
|
|
parent_id: Option<i32>, // TODO why are the parent_id, creator_id, post_id, etc fields required? They aren't going to change
|
|
|
|
|
edit_id: i32,
|
|
|
|
|
creator_id: i32,
|
|
|
|
|
pub post_id: i32,
|
|
|
|
|
removed: Option<bool>,
|
|
|
|
|
deleted: Option<bool>,
|
|
|
|
|
auth: String,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
|
pub struct DeleteComment {
|
|
|
|
|
edit_id: i32,
|
|
|
|
|
deleted: bool,
|
|
|
|
|
auth: String,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
|
pub struct RemoveComment {
|
|
|
|
|
edit_id: i32,
|
|
|
|
|
removed: bool,
|
|
|
|
|
reason: Option<String>,
|
|
|
|
|
read: Option<bool>,
|
|
|
|
|
auth: String,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
|
pub struct MarkCommentAsRead {
|
|
|
|
|
edit_id: i32,
|
|
|
|
|
read: bool,
|
|
|
|
|
auth: String,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -79,7 +92,6 @@ pub struct CommentResponse {
|
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
|
pub struct CreateCommentLike {
|
|
|
|
|
comment_id: i32,
|
|
|
|
|
pub post_id: i32,
|
|
|
|
|
score: i16,
|
|
|
|
|
auth: String,
|
|
|
|
|
}
|
|
|
|
@ -150,6 +162,7 @@ impl Perform for Oper<CreateComment> {
|
|
|
|
|
return Err(APIError::err("site_ban").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create the comment
|
|
|
|
|
let comment_form2 = comment_form.clone();
|
|
|
|
|
let inserted_comment =
|
|
|
|
|
match blocking(pool, move |conn| Comment::create(&conn, &comment_form2)).await? {
|
|
|
|
@ -157,6 +170,7 @@ impl Perform for Oper<CreateComment> {
|
|
|
|
|
Err(_e) => return Err(APIError::err("couldnt_create_comment").into()),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Necessary to update the ap_id
|
|
|
|
|
let inserted_comment_id = inserted_comment.id;
|
|
|
|
|
let updated_comment: Comment = match blocking(pool, move |conn| {
|
|
|
|
|
let apub_id =
|
|
|
|
@ -175,8 +189,15 @@ impl Perform for Oper<CreateComment> {
|
|
|
|
|
|
|
|
|
|
// Scan the comment for user mentions, add those rows
|
|
|
|
|
let mentions = scrape_text_for_mentions(&comment_form.content);
|
|
|
|
|
let recipient_ids =
|
|
|
|
|
send_local_notifs(mentions, updated_comment.clone(), user.clone(), post, pool).await?;
|
|
|
|
|
let recipient_ids = send_local_notifs(
|
|
|
|
|
mentions,
|
|
|
|
|
updated_comment.clone(),
|
|
|
|
|
user.clone(),
|
|
|
|
|
post,
|
|
|
|
|
pool,
|
|
|
|
|
true,
|
|
|
|
|
)
|
|
|
|
|
.await?;
|
|
|
|
|
|
|
|
|
|
// You like your own comment by default
|
|
|
|
|
let like_form = CommentLikeForm {
|
|
|
|
@ -237,122 +258,127 @@ impl Perform for Oper<EditComment> {
|
|
|
|
|
|
|
|
|
|
let user_id = claims.id;
|
|
|
|
|
|
|
|
|
|
let user = blocking(pool, move |conn| User_::read(&conn, user_id)).await??;
|
|
|
|
|
|
|
|
|
|
let edit_id = data.edit_id;
|
|
|
|
|
let orig_comment =
|
|
|
|
|
blocking(pool, move |conn| CommentView::read(&conn, edit_id, None)).await??;
|
|
|
|
|
|
|
|
|
|
let mut editors: Vec<i32> = vec![orig_comment.creator_id];
|
|
|
|
|
let mut moderators: Vec<i32> = vec![];
|
|
|
|
|
// Check for a site ban
|
|
|
|
|
let user = blocking(pool, move |conn| User_::read(conn, user_id)).await??;
|
|
|
|
|
if user.banned {
|
|
|
|
|
return Err(APIError::err("site_ban").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for a community ban
|
|
|
|
|
let community_id = orig_comment.community_id;
|
|
|
|
|
moderators.append(
|
|
|
|
|
&mut blocking(pool, move |conn| {
|
|
|
|
|
CommunityModeratorView::for_community(&conn, community_id)
|
|
|
|
|
.map(|v| v.into_iter().map(|m| m.user_id).collect())
|
|
|
|
|
})
|
|
|
|
|
.await??,
|
|
|
|
|
);
|
|
|
|
|
moderators.append(
|
|
|
|
|
&mut blocking(pool, move |conn| {
|
|
|
|
|
UserView::admins(conn).map(|v| v.into_iter().map(|a| a.id).collect())
|
|
|
|
|
})
|
|
|
|
|
.await??,
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
editors.extend(&moderators);
|
|
|
|
|
// You are allowed to mark the comment as read even if you're banned.
|
|
|
|
|
if data.read.is_none() {
|
|
|
|
|
// Verify its the creator or a mod, or an admin
|
|
|
|
|
|
|
|
|
|
if !editors.contains(&user_id) {
|
|
|
|
|
return Err(APIError::err("no_comment_edit_allowed").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for a community ban
|
|
|
|
|
let community_id = orig_comment.community_id;
|
|
|
|
|
let is_banned =
|
|
|
|
|
move |conn: &'_ _| CommunityUserBanView::get(conn, user_id, community_id).is_ok();
|
|
|
|
|
if blocking(pool, is_banned).await? {
|
|
|
|
|
return Err(APIError::err("community_ban").into());
|
|
|
|
|
}
|
|
|
|
|
let is_banned =
|
|
|
|
|
move |conn: &'_ _| CommunityUserBanView::get(conn, user_id, community_id).is_ok();
|
|
|
|
|
if blocking(pool, is_banned).await? {
|
|
|
|
|
return Err(APIError::err("community_ban").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for a site ban
|
|
|
|
|
if user.banned {
|
|
|
|
|
return Err(APIError::err("site_ban").into());
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// check that user can mark as read
|
|
|
|
|
let parent_id = orig_comment.parent_id;
|
|
|
|
|
match parent_id {
|
|
|
|
|
Some(pid) => {
|
|
|
|
|
let parent_comment =
|
|
|
|
|
blocking(pool, move |conn| CommentView::read(&conn, pid, None)).await??;
|
|
|
|
|
if user_id != parent_comment.creator_id {
|
|
|
|
|
return Err(APIError::err("no_comment_edit_allowed").into());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
None => {
|
|
|
|
|
let parent_post_id = orig_comment.post_id;
|
|
|
|
|
let parent_post = blocking(pool, move |conn| Post::read(conn, parent_post_id)).await??;
|
|
|
|
|
if user_id != parent_post.creator_id {
|
|
|
|
|
return Err(APIError::err("no_comment_edit_allowed").into());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// Verify that only the creator can edit
|
|
|
|
|
if user_id != orig_comment.creator_id {
|
|
|
|
|
return Err(APIError::err("no_comment_edit_allowed").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Do the update
|
|
|
|
|
let content_slurs_removed = remove_slurs(&data.content.to_owned());
|
|
|
|
|
let edit_id = data.edit_id;
|
|
|
|
|
let updated_comment = match blocking(pool, move |conn| {
|
|
|
|
|
Comment::update_content(conn, edit_id, &content_slurs_removed)
|
|
|
|
|
})
|
|
|
|
|
.await?
|
|
|
|
|
{
|
|
|
|
|
Ok(comment) => comment,
|
|
|
|
|
Err(_e) => return Err(APIError::err("couldnt_update_comment").into()),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Send the apub update
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_update(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
|
|
|
|
|
// Do the mentions / recipients
|
|
|
|
|
let post_id = orig_comment.post_id;
|
|
|
|
|
let post = blocking(pool, move |conn| Post::read(conn, post_id)).await??;
|
|
|
|
|
|
|
|
|
|
let updated_comment_content = updated_comment.content.to_owned();
|
|
|
|
|
let mentions = scrape_text_for_mentions(&updated_comment_content);
|
|
|
|
|
let recipient_ids =
|
|
|
|
|
send_local_notifs(mentions, updated_comment, user, post, pool, false).await?;
|
|
|
|
|
|
|
|
|
|
let edit_id = data.edit_id;
|
|
|
|
|
let read_comment = blocking(pool, move |conn| Comment::read(conn, edit_id)).await??;
|
|
|
|
|
|
|
|
|
|
let comment_form = {
|
|
|
|
|
if data.read.is_none() {
|
|
|
|
|
// the ban etc checks should been made and have passed
|
|
|
|
|
// the comment can be properly edited
|
|
|
|
|
let post_removed = if moderators.contains(&user_id) {
|
|
|
|
|
data.removed
|
|
|
|
|
} else {
|
|
|
|
|
Some(read_comment.removed)
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
CommentForm {
|
|
|
|
|
content: content_slurs_removed,
|
|
|
|
|
parent_id: read_comment.parent_id,
|
|
|
|
|
post_id: read_comment.post_id,
|
|
|
|
|
creator_id: read_comment.creator_id,
|
|
|
|
|
removed: post_removed.to_owned(),
|
|
|
|
|
deleted: data.deleted.to_owned(),
|
|
|
|
|
read: Some(read_comment.read),
|
|
|
|
|
published: None,
|
|
|
|
|
updated: Some(naive_now()),
|
|
|
|
|
ap_id: read_comment.ap_id,
|
|
|
|
|
local: read_comment.local,
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// the only field that can be updated it the read field
|
|
|
|
|
CommentForm {
|
|
|
|
|
content: read_comment.content,
|
|
|
|
|
parent_id: read_comment.parent_id,
|
|
|
|
|
post_id: read_comment.post_id,
|
|
|
|
|
creator_id: read_comment.creator_id,
|
|
|
|
|
removed: Some(read_comment.removed).to_owned(),
|
|
|
|
|
deleted: Some(read_comment.deleted).to_owned(),
|
|
|
|
|
read: data.read.to_owned(),
|
|
|
|
|
published: None,
|
|
|
|
|
updated: orig_comment.updated,
|
|
|
|
|
ap_id: read_comment.ap_id,
|
|
|
|
|
local: read_comment.local,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
let comment_view = blocking(pool, move |conn| {
|
|
|
|
|
CommentView::read(conn, edit_id, Some(user_id))
|
|
|
|
|
})
|
|
|
|
|
.await??;
|
|
|
|
|
|
|
|
|
|
let mut res = CommentResponse {
|
|
|
|
|
comment: comment_view,
|
|
|
|
|
recipient_ids,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if let Some(ws) = websocket_info {
|
|
|
|
|
ws.chatserver.do_send(SendComment {
|
|
|
|
|
op: UserOperation::EditComment,
|
|
|
|
|
comment: res.clone(),
|
|
|
|
|
my_id: ws.id,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// strip out the recipient_ids, so that
|
|
|
|
|
// users don't get double notifs
|
|
|
|
|
res.recipient_ids = Vec::new();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[async_trait::async_trait(?Send)]
|
|
|
|
|
impl Perform for Oper<DeleteComment> {
|
|
|
|
|
type Response = CommentResponse;
|
|
|
|
|
|
|
|
|
|
async fn perform(
|
|
|
|
|
&self,
|
|
|
|
|
pool: &DbPool,
|
|
|
|
|
websocket_info: Option<WebsocketInfo>,
|
|
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
|
|
|
|
let data: &DeleteComment = &self.data;
|
|
|
|
|
|
|
|
|
|
let claims = match Claims::decode(&data.auth) {
|
|
|
|
|
Ok(claims) => claims.claims,
|
|
|
|
|
Err(_e) => return Err(APIError::err("not_logged_in").into()),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let user_id = claims.id;
|
|
|
|
|
|
|
|
|
|
let edit_id = data.edit_id;
|
|
|
|
|
let comment_form2 = comment_form.clone();
|
|
|
|
|
let orig_comment =
|
|
|
|
|
blocking(pool, move |conn| CommentView::read(&conn, edit_id, None)).await??;
|
|
|
|
|
|
|
|
|
|
// Check for a site ban
|
|
|
|
|
let user = blocking(pool, move |conn| User_::read(conn, user_id)).await??;
|
|
|
|
|
if user.banned {
|
|
|
|
|
return Err(APIError::err("site_ban").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for a community ban
|
|
|
|
|
let community_id = orig_comment.community_id;
|
|
|
|
|
let is_banned =
|
|
|
|
|
move |conn: &'_ _| CommunityUserBanView::get(conn, user_id, community_id).is_ok();
|
|
|
|
|
if blocking(pool, is_banned).await? {
|
|
|
|
|
return Err(APIError::err("community_ban").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Verify that only the creator can delete
|
|
|
|
|
if user_id != orig_comment.creator_id {
|
|
|
|
|
return Err(APIError::err("no_comment_edit_allowed").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Do the delete
|
|
|
|
|
let deleted = data.deleted;
|
|
|
|
|
let updated_comment = match blocking(pool, move |conn| {
|
|
|
|
|
Comment::update(conn, edit_id, &comment_form2)
|
|
|
|
|
Comment::update_deleted(conn, edit_id, deleted)
|
|
|
|
|
})
|
|
|
|
|
.await?
|
|
|
|
|
{
|
|
|
|
@ -360,61 +386,142 @@ impl Perform for Oper<EditComment> {
|
|
|
|
|
Err(_e) => return Err(APIError::err("couldnt_update_comment").into()),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if data.read.is_none() {
|
|
|
|
|
if let Some(deleted) = data.deleted.to_owned() {
|
|
|
|
|
if deleted {
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_delete(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
} else {
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_undo_delete(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
}
|
|
|
|
|
} else if let Some(removed) = data.removed.to_owned() {
|
|
|
|
|
if moderators.contains(&user_id) {
|
|
|
|
|
if removed {
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_remove(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
} else {
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_undo_remove(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_update(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Mod tables
|
|
|
|
|
if moderators.contains(&user_id) {
|
|
|
|
|
if let Some(removed) = data.removed.to_owned() {
|
|
|
|
|
let form = ModRemoveCommentForm {
|
|
|
|
|
mod_user_id: user_id,
|
|
|
|
|
comment_id: data.edit_id,
|
|
|
|
|
removed: Some(removed),
|
|
|
|
|
reason: data.reason.to_owned(),
|
|
|
|
|
};
|
|
|
|
|
blocking(pool, move |conn| ModRemoveComment::create(conn, &form)).await??;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// Send the apub message
|
|
|
|
|
if deleted {
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_delete(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
} else {
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_undo_delete(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let post_id = data.post_id;
|
|
|
|
|
// Refetch it
|
|
|
|
|
let edit_id = data.edit_id;
|
|
|
|
|
let comment_view = blocking(pool, move |conn| {
|
|
|
|
|
CommentView::read(conn, edit_id, Some(user_id))
|
|
|
|
|
})
|
|
|
|
|
.await??;
|
|
|
|
|
|
|
|
|
|
// Build the recipients
|
|
|
|
|
let post_id = comment_view.post_id;
|
|
|
|
|
let post = blocking(pool, move |conn| Post::read(conn, post_id)).await??;
|
|
|
|
|
let mentions = vec![];
|
|
|
|
|
let recipient_ids =
|
|
|
|
|
send_local_notifs(mentions, updated_comment, user, post, pool, false).await?;
|
|
|
|
|
|
|
|
|
|
let mentions = scrape_text_for_mentions(&comment_form.content);
|
|
|
|
|
let recipient_ids = send_local_notifs(mentions, updated_comment, user, post, pool).await?;
|
|
|
|
|
let mut res = CommentResponse {
|
|
|
|
|
comment: comment_view,
|
|
|
|
|
recipient_ids,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if let Some(ws) = websocket_info {
|
|
|
|
|
ws.chatserver.do_send(SendComment {
|
|
|
|
|
op: UserOperation::DeleteComment,
|
|
|
|
|
comment: res.clone(),
|
|
|
|
|
my_id: ws.id,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// strip out the recipient_ids, so that
|
|
|
|
|
// users don't get double notifs
|
|
|
|
|
res.recipient_ids = Vec::new();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[async_trait::async_trait(?Send)]
|
|
|
|
|
impl Perform for Oper<RemoveComment> {
|
|
|
|
|
type Response = CommentResponse;
|
|
|
|
|
|
|
|
|
|
async fn perform(
|
|
|
|
|
&self,
|
|
|
|
|
pool: &DbPool,
|
|
|
|
|
websocket_info: Option<WebsocketInfo>,
|
|
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
|
|
|
|
let data: &RemoveComment = &self.data;
|
|
|
|
|
|
|
|
|
|
let claims = match Claims::decode(&data.auth) {
|
|
|
|
|
Ok(claims) => claims.claims,
|
|
|
|
|
Err(_e) => return Err(APIError::err("not_logged_in").into()),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let user_id = claims.id;
|
|
|
|
|
|
|
|
|
|
let edit_id = data.edit_id;
|
|
|
|
|
let orig_comment =
|
|
|
|
|
blocking(pool, move |conn| CommentView::read(&conn, edit_id, None)).await??;
|
|
|
|
|
|
|
|
|
|
// Check for a site ban
|
|
|
|
|
let user = blocking(pool, move |conn| User_::read(conn, user_id)).await??;
|
|
|
|
|
if user.banned {
|
|
|
|
|
return Err(APIError::err("site_ban").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for a community ban
|
|
|
|
|
let community_id = orig_comment.community_id;
|
|
|
|
|
let is_banned =
|
|
|
|
|
move |conn: &'_ _| CommunityUserBanView::get(conn, user_id, community_id).is_ok();
|
|
|
|
|
if blocking(pool, is_banned).await? {
|
|
|
|
|
return Err(APIError::err("community_ban").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Verify that only a mod or admin can remove
|
|
|
|
|
let mods_and_admins = blocking(pool, move |conn| {
|
|
|
|
|
Community::community_mods_and_admins(conn, community_id)
|
|
|
|
|
})
|
|
|
|
|
.await??;
|
|
|
|
|
if !mods_and_admins.contains(&user_id) {
|
|
|
|
|
return Err(APIError::err("not_an_admin").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Do the remove
|
|
|
|
|
let removed = data.removed;
|
|
|
|
|
let updated_comment = match blocking(pool, move |conn| {
|
|
|
|
|
Comment::update_removed(conn, edit_id, removed)
|
|
|
|
|
})
|
|
|
|
|
.await?
|
|
|
|
|
{
|
|
|
|
|
Ok(comment) => comment,
|
|
|
|
|
Err(_e) => return Err(APIError::err("couldnt_update_comment").into()),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Mod tables
|
|
|
|
|
let form = ModRemoveCommentForm {
|
|
|
|
|
mod_user_id: user_id,
|
|
|
|
|
comment_id: data.edit_id,
|
|
|
|
|
removed: Some(removed),
|
|
|
|
|
reason: data.reason.to_owned(),
|
|
|
|
|
};
|
|
|
|
|
blocking(pool, move |conn| ModRemoveComment::create(conn, &form)).await??;
|
|
|
|
|
|
|
|
|
|
// Send the apub message
|
|
|
|
|
if removed {
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_remove(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
} else {
|
|
|
|
|
updated_comment
|
|
|
|
|
.send_undo_remove(&user, &self.client, pool)
|
|
|
|
|
.await?;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Refetch it
|
|
|
|
|
let edit_id = data.edit_id;
|
|
|
|
|
let comment_view = blocking(pool, move |conn| {
|
|
|
|
|
CommentView::read(conn, edit_id, Some(user_id))
|
|
|
|
|
})
|
|
|
|
|
.await??;
|
|
|
|
|
|
|
|
|
|
// Build the recipients
|
|
|
|
|
let post_id = comment_view.post_id;
|
|
|
|
|
let post = blocking(pool, move |conn| Post::read(conn, post_id)).await??;
|
|
|
|
|
let mentions = vec![];
|
|
|
|
|
let recipient_ids =
|
|
|
|
|
send_local_notifs(mentions, updated_comment, user, post, pool, false).await?;
|
|
|
|
|
|
|
|
|
|
let mut res = CommentResponse {
|
|
|
|
|
comment: comment_view,
|
|
|
|
|
recipient_ids,
|
|
|
|
@ -422,7 +529,7 @@ impl Perform for Oper<EditComment> {
|
|
|
|
|
|
|
|
|
|
if let Some(ws) = websocket_info {
|
|
|
|
|
ws.chatserver.do_send(SendComment {
|
|
|
|
|
op: UserOperation::EditComment,
|
|
|
|
|
op: UserOperation::RemoveComment,
|
|
|
|
|
comment: res.clone(),
|
|
|
|
|
my_id: ws.id,
|
|
|
|
|
});
|
|
|
|
@ -436,6 +543,85 @@ impl Perform for Oper<EditComment> {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[async_trait::async_trait(?Send)]
|
|
|
|
|
impl Perform for Oper<MarkCommentAsRead> {
|
|
|
|
|
type Response = CommentResponse;
|
|
|
|
|
|
|
|
|
|
async fn perform(
|
|
|
|
|
&self,
|
|
|
|
|
pool: &DbPool,
|
|
|
|
|
_websocket_info: Option<WebsocketInfo>,
|
|
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
|
|
|
|
let data: &MarkCommentAsRead = &self.data;
|
|
|
|
|
|
|
|
|
|
let claims = match Claims::decode(&data.auth) {
|
|
|
|
|
Ok(claims) => claims.claims,
|
|
|
|
|
Err(_e) => return Err(APIError::err("not_logged_in").into()),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let user_id = claims.id;
|
|
|
|
|
|
|
|
|
|
let edit_id = data.edit_id;
|
|
|
|
|
let orig_comment =
|
|
|
|
|
blocking(pool, move |conn| CommentView::read(&conn, edit_id, None)).await??;
|
|
|
|
|
|
|
|
|
|
// Check for a site ban
|
|
|
|
|
let user = blocking(pool, move |conn| User_::read(conn, user_id)).await??;
|
|
|
|
|
if user.banned {
|
|
|
|
|
return Err(APIError::err("site_ban").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for a community ban
|
|
|
|
|
let community_id = orig_comment.community_id;
|
|
|
|
|
let is_banned =
|
|
|
|
|
move |conn: &'_ _| CommunityUserBanView::get(conn, user_id, community_id).is_ok();
|
|
|
|
|
if blocking(pool, is_banned).await? {
|
|
|
|
|
return Err(APIError::err("community_ban").into());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Verify that only the recipient can mark as read
|
|
|
|
|
// Needs to fetch the parent comment / post to get the recipient
|
|
|
|
|
let parent_id = orig_comment.parent_id;
|
|
|
|
|
match parent_id {
|
|
|
|
|
Some(pid) => {
|
|
|
|
|
let parent_comment =
|
|
|
|
|
blocking(pool, move |conn| CommentView::read(&conn, pid, None)).await??;
|
|
|
|
|
if user_id != parent_comment.creator_id {
|
|
|
|
|
return Err(APIError::err("no_comment_edit_allowed").into());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
None => {
|
|
|
|
|
let parent_post_id = orig_comment.post_id;
|
|
|
|
|
let parent_post = blocking(pool, move |conn| Post::read(conn, parent_post_id)).await??;
|
|
|
|
|
if user_id != parent_post.creator_id {
|
|
|
|
|
return Err(APIError::err("no_comment_edit_allowed").into());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Do the mark as read
|
|
|
|
|
let read = data.read;
|
|
|
|
|
match blocking(pool, move |conn| Comment::update_read(conn, edit_id, read)).await? {
|
|
|
|
|
Ok(comment) => comment,
|
|
|
|
|
Err(_e) => return Err(APIError::err("couldnt_update_comment").into()),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Refetch it
|
|
|
|
|
let edit_id = data.edit_id;
|
|
|
|
|
let comment_view = blocking(pool, move |conn| {
|
|
|
|
|
CommentView::read(conn, edit_id, Some(user_id))
|
|
|
|
|
})
|
|
|
|
|
.await??;
|
|
|
|
|
|
|
|
|
|
let res = CommentResponse {
|
|
|
|
|
comment: comment_view,
|
|
|
|
|
recipient_ids: Vec::new(),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[async_trait::async_trait(?Send)]
|
|
|
|
|
impl Perform for Oper<SaveComment> {
|
|
|
|
|
type Response = CommentResponse;
|
|
|
|
@ -512,8 +698,12 @@ impl Perform for Oper<CreateCommentLike> {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let comment_id = data.comment_id;
|
|
|
|
|
let orig_comment =
|
|
|
|
|
blocking(pool, move |conn| CommentView::read(&conn, comment_id, None)).await??;
|
|
|
|
|
|
|
|
|
|
// Check for a community ban
|
|
|
|
|
let post_id = data.post_id;
|
|
|
|
|
let post_id = orig_comment.post_id;
|
|
|
|
|
let post = blocking(pool, move |conn| Post::read(conn, post_id)).await??;
|
|
|
|
|
let community_id = post.community_id;
|
|
|
|
|
let is_banned =
|
|
|
|
@ -550,7 +740,7 @@ impl Perform for Oper<CreateCommentLike> {
|
|
|
|
|
|
|
|
|
|
let like_form = CommentLikeForm {
|
|
|
|
|
comment_id: data.comment_id,
|
|
|
|
|
post_id: data.post_id,
|
|
|
|
|
post_id,
|
|
|
|
|
user_id,
|
|
|
|
|
score: data.score,
|
|
|
|
|
};
|
|
|
|
@ -675,9 +865,10 @@ pub async fn send_local_notifs(
|
|
|
|
|
user: User_,
|
|
|
|
|
post: Post,
|
|
|
|
|
pool: &DbPool,
|
|
|
|
|
do_send_email: bool,
|
|
|
|
|
) -> Result<Vec<i32>, LemmyError> {
|
|
|
|
|
let ids = blocking(pool, move |conn| {
|
|
|
|
|
do_send_local_notifs(conn, &mentions, &comment, &user, &post)
|
|
|
|
|
do_send_local_notifs(conn, &mentions, &comment, &user, &post, do_send_email)
|
|
|
|
|
})
|
|
|
|
|
.await?;
|
|
|
|
|
|
|
|
|
@ -690,6 +881,7 @@ fn do_send_local_notifs(
|
|
|
|
|
comment: &Comment,
|
|
|
|
|
user: &User_,
|
|
|
|
|
post: &Post,
|
|
|
|
|
do_send_email: bool,
|
|
|
|
|
) -> Vec<i32> {
|
|
|
|
|
let mut recipient_ids = Vec::new();
|
|
|
|
|
let hostname = &format!("https://{}", Settings::get().hostname);
|
|
|
|
@ -720,7 +912,7 @@ fn do_send_local_notifs(
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Send an email to those users that have notifications on
|
|
|
|
|
if mention_user.send_notifications_to_email {
|
|
|
|
|
if do_send_email && mention_user.send_notifications_to_email {
|
|
|
|
|
if let Some(mention_email) = mention_user.email {
|
|
|
|
|
let subject = &format!("{} - Mentioned by {}", Settings::get().hostname, user.name,);
|
|
|
|
|
let html = &format!(
|
|
|
|
@ -744,7 +936,7 @@ fn do_send_local_notifs(
|
|
|
|
|
if let Ok(parent_user) = User_::read(&conn, parent_comment.creator_id) {
|
|
|
|
|
recipient_ids.push(parent_user.id);
|
|
|
|
|
|
|
|
|
|
if parent_user.send_notifications_to_email {
|
|
|
|
|
if do_send_email && parent_user.send_notifications_to_email {
|
|
|
|
|
if let Some(comment_reply_email) = parent_user.email {
|
|
|
|
|
let subject = &format!("{} - Reply from {}", Settings::get().hostname, user.name,);
|
|
|
|
|
let html = &format!(
|
|
|
|
@ -767,7 +959,7 @@ fn do_send_local_notifs(
|
|
|
|
|
if let Ok(parent_user) = User_::read(&conn, post.creator_id) {
|
|
|
|
|
recipient_ids.push(parent_user.id);
|
|
|
|
|
|
|
|
|
|
if parent_user.send_notifications_to_email {
|
|
|
|
|
if do_send_email && parent_user.send_notifications_to_email {
|
|
|
|
|
if let Some(post_reply_email) = parent_user.email {
|
|
|
|
|
let subject = &format!("{} - Reply from {}", Settings::get().hostname, user.name,);
|
|
|
|
|
let html = &format!(
|
|
|
|
|