2020-05-16 14:04:08 +00:00
|
|
|
use crate::{
|
2020-09-24 13:53:21 +00:00
|
|
|
check_community_ban,
|
2020-12-20 01:10:47 +00:00
|
|
|
check_downvotes_enabled,
|
2020-11-04 02:15:11 +00:00
|
|
|
collect_moderated_communities,
|
2021-03-10 22:33:55 +00:00
|
|
|
get_local_user_view_from_jwt,
|
|
|
|
get_local_user_view_from_jwt_opt,
|
2021-03-11 04:43:11 +00:00
|
|
|
get_post,
|
2020-09-24 13:53:21 +00:00
|
|
|
is_mod_or_admin,
|
|
|
|
Perform,
|
2020-07-10 18:15:41 +00:00
|
|
|
};
|
2020-08-18 13:43:50 +00:00
|
|
|
use actix_web::web::Data;
|
2021-03-01 13:08:41 +00:00
|
|
|
use lemmy_api_structs::{blocking, comment::*, send_local_notifs};
|
2021-02-04 16:34:58 +00:00
|
|
|
use lemmy_apub::{generate_apub_endpoint, ApubLikeableType, ApubObjectType, EndpointType};
|
2020-12-21 23:27:42 +00:00
|
|
|
use lemmy_db_queries::{
|
2020-12-21 13:38:34 +00:00
|
|
|
source::comment::Comment_,
|
2020-07-10 18:15:41 +00:00
|
|
|
Crud,
|
|
|
|
Likeable,
|
|
|
|
ListingType,
|
2020-11-04 02:15:11 +00:00
|
|
|
Reportable,
|
2020-07-10 18:15:41 +00:00
|
|
|
Saveable,
|
|
|
|
SortType,
|
|
|
|
};
|
2021-03-18 20:25:21 +00:00
|
|
|
use lemmy_db_schema::{
|
|
|
|
source::{comment::*, comment_report::*, moderator::*},
|
|
|
|
LocalUserId,
|
|
|
|
};
|
2020-12-21 16:30:34 +00:00
|
|
|
use lemmy_db_views::{
|
|
|
|
comment_report_view::{CommentReportQueryBuilder, CommentReportView},
|
|
|
|
comment_view::{CommentQueryBuilder, CommentView},
|
2021-03-18 20:25:21 +00:00
|
|
|
local_user_view::LocalUserView,
|
2020-12-21 16:30:34 +00:00
|
|
|
};
|
2020-07-10 18:15:41 +00:00
|
|
|
use lemmy_utils::{
|
2020-09-14 15:29:50 +00:00
|
|
|
utils::{remove_slurs, scrape_text_for_mentions},
|
2021-02-22 18:04:32 +00:00
|
|
|
ApiError,
|
2020-09-01 14:25:34 +00:00
|
|
|
ConnectionId,
|
|
|
|
LemmyError,
|
2020-05-16 14:04:08 +00:00
|
|
|
};
|
2020-11-04 02:15:11 +00:00
|
|
|
use lemmy_websocket::{
|
|
|
|
messages::{SendComment, SendModRoomMessage, SendUserRoomMessage},
|
|
|
|
LemmyContext,
|
|
|
|
UserOperation,
|
|
|
|
};
|
2020-05-16 14:04:08 +00:00
|
|
|
use std::str::FromStr;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
#[async_trait::async_trait(?Send)]
|
2020-08-12 11:31:45 +00:00
|
|
|
impl Perform for CreateComment {
|
2020-04-20 03:59:07 +00:00
|
|
|
type Response = CommentResponse;
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
async fn perform(
|
2020-04-19 22:08:25 +00:00
|
|
|
&self,
|
2020-08-18 13:43:50 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2020-08-24 11:58:24 +00:00
|
|
|
websocket_id: Option<ConnectionId>,
|
2020-07-01 12:54:29 +00:00
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
2020-08-12 11:31:45 +00:00
|
|
|
let data: &CreateComment = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
|
|
|
let content_slurs_removed = remove_slurs(&data.content.to_owned());
|
|
|
|
|
2021-01-06 21:02:08 +00:00
|
|
|
// Check for a community ban
|
|
|
|
let post_id = data.post_id;
|
|
|
|
let post = get_post(post_id, context.pool()).await?;
|
|
|
|
|
2021-03-10 22:33:55 +00:00
|
|
|
check_community_ban(local_user_view.person.id, post.community_id, context.pool()).await?;
|
2021-01-06 21:02:08 +00:00
|
|
|
|
|
|
|
// Check if post is locked, no new comments
|
|
|
|
if post.locked {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("locked").into());
|
2021-01-06 21:02:08 +00:00
|
|
|
}
|
|
|
|
|
2021-01-31 04:10:16 +00:00
|
|
|
// If there's a parent_id, check to make sure that comment is in that post
|
|
|
|
if let Some(parent_id) = data.parent_id {
|
|
|
|
// Make sure the parent comment exists
|
|
|
|
let parent =
|
|
|
|
match blocking(context.pool(), move |conn| Comment::read(&conn, parent_id)).await? {
|
|
|
|
Ok(comment) => comment,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_e) => return Err(ApiError::err("couldnt_create_comment").into()),
|
2021-01-31 04:10:16 +00:00
|
|
|
};
|
|
|
|
if parent.post_id != post_id {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("couldnt_create_comment").into());
|
2021-01-31 04:10:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-05 05:20:38 +00:00
|
|
|
let comment_form = CommentForm {
|
|
|
|
content: content_slurs_removed,
|
|
|
|
parent_id: data.parent_id.to_owned(),
|
|
|
|
post_id: data.post_id,
|
2021-03-10 22:33:55 +00:00
|
|
|
creator_id: local_user_view.person.id,
|
2019-05-05 05:20:38 +00:00
|
|
|
removed: None,
|
|
|
|
deleted: None,
|
|
|
|
read: None,
|
2020-04-27 16:57:00 +00:00
|
|
|
published: None,
|
2019-09-07 15:35:05 +00:00
|
|
|
updated: None,
|
2020-08-31 13:48:02 +00:00
|
|
|
ap_id: None,
|
2020-04-04 00:04:57 +00:00
|
|
|
local: true,
|
2019-05-05 05:20:38 +00:00
|
|
|
};
|
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
// Create the comment
|
2020-07-01 12:54:29 +00:00
|
|
|
let comment_form2 = comment_form.clone();
|
2020-08-18 13:43:50 +00:00
|
|
|
let inserted_comment = match blocking(context.pool(), move |conn| {
|
|
|
|
Comment::create(&conn, &comment_form2)
|
|
|
|
})
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
Ok(comment) => comment,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_e) => return Err(ApiError::err("couldnt_create_comment").into()),
|
2020-08-18 13:43:50 +00:00
|
|
|
};
|
2020-07-01 12:54:29 +00:00
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
// Necessary to update the ap_id
|
2020-07-01 12:54:29 +00:00
|
|
|
let inserted_comment_id = inserted_comment.id;
|
2021-02-04 16:34:58 +00:00
|
|
|
let updated_comment: Comment =
|
|
|
|
match blocking(context.pool(), move |conn| -> Result<Comment, LemmyError> {
|
|
|
|
let apub_id =
|
|
|
|
generate_apub_endpoint(EndpointType::Comment, &inserted_comment_id.to_string())?;
|
|
|
|
Ok(Comment::update_ap_id(&conn, inserted_comment_id, apub_id)?)
|
|
|
|
})
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
Ok(comment) => comment,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_e) => return Err(ApiError::err("couldnt_create_comment").into()),
|
2021-02-04 16:34:58 +00:00
|
|
|
};
|
2020-04-04 00:04:57 +00:00
|
|
|
|
2021-03-11 04:43:11 +00:00
|
|
|
updated_comment
|
|
|
|
.send_create(&local_user_view.person, context)
|
|
|
|
.await?;
|
2020-04-27 16:57:00 +00:00
|
|
|
|
2019-10-20 00:46:29 +00:00
|
|
|
// Scan the comment for user mentions, add those rows
|
2020-12-20 01:10:47 +00:00
|
|
|
let post_id = post.id;
|
2020-05-15 16:36:11 +00:00
|
|
|
let mentions = scrape_text_for_mentions(&comment_form.content);
|
2020-08-18 13:43:50 +00:00
|
|
|
let recipient_ids = send_local_notifs(
|
|
|
|
mentions,
|
|
|
|
updated_comment.clone(),
|
2021-03-10 22:33:55 +00:00
|
|
|
local_user_view.person.clone(),
|
2020-08-18 13:43:50 +00:00
|
|
|
post,
|
|
|
|
context.pool(),
|
|
|
|
true,
|
|
|
|
)
|
|
|
|
.await?;
|
2020-01-02 21:55:54 +00:00
|
|
|
|
2019-05-05 05:20:38 +00:00
|
|
|
// You like your own comment by default
|
|
|
|
let like_form = CommentLikeForm {
|
|
|
|
comment_id: inserted_comment.id,
|
2020-12-20 01:10:47 +00:00
|
|
|
post_id,
|
2021-03-10 22:33:55 +00:00
|
|
|
person_id: local_user_view.person.id,
|
2019-09-07 15:35:05 +00:00
|
|
|
score: 1,
|
2019-05-05 05:20:38 +00:00
|
|
|
};
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
let like = move |conn: &'_ _| CommentLike::like(&conn, &like_form);
|
2020-08-18 13:43:50 +00:00
|
|
|
if blocking(context.pool(), like).await?.is_err() {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("couldnt_like_comment").into());
|
2020-07-01 12:54:29 +00:00
|
|
|
}
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2021-03-11 04:43:11 +00:00
|
|
|
updated_comment
|
|
|
|
.send_like(&local_user_view.person, context)
|
|
|
|
.await?;
|
2020-04-28 04:16:02 +00:00
|
|
|
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2021-01-06 21:02:08 +00:00
|
|
|
let mut comment_view = blocking(context.pool(), move |conn| {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentView::read(&conn, inserted_comment.id, Some(person_id))
|
2020-07-01 12:54:29 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2021-01-06 21:02:08 +00:00
|
|
|
// If its a comment to yourself, mark it as read
|
|
|
|
let comment_id = comment_view.comment.id;
|
2021-03-10 22:33:55 +00:00
|
|
|
if local_user_view.person.id == comment_view.get_recipient_id() {
|
2021-01-06 21:02:08 +00:00
|
|
|
match blocking(context.pool(), move |conn| {
|
|
|
|
Comment::update_read(conn, comment_id, true)
|
|
|
|
})
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
Ok(comment) => comment,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_e) => return Err(ApiError::err("couldnt_update_comment").into()),
|
2021-01-06 21:02:08 +00:00
|
|
|
};
|
|
|
|
comment_view.comment.read = true;
|
|
|
|
}
|
|
|
|
|
2021-01-13 17:01:42 +00:00
|
|
|
let mut res = CommentResponse {
|
2020-12-15 19:39:18 +00:00
|
|
|
comment_view,
|
2020-02-01 01:02:20 +00:00
|
|
|
recipient_ids,
|
2020-07-21 14:56:41 +00:00
|
|
|
form_id: data.form_id.to_owned(),
|
2020-04-19 22:08:25 +00:00
|
|
|
};
|
|
|
|
|
2020-08-24 11:58:24 +00:00
|
|
|
context.chat_server().do_send(SendComment {
|
|
|
|
op: UserOperation::CreateComment,
|
|
|
|
comment: res.clone(),
|
|
|
|
websocket_id,
|
|
|
|
});
|
2020-04-19 22:08:25 +00:00
|
|
|
|
2021-01-13 17:01:42 +00:00
|
|
|
res.recipient_ids = Vec::new(); // Necessary to avoid doubles
|
|
|
|
|
2020-04-19 22:08:25 +00:00
|
|
|
Ok(res)
|
2019-05-05 05:20:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
#[async_trait::async_trait(?Send)]
|
2020-08-12 11:31:45 +00:00
|
|
|
impl Perform for EditComment {
|
2020-04-20 03:59:07 +00:00
|
|
|
type Response = CommentResponse;
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
async fn perform(
|
2020-04-19 22:08:25 +00:00
|
|
|
&self,
|
2020-08-18 13:43:50 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2020-08-24 11:58:24 +00:00
|
|
|
websocket_id: Option<ConnectionId>,
|
2020-07-01 12:54:29 +00:00
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
2020-08-12 11:31:45 +00:00
|
|
|
let data: &EditComment = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2021-01-18 21:57:31 +00:00
|
|
|
let comment_id = data.comment_id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let orig_comment = blocking(context.pool(), move |conn| {
|
2021-01-18 21:57:31 +00:00
|
|
|
CommentView::read(&conn, comment_id, None)
|
2020-08-18 13:43:50 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2021-03-11 04:43:11 +00:00
|
|
|
check_community_ban(
|
|
|
|
local_user_view.person.id,
|
|
|
|
orig_comment.community.id,
|
|
|
|
context.pool(),
|
|
|
|
)
|
|
|
|
.await?;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
// Verify that only the creator can edit
|
2021-03-10 22:33:55 +00:00
|
|
|
if local_user_view.person.id != orig_comment.creator.id {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("no_comment_edit_allowed").into());
|
2019-05-05 05:20:38 +00:00
|
|
|
}
|
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
// Do the update
|
2019-05-05 05:20:38 +00:00
|
|
|
let content_slurs_removed = remove_slurs(&data.content.to_owned());
|
2021-01-18 21:57:31 +00:00
|
|
|
let comment_id = data.comment_id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let updated_comment = match blocking(context.pool(), move |conn| {
|
2021-01-18 21:57:31 +00:00
|
|
|
Comment::update_content(conn, comment_id, &content_slurs_removed)
|
2020-07-21 01:37:44 +00:00
|
|
|
})
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
Ok(comment) => comment,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_e) => return Err(ApiError::err("couldnt_update_comment").into()),
|
2020-07-21 01:37:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Send the apub update
|
2021-03-11 04:43:11 +00:00
|
|
|
updated_comment
|
|
|
|
.send_update(&local_user_view.person, context)
|
|
|
|
.await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
|
|
|
// Do the mentions / recipients
|
|
|
|
let updated_comment_content = updated_comment.content.to_owned();
|
|
|
|
let mentions = scrape_text_for_mentions(&updated_comment_content);
|
2020-08-18 13:43:50 +00:00
|
|
|
let recipient_ids = send_local_notifs(
|
|
|
|
mentions,
|
|
|
|
updated_comment,
|
2021-03-10 22:33:55 +00:00
|
|
|
local_user_view.person.clone(),
|
2020-12-20 01:10:47 +00:00
|
|
|
orig_comment.post,
|
2020-08-18 13:43:50 +00:00
|
|
|
context.pool(),
|
|
|
|
false,
|
|
|
|
)
|
|
|
|
.await?;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2021-01-18 21:57:31 +00:00
|
|
|
let comment_id = data.comment_id;
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let comment_view = blocking(context.pool(), move |conn| {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentView::read(conn, comment_id, Some(person_id))
|
2020-07-21 01:37:44 +00:00
|
|
|
})
|
|
|
|
.await??;
|
|
|
|
|
2021-01-07 06:17:42 +00:00
|
|
|
let res = CommentResponse {
|
2020-12-15 19:39:18 +00:00
|
|
|
comment_view,
|
2020-07-21 01:37:44 +00:00
|
|
|
recipient_ids,
|
2020-07-21 14:56:41 +00:00
|
|
|
form_id: data.form_id.to_owned(),
|
2019-05-05 05:20:38 +00:00
|
|
|
};
|
|
|
|
|
2020-08-24 11:58:24 +00:00
|
|
|
context.chat_server().do_send(SendComment {
|
|
|
|
op: UserOperation::EditComment,
|
|
|
|
comment: res.clone(),
|
|
|
|
websocket_id,
|
|
|
|
});
|
2020-07-21 01:37:44 +00:00
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[async_trait::async_trait(?Send)]
|
2020-08-12 11:31:45 +00:00
|
|
|
impl Perform for DeleteComment {
|
2020-07-21 01:37:44 +00:00
|
|
|
type Response = CommentResponse;
|
|
|
|
|
|
|
|
async fn perform(
|
|
|
|
&self,
|
2020-08-18 13:43:50 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2020-08-24 11:58:24 +00:00
|
|
|
websocket_id: Option<ConnectionId>,
|
2020-07-21 01:37:44 +00:00
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
2020-08-12 11:31:45 +00:00
|
|
|
let data: &DeleteComment = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
2021-01-18 21:57:31 +00:00
|
|
|
let comment_id = data.comment_id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let orig_comment = blocking(context.pool(), move |conn| {
|
2021-01-18 21:57:31 +00:00
|
|
|
CommentView::read(&conn, comment_id, None)
|
2020-08-18 13:43:50 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
2021-03-11 04:43:11 +00:00
|
|
|
check_community_ban(
|
|
|
|
local_user_view.person.id,
|
|
|
|
orig_comment.community.id,
|
|
|
|
context.pool(),
|
|
|
|
)
|
|
|
|
.await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
|
|
|
// Verify that only the creator can delete
|
2021-03-10 22:33:55 +00:00
|
|
|
if local_user_view.person.id != orig_comment.creator.id {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("no_comment_edit_allowed").into());
|
2020-07-21 01:37:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Do the delete
|
|
|
|
let deleted = data.deleted;
|
2020-08-18 13:43:50 +00:00
|
|
|
let updated_comment = match blocking(context.pool(), move |conn| {
|
2021-01-18 21:57:31 +00:00
|
|
|
Comment::update_deleted(conn, comment_id, deleted)
|
2020-07-01 12:54:29 +00:00
|
|
|
})
|
|
|
|
.await?
|
|
|
|
{
|
2019-05-05 05:20:38 +00:00
|
|
|
Ok(comment) => comment,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_e) => return Err(ApiError::err("couldnt_update_comment").into()),
|
2019-05-05 05:20:38 +00:00
|
|
|
};
|
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
// Send the apub message
|
|
|
|
if deleted {
|
2021-03-11 04:43:11 +00:00
|
|
|
updated_comment
|
|
|
|
.send_delete(&local_user_view.person, context)
|
|
|
|
.await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
} else {
|
2021-03-11 04:43:11 +00:00
|
|
|
updated_comment
|
|
|
|
.send_undo_delete(&local_user_view.person, context)
|
|
|
|
.await?;
|
2020-04-29 14:51:25 +00:00
|
|
|
}
|
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
// Refetch it
|
2021-01-18 21:57:31 +00:00
|
|
|
let comment_id = data.comment_id;
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let comment_view = blocking(context.pool(), move |conn| {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentView::read(conn, comment_id, Some(person_id))
|
2020-07-21 01:37:44 +00:00
|
|
|
})
|
|
|
|
.await??;
|
|
|
|
|
|
|
|
// Build the recipients
|
2020-12-20 01:10:47 +00:00
|
|
|
let comment_view_2 = comment_view.clone();
|
2020-07-21 01:37:44 +00:00
|
|
|
let mentions = vec![];
|
2020-08-18 13:43:50 +00:00
|
|
|
let recipient_ids = send_local_notifs(
|
|
|
|
mentions,
|
|
|
|
updated_comment,
|
2021-03-10 22:33:55 +00:00
|
|
|
local_user_view.person.clone(),
|
2020-12-20 01:10:47 +00:00
|
|
|
comment_view_2.post,
|
2020-08-18 13:43:50 +00:00
|
|
|
context.pool(),
|
|
|
|
false,
|
|
|
|
)
|
|
|
|
.await?;
|
2019-10-20 00:46:29 +00:00
|
|
|
|
2021-01-07 06:17:42 +00:00
|
|
|
let res = CommentResponse {
|
2020-12-15 19:39:18 +00:00
|
|
|
comment_view,
|
2020-07-21 01:37:44 +00:00
|
|
|
recipient_ids,
|
2020-12-20 01:10:47 +00:00
|
|
|
form_id: None, // TODO a comment delete might clear forms?
|
2020-07-21 01:37:44 +00:00
|
|
|
};
|
|
|
|
|
2020-08-24 11:58:24 +00:00
|
|
|
context.chat_server().do_send(SendComment {
|
|
|
|
op: UserOperation::DeleteComment,
|
|
|
|
comment: res.clone(),
|
|
|
|
websocket_id,
|
|
|
|
});
|
2020-07-21 01:37:44 +00:00
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[async_trait::async_trait(?Send)]
|
2020-08-12 11:31:45 +00:00
|
|
|
impl Perform for RemoveComment {
|
2020-07-21 01:37:44 +00:00
|
|
|
type Response = CommentResponse;
|
|
|
|
|
|
|
|
async fn perform(
|
|
|
|
&self,
|
2020-08-18 13:43:50 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2020-08-24 11:58:24 +00:00
|
|
|
websocket_id: Option<ConnectionId>,
|
2020-07-21 01:37:44 +00:00
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
2020-08-12 11:31:45 +00:00
|
|
|
let data: &RemoveComment = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
2021-01-18 21:57:31 +00:00
|
|
|
let comment_id = data.comment_id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let orig_comment = blocking(context.pool(), move |conn| {
|
2021-01-18 21:57:31 +00:00
|
|
|
CommentView::read(&conn, comment_id, None)
|
2020-08-18 13:43:50 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
2021-03-11 04:43:11 +00:00
|
|
|
check_community_ban(
|
|
|
|
local_user_view.person.id,
|
|
|
|
orig_comment.community.id,
|
|
|
|
context.pool(),
|
|
|
|
)
|
|
|
|
.await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
|
|
|
// Verify that only a mod or admin can remove
|
2021-03-11 04:43:11 +00:00
|
|
|
is_mod_or_admin(
|
|
|
|
context.pool(),
|
|
|
|
local_user_view.person.id,
|
|
|
|
orig_comment.community.id,
|
|
|
|
)
|
|
|
|
.await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
|
|
|
// Do the remove
|
|
|
|
let removed = data.removed;
|
2020-08-18 13:43:50 +00:00
|
|
|
let updated_comment = match blocking(context.pool(), move |conn| {
|
2021-01-18 21:57:31 +00:00
|
|
|
Comment::update_removed(conn, comment_id, removed)
|
2020-07-21 01:37:44 +00:00
|
|
|
})
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
Ok(comment) => comment,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_e) => return Err(ApiError::err("couldnt_update_comment").into()),
|
2020-07-21 01:37:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Mod tables
|
|
|
|
let form = ModRemoveCommentForm {
|
2021-03-10 22:33:55 +00:00
|
|
|
mod_person_id: local_user_view.person.id,
|
2021-01-18 21:57:31 +00:00
|
|
|
comment_id: data.comment_id,
|
2020-07-21 01:37:44 +00:00
|
|
|
removed: Some(removed),
|
|
|
|
reason: data.reason.to_owned(),
|
|
|
|
};
|
2020-08-18 13:43:50 +00:00
|
|
|
blocking(context.pool(), move |conn| {
|
|
|
|
ModRemoveComment::create(conn, &form)
|
|
|
|
})
|
|
|
|
.await??;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
|
|
|
// Send the apub message
|
|
|
|
if removed {
|
2021-03-11 04:43:11 +00:00
|
|
|
updated_comment
|
|
|
|
.send_remove(&local_user_view.person, context)
|
|
|
|
.await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
} else {
|
2021-03-11 04:43:11 +00:00
|
|
|
updated_comment
|
|
|
|
.send_undo_remove(&local_user_view.person, context)
|
|
|
|
.await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
}
|
2020-02-01 01:02:20 +00:00
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
// Refetch it
|
2021-01-18 21:57:31 +00:00
|
|
|
let comment_id = data.comment_id;
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let comment_view = blocking(context.pool(), move |conn| {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentView::read(conn, comment_id, Some(person_id))
|
2020-07-01 12:54:29 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
// Build the recipients
|
2020-12-20 01:10:47 +00:00
|
|
|
let comment_view_2 = comment_view.clone();
|
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
let mentions = vec![];
|
2020-08-18 13:43:50 +00:00
|
|
|
let recipient_ids = send_local_notifs(
|
|
|
|
mentions,
|
|
|
|
updated_comment,
|
2021-03-10 22:33:55 +00:00
|
|
|
local_user_view.person.clone(),
|
2020-12-20 01:10:47 +00:00
|
|
|
comment_view_2.post,
|
2020-08-18 13:43:50 +00:00
|
|
|
context.pool(),
|
|
|
|
false,
|
|
|
|
)
|
|
|
|
.await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
2021-01-07 06:17:42 +00:00
|
|
|
let res = CommentResponse {
|
2020-12-15 19:39:18 +00:00
|
|
|
comment_view,
|
2020-02-01 01:02:20 +00:00
|
|
|
recipient_ids,
|
2020-12-20 01:10:47 +00:00
|
|
|
form_id: None, // TODO maybe this might clear other forms
|
2020-04-19 22:08:25 +00:00
|
|
|
};
|
|
|
|
|
2020-08-24 11:58:24 +00:00
|
|
|
context.chat_server().do_send(SendComment {
|
|
|
|
op: UserOperation::RemoveComment,
|
|
|
|
comment: res.clone(),
|
|
|
|
websocket_id,
|
|
|
|
});
|
2020-04-19 22:08:25 +00:00
|
|
|
|
|
|
|
Ok(res)
|
2019-05-05 05:20:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
#[async_trait::async_trait(?Send)]
|
2020-08-12 11:31:45 +00:00
|
|
|
impl Perform for MarkCommentAsRead {
|
2020-07-21 01:37:44 +00:00
|
|
|
type Response = CommentResponse;
|
|
|
|
|
|
|
|
async fn perform(
|
|
|
|
&self,
|
2020-08-18 13:43:50 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2020-08-24 11:58:24 +00:00
|
|
|
_websocket_id: Option<ConnectionId>,
|
2020-07-21 01:37:44 +00:00
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
2020-08-12 11:31:45 +00:00
|
|
|
let data: &MarkCommentAsRead = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
2020-12-20 01:10:47 +00:00
|
|
|
let comment_id = data.comment_id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let orig_comment = blocking(context.pool(), move |conn| {
|
2020-12-20 01:10:47 +00:00
|
|
|
CommentView::read(&conn, comment_id, None)
|
2020-08-18 13:43:50 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
2021-03-11 04:43:11 +00:00
|
|
|
check_community_ban(
|
|
|
|
local_user_view.person.id,
|
|
|
|
orig_comment.community.id,
|
|
|
|
context.pool(),
|
|
|
|
)
|
|
|
|
.await?;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
|
|
|
// Verify that only the recipient can mark as read
|
2021-03-10 22:33:55 +00:00
|
|
|
if local_user_view.person.id != orig_comment.get_recipient_id() {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("no_comment_edit_allowed").into());
|
2020-07-21 01:37:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Do the mark as read
|
|
|
|
let read = data.read;
|
2020-08-18 13:43:50 +00:00
|
|
|
match blocking(context.pool(), move |conn| {
|
2020-12-20 01:10:47 +00:00
|
|
|
Comment::update_read(conn, comment_id, read)
|
2020-08-18 13:43:50 +00:00
|
|
|
})
|
|
|
|
.await?
|
|
|
|
{
|
2020-07-21 01:37:44 +00:00
|
|
|
Ok(comment) => comment,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_e) => return Err(ApiError::err("couldnt_update_comment").into()),
|
2020-07-21 01:37:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Refetch it
|
2021-01-18 21:57:31 +00:00
|
|
|
let comment_id = data.comment_id;
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let comment_view = blocking(context.pool(), move |conn| {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentView::read(conn, comment_id, Some(person_id))
|
2020-07-21 01:37:44 +00:00
|
|
|
})
|
|
|
|
.await??;
|
|
|
|
|
|
|
|
let res = CommentResponse {
|
2020-12-15 19:39:18 +00:00
|
|
|
comment_view,
|
2020-07-21 01:37:44 +00:00
|
|
|
recipient_ids: Vec::new(),
|
2020-07-21 14:56:41 +00:00
|
|
|
form_id: None,
|
2020-07-21 01:37:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
#[async_trait::async_trait(?Send)]
|
2020-08-12 11:31:45 +00:00
|
|
|
impl Perform for SaveComment {
|
2020-04-20 03:59:07 +00:00
|
|
|
type Response = CommentResponse;
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
async fn perform(
|
2020-04-19 22:08:25 +00:00
|
|
|
&self,
|
2020-08-18 13:43:50 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2020-08-24 11:58:24 +00:00
|
|
|
_websocket_id: Option<ConnectionId>,
|
2020-07-01 12:54:29 +00:00
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
2020-08-12 11:31:45 +00:00
|
|
|
let data: &SaveComment = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
|
|
|
let comment_saved_form = CommentSavedForm {
|
|
|
|
comment_id: data.comment_id,
|
2021-03-10 22:33:55 +00:00
|
|
|
person_id: local_user_view.person.id,
|
2019-05-05 05:20:38 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
if data.save {
|
2020-07-01 12:54:29 +00:00
|
|
|
let save_comment = move |conn: &'_ _| CommentSaved::save(conn, &comment_saved_form);
|
2020-08-18 13:43:50 +00:00
|
|
|
if blocking(context.pool(), save_comment).await?.is_err() {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("couldnt_save_comment").into());
|
2020-07-01 12:54:29 +00:00
|
|
|
}
|
2019-05-05 05:20:38 +00:00
|
|
|
} else {
|
2020-07-01 12:54:29 +00:00
|
|
|
let unsave_comment = move |conn: &'_ _| CommentSaved::unsave(conn, &comment_saved_form);
|
2020-08-18 13:43:50 +00:00
|
|
|
if blocking(context.pool(), unsave_comment).await?.is_err() {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("couldnt_save_comment").into());
|
2020-07-01 12:54:29 +00:00
|
|
|
}
|
2019-05-05 05:20:38 +00:00
|
|
|
}
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
let comment_id = data.comment_id;
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let comment_view = blocking(context.pool(), move |conn| {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentView::read(conn, comment_id, Some(person_id))
|
2020-07-01 12:54:29 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2019-09-07 15:35:05 +00:00
|
|
|
Ok(CommentResponse {
|
2020-12-15 19:39:18 +00:00
|
|
|
comment_view,
|
2020-02-01 01:02:20 +00:00
|
|
|
recipient_ids: Vec::new(),
|
2020-07-21 14:56:41 +00:00
|
|
|
form_id: None,
|
2019-09-07 15:35:05 +00:00
|
|
|
})
|
2019-05-05 05:20:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
#[async_trait::async_trait(?Send)]
|
2020-08-12 11:31:45 +00:00
|
|
|
impl Perform for CreateCommentLike {
|
2020-04-20 03:59:07 +00:00
|
|
|
type Response = CommentResponse;
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
async fn perform(
|
2020-04-19 22:08:25 +00:00
|
|
|
&self,
|
2020-08-18 13:43:50 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2020-08-24 11:58:24 +00:00
|
|
|
websocket_id: Option<ConnectionId>,
|
2020-07-01 12:54:29 +00:00
|
|
|
) -> Result<CommentResponse, LemmyError> {
|
2020-08-12 11:31:45 +00:00
|
|
|
let data: &CreateCommentLike = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2021-03-18 20:25:21 +00:00
|
|
|
let mut recipient_ids = Vec::<LocalUserId>::new();
|
2020-02-01 01:02:20 +00:00
|
|
|
|
2019-12-11 20:21:47 +00:00
|
|
|
// Don't do a downvote if site has downvotes disabled
|
2020-12-20 01:10:47 +00:00
|
|
|
check_downvotes_enabled(data.score, context.pool()).await?;
|
2019-12-11 20:21:47 +00:00
|
|
|
|
2020-07-21 01:37:44 +00:00
|
|
|
let comment_id = data.comment_id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let orig_comment = blocking(context.pool(), move |conn| {
|
|
|
|
CommentView::read(&conn, comment_id, None)
|
|
|
|
})
|
|
|
|
.await??;
|
2020-07-21 01:37:44 +00:00
|
|
|
|
2021-03-11 04:43:11 +00:00
|
|
|
check_community_ban(
|
|
|
|
local_user_view.person.id,
|
|
|
|
orig_comment.community.id,
|
|
|
|
context.pool(),
|
|
|
|
)
|
|
|
|
.await?;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2020-12-20 01:10:47 +00:00
|
|
|
// Add parent user to recipients
|
2021-03-18 20:25:21 +00:00
|
|
|
let recipient_id = orig_comment.get_recipient_id();
|
|
|
|
if let Ok(local_recipient) = blocking(context.pool(), move |conn| {
|
|
|
|
LocalUserView::read_person(conn, recipient_id)
|
|
|
|
})
|
|
|
|
.await?
|
|
|
|
{
|
|
|
|
recipient_ids.push(local_recipient.local_user.id);
|
|
|
|
}
|
2020-02-01 01:02:20 +00:00
|
|
|
|
2019-05-05 05:20:38 +00:00
|
|
|
let like_form = CommentLikeForm {
|
|
|
|
comment_id: data.comment_id,
|
2020-12-20 01:10:47 +00:00
|
|
|
post_id: orig_comment.post.id,
|
2021-03-10 22:33:55 +00:00
|
|
|
person_id: local_user_view.person.id,
|
2019-09-07 15:35:05 +00:00
|
|
|
score: data.score,
|
2019-05-05 05:20:38 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
// Remove any likes first
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2020-08-18 13:43:50 +00:00
|
|
|
blocking(context.pool(), move |conn| {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentLike::remove(conn, person_id, comment_id)
|
2020-08-12 14:43:45 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
|
|
|
// Only add the like if the score isnt 0
|
2020-12-20 01:10:47 +00:00
|
|
|
let comment = orig_comment.comment;
|
2019-12-19 21:59:13 +00:00
|
|
|
let do_add = like_form.score != 0 && (like_form.score == 1 || like_form.score == -1);
|
2019-05-15 16:46:39 +00:00
|
|
|
if do_add {
|
2020-07-01 12:54:29 +00:00
|
|
|
let like_form2 = like_form.clone();
|
|
|
|
let like = move |conn: &'_ _| CommentLike::like(conn, &like_form2);
|
2020-08-18 13:43:50 +00:00
|
|
|
if blocking(context.pool(), like).await?.is_err() {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("couldnt_like_comment").into());
|
2020-07-01 12:54:29 +00:00
|
|
|
}
|
2020-04-28 04:16:02 +00:00
|
|
|
|
|
|
|
if like_form.score == 1 {
|
2021-03-10 22:33:55 +00:00
|
|
|
comment.send_like(&local_user_view.person, context).await?;
|
2020-04-28 04:16:02 +00:00
|
|
|
} else if like_form.score == -1 {
|
2021-03-11 04:43:11 +00:00
|
|
|
comment
|
|
|
|
.send_dislike(&local_user_view.person, context)
|
|
|
|
.await?;
|
2020-04-28 04:16:02 +00:00
|
|
|
}
|
|
|
|
} else {
|
2021-03-11 04:43:11 +00:00
|
|
|
comment
|
|
|
|
.send_undo_like(&local_user_view.person, context)
|
|
|
|
.await?;
|
2019-05-05 05:20:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Have to refetch the comment to get the current state
|
2020-07-01 12:54:29 +00:00
|
|
|
let comment_id = data.comment_id;
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2020-08-18 13:43:50 +00:00
|
|
|
let liked_comment = blocking(context.pool(), move |conn| {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentView::read(conn, comment_id, Some(person_id))
|
2020-07-01 12:54:29 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2019-05-05 05:20:38 +00:00
|
|
|
|
2021-01-07 06:17:42 +00:00
|
|
|
let res = CommentResponse {
|
2020-12-15 19:39:18 +00:00
|
|
|
comment_view: liked_comment,
|
2020-02-01 01:02:20 +00:00
|
|
|
recipient_ids,
|
2020-07-21 14:56:41 +00:00
|
|
|
form_id: None,
|
2020-04-19 22:08:25 +00:00
|
|
|
};
|
|
|
|
|
2020-08-24 11:58:24 +00:00
|
|
|
context.chat_server().do_send(SendComment {
|
|
|
|
op: UserOperation::CreateCommentLike,
|
|
|
|
comment: res.clone(),
|
|
|
|
websocket_id,
|
|
|
|
});
|
2020-04-19 22:08:25 +00:00
|
|
|
|
|
|
|
Ok(res)
|
2019-05-05 05:20:38 +00:00
|
|
|
}
|
|
|
|
}
|
2020-02-08 04:05:15 +00:00
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
#[async_trait::async_trait(?Send)]
|
2020-08-12 11:31:45 +00:00
|
|
|
impl Perform for GetComments {
|
2020-04-20 03:59:07 +00:00
|
|
|
type Response = GetCommentsResponse;
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
async fn perform(
|
2020-04-19 22:08:25 +00:00
|
|
|
&self,
|
2020-08-18 13:43:50 +00:00
|
|
|
context: &Data<LemmyContext>,
|
2020-09-15 19:26:47 +00:00
|
|
|
_websocket_id: Option<ConnectionId>,
|
2020-07-01 12:54:29 +00:00
|
|
|
) -> Result<GetCommentsResponse, LemmyError> {
|
2020-08-12 11:31:45 +00:00
|
|
|
let data: &GetComments = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt_opt(&data.auth, context.pool()).await?;
|
|
|
|
let person_id = local_user_view.map(|u| u.person.id);
|
2020-02-08 04:05:15 +00:00
|
|
|
|
|
|
|
let type_ = ListingType::from_str(&data.type_)?;
|
|
|
|
let sort = SortType::from_str(&data.sort)?;
|
|
|
|
|
2020-07-01 12:54:29 +00:00
|
|
|
let community_id = data.community_id;
|
2020-09-15 19:26:47 +00:00
|
|
|
let community_name = data.community_name.to_owned();
|
2021-03-22 14:39:54 +00:00
|
|
|
let saved_only = data.saved_only;
|
2020-07-01 12:54:29 +00:00
|
|
|
let page = data.page;
|
|
|
|
let limit = data.limit;
|
2020-08-18 13:43:50 +00:00
|
|
|
let comments = blocking(context.pool(), move |conn| {
|
2020-12-16 18:59:43 +00:00
|
|
|
CommentQueryBuilder::create(conn)
|
2020-07-01 12:54:29 +00:00
|
|
|
.listing_type(type_)
|
|
|
|
.sort(&sort)
|
2021-03-22 14:39:54 +00:00
|
|
|
.saved_only(saved_only)
|
2020-12-16 18:59:43 +00:00
|
|
|
.community_id(community_id)
|
|
|
|
.community_name(community_name)
|
2021-03-10 22:33:55 +00:00
|
|
|
.my_person_id(person_id)
|
2020-07-01 12:54:29 +00:00
|
|
|
.page(page)
|
|
|
|
.limit(limit)
|
|
|
|
.list()
|
|
|
|
})
|
|
|
|
.await?;
|
|
|
|
let comments = match comments {
|
2020-02-08 04:05:15 +00:00
|
|
|
Ok(comments) => comments,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_) => return Err(ApiError::err("couldnt_get_comments").into()),
|
2020-02-08 04:05:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Ok(GetCommentsResponse { comments })
|
|
|
|
}
|
|
|
|
}
|
2020-10-25 02:59:13 +00:00
|
|
|
|
2020-11-04 02:15:11 +00:00
|
|
|
/// Creates a comment report and notifies the moderators of the community
|
2020-10-25 02:59:13 +00:00
|
|
|
#[async_trait::async_trait(?Send)]
|
|
|
|
impl Perform for CreateCommentReport {
|
|
|
|
type Response = CreateCommentReportResponse;
|
|
|
|
|
|
|
|
async fn perform(
|
|
|
|
&self,
|
|
|
|
context: &Data<LemmyContext>,
|
2020-10-27 00:25:18 +00:00
|
|
|
websocket_id: Option<ConnectionId>,
|
2020-10-25 02:59:13 +00:00
|
|
|
) -> Result<CreateCommentReportResponse, LemmyError> {
|
|
|
|
let data: &CreateCommentReport = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2020-10-25 02:59:13 +00:00
|
|
|
|
|
|
|
// check size of report and check for whitespace
|
|
|
|
let reason = data.reason.trim();
|
|
|
|
if reason.is_empty() {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("report_reason_required").into());
|
2020-10-25 02:59:13 +00:00
|
|
|
}
|
2021-02-10 15:36:22 +00:00
|
|
|
if reason.chars().count() > 1000 {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("report_too_long").into());
|
2020-10-25 02:59:13 +00:00
|
|
|
}
|
2020-11-04 02:15:11 +00:00
|
|
|
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2020-10-25 02:59:13 +00:00
|
|
|
let comment_id = data.comment_id;
|
2020-12-15 19:39:18 +00:00
|
|
|
let comment_view = blocking(context.pool(), move |conn| {
|
2020-10-25 02:59:13 +00:00
|
|
|
CommentView::read(&conn, comment_id, None)
|
2020-11-04 02:15:11 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2020-10-25 02:59:13 +00:00
|
|
|
|
2021-03-10 22:33:55 +00:00
|
|
|
check_community_ban(person_id, comment_view.community.id, context.pool()).await?;
|
2020-10-25 02:59:13 +00:00
|
|
|
|
|
|
|
let report_form = CommentReportForm {
|
2021-03-10 22:33:55 +00:00
|
|
|
creator_id: person_id,
|
2020-10-25 02:59:13 +00:00
|
|
|
comment_id,
|
2020-12-15 19:39:18 +00:00
|
|
|
original_comment_text: comment_view.comment.content,
|
2020-10-25 02:59:13 +00:00
|
|
|
reason: data.reason.to_owned(),
|
|
|
|
};
|
|
|
|
|
2020-10-27 00:25:18 +00:00
|
|
|
let report = match blocking(context.pool(), move |conn| {
|
2020-10-25 02:59:13 +00:00
|
|
|
CommentReport::report(conn, &report_form)
|
2020-11-04 02:15:11 +00:00
|
|
|
})
|
|
|
|
.await?
|
|
|
|
{
|
2020-10-25 02:59:13 +00:00
|
|
|
Ok(report) => report,
|
2021-02-22 18:04:32 +00:00
|
|
|
Err(_e) => return Err(ApiError::err("couldnt_create_report").into()),
|
2020-10-25 02:59:13 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
let res = CreateCommentReportResponse { success: true };
|
|
|
|
|
2020-10-27 00:25:18 +00:00
|
|
|
context.chat_server().do_send(SendUserRoomMessage {
|
|
|
|
op: UserOperation::CreateCommentReport,
|
|
|
|
response: res.clone(),
|
2021-03-18 20:25:21 +00:00
|
|
|
local_recipient_id: local_user_view.local_user.id,
|
2020-10-27 00:25:18 +00:00
|
|
|
websocket_id,
|
|
|
|
});
|
|
|
|
|
|
|
|
context.chat_server().do_send(SendModRoomMessage {
|
|
|
|
op: UserOperation::CreateCommentReport,
|
|
|
|
response: report,
|
2020-12-15 19:39:18 +00:00
|
|
|
community_id: comment_view.community.id,
|
2020-10-27 00:25:18 +00:00
|
|
|
websocket_id,
|
|
|
|
});
|
2020-10-25 02:59:13 +00:00
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-04 02:15:11 +00:00
|
|
|
/// Resolves or unresolves a comment report and notifies the moderators of the community
|
2020-10-25 02:59:13 +00:00
|
|
|
#[async_trait::async_trait(?Send)]
|
|
|
|
impl Perform for ResolveCommentReport {
|
|
|
|
type Response = ResolveCommentReportResponse;
|
|
|
|
|
|
|
|
async fn perform(
|
|
|
|
&self,
|
|
|
|
context: &Data<LemmyContext>,
|
2020-10-27 00:25:18 +00:00
|
|
|
websocket_id: Option<ConnectionId>,
|
2020-10-25 02:59:13 +00:00
|
|
|
) -> Result<ResolveCommentReportResponse, LemmyError> {
|
|
|
|
let data: &ResolveCommentReport = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2020-10-25 02:59:13 +00:00
|
|
|
|
|
|
|
let report_id = data.report_id;
|
|
|
|
let report = blocking(context.pool(), move |conn| {
|
|
|
|
CommentReportView::read(&conn, report_id)
|
2020-11-04 02:15:11 +00:00
|
|
|
})
|
|
|
|
.await??;
|
2020-10-25 02:59:13 +00:00
|
|
|
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
|
|
|
is_mod_or_admin(context.pool(), person_id, report.community.id).await?;
|
2020-10-25 02:59:13 +00:00
|
|
|
|
|
|
|
let resolved = data.resolved;
|
|
|
|
let resolve_fun = move |conn: &'_ _| {
|
|
|
|
if resolved {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentReport::resolve(conn, report_id, person_id)
|
2020-10-25 02:59:13 +00:00
|
|
|
} else {
|
2021-03-10 22:33:55 +00:00
|
|
|
CommentReport::unresolve(conn, report_id, person_id)
|
2020-10-25 02:59:13 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-11-04 02:15:11 +00:00
|
|
|
if blocking(context.pool(), resolve_fun).await?.is_err() {
|
2021-02-22 18:04:32 +00:00
|
|
|
return Err(ApiError::err("couldnt_resolve_report").into());
|
2020-10-25 02:59:13 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
let report_id = data.report_id;
|
|
|
|
let res = ResolveCommentReportResponse {
|
|
|
|
report_id,
|
|
|
|
resolved,
|
|
|
|
};
|
|
|
|
|
2020-10-27 00:25:18 +00:00
|
|
|
context.chat_server().do_send(SendModRoomMessage {
|
|
|
|
op: UserOperation::ResolveCommentReport,
|
|
|
|
response: res.clone(),
|
2020-12-17 03:03:03 +00:00
|
|
|
community_id: report.community.id,
|
2020-10-27 00:25:18 +00:00
|
|
|
websocket_id,
|
|
|
|
});
|
2020-10-25 02:59:13 +00:00
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-04 02:15:11 +00:00
|
|
|
/// Lists comment reports for a community if an id is supplied
|
|
|
|
/// or returns all comment reports for communities a user moderates
|
2020-10-25 02:59:13 +00:00
|
|
|
#[async_trait::async_trait(?Send)]
|
|
|
|
impl Perform for ListCommentReports {
|
|
|
|
type Response = ListCommentReportsResponse;
|
|
|
|
|
|
|
|
async fn perform(
|
|
|
|
&self,
|
|
|
|
context: &Data<LemmyContext>,
|
|
|
|
websocket_id: Option<ConnectionId>,
|
|
|
|
) -> Result<ListCommentReportsResponse, LemmyError> {
|
|
|
|
let data: &ListCommentReports = &self;
|
2021-03-10 22:33:55 +00:00
|
|
|
let local_user_view = get_local_user_view_from_jwt(&data.auth, context.pool()).await?;
|
2020-10-25 02:59:13 +00:00
|
|
|
|
2021-03-10 22:33:55 +00:00
|
|
|
let person_id = local_user_view.person.id;
|
2020-10-25 02:59:13 +00:00
|
|
|
let community_id = data.community;
|
2020-11-04 02:15:11 +00:00
|
|
|
let community_ids =
|
2021-03-10 22:33:55 +00:00
|
|
|
collect_moderated_communities(person_id, community_id, context.pool()).await?;
|
2020-10-25 02:59:13 +00:00
|
|
|
|
|
|
|
let page = data.page;
|
|
|
|
let limit = data.limit;
|
|
|
|
let comments = blocking(context.pool(), move |conn| {
|
|
|
|
CommentReportQueryBuilder::create(conn)
|
2020-11-04 02:15:11 +00:00
|
|
|
.community_ids(community_ids)
|
|
|
|
.page(page)
|
|
|
|
.limit(limit)
|
|
|
|
.list()
|
2020-10-25 02:59:13 +00:00
|
|
|
})
|
2020-11-04 02:15:11 +00:00
|
|
|
.await??;
|
2020-10-25 02:59:13 +00:00
|
|
|
|
|
|
|
let res = ListCommentReportsResponse { comments };
|
|
|
|
|
|
|
|
context.chat_server().do_send(SendUserRoomMessage {
|
|
|
|
op: UserOperation::ListCommentReports,
|
|
|
|
response: res.clone(),
|
2021-03-18 20:25:21 +00:00
|
|
|
local_recipient_id: local_user_view.local_user.id,
|
2020-10-25 02:59:13 +00:00
|
|
|
websocket_id,
|
|
|
|
});
|
|
|
|
|
|
|
|
Ok(res)
|
|
|
|
}
|
2020-11-04 02:15:11 +00:00
|
|
|
}
|