+use crate::PerformCrud;
use actix_web::web::Data;
use lemmy_api_common::{
+ build_response::{build_comment_response, send_local_notifs},
comment::{CommentResponse, EditComment},
+ context::LemmyContext,
utils::{
- blocking,
check_community_ban,
- check_community_deleted_or_removed,
- check_post_deleted_or_removed,
- get_local_user_view_from_jwt,
- is_mod_or_admin,
+ local_site_to_slur_regex,
+ local_user_view_from_jwt,
+ sanitize_html_opt,
},
};
-use lemmy_apub::protocol::activities::{
- create_or_update::comment::CreateOrUpdateComment,
- CreateOrUpdateType,
-};
use lemmy_db_schema::{
- source::comment::{Comment, CommentForm},
+ source::{
+ actor_language::CommunityLanguage,
+ comment::{Comment, CommentUpdateForm},
+ local_site::LocalSite,
+ },
traits::Crud,
+ utils::naive_now,
};
use lemmy_db_views::structs::CommentView;
use lemmy_utils::{
- error::LemmyError,
- utils::{remove_slurs, scrape_text_for_mentions},
- ConnectionId,
-};
-use lemmy_websocket::{
- send::{send_comment_ws_message, send_local_notifs},
- LemmyContext,
- UserOperationCrud,
+ error::{LemmyError, LemmyErrorExt, LemmyErrorType},
+ utils::{
+ mention::scrape_text_for_mentions,
+ slurs::remove_slurs,
+ validation::is_valid_body_field,
+ },
};
-use crate::PerformCrud;
-
#[async_trait::async_trait(?Send)]
impl PerformCrud for EditComment {
type Response = CommentResponse;
- #[tracing::instrument(skip(context, websocket_id))]
- async fn perform(
- &self,
- context: &Data<LemmyContext>,
- websocket_id: Option<ConnectionId>,
- ) -> Result<CommentResponse, LemmyError> {
+ #[tracing::instrument(skip(context))]
+ async fn perform(&self, context: &Data<LemmyContext>) -> Result<CommentResponse, LemmyError> {
let data: &EditComment = self;
- let local_user_view =
- get_local_user_view_from_jwt(&data.auth, context.pool(), context.secret()).await?;
+ let local_user_view = local_user_view_from_jwt(&data.auth, context).await?;
+ let local_site = LocalSite::read(&mut context.pool()).await?;
let comment_id = data.comment_id;
- let orig_comment = blocking(context.pool(), move |conn| {
- CommentView::read(conn, comment_id, None)
- })
- .await??;
+ let orig_comment = CommentView::read(&mut context.pool(), comment_id, None).await?;
- // TODO is this necessary? It should really only need to check on create
check_community_ban(
local_user_view.person.id,
orig_comment.community.id,
- context.pool(),
+ &mut context.pool(),
)
.await?;
- check_community_deleted_or_removed(orig_comment.community.id, context.pool()).await?;
- check_post_deleted_or_removed(&orig_comment.post)?;
// Verify that only the creator can edit
if local_user_view.person.id != orig_comment.creator.id {
- return Err(LemmyError::from_message("no_comment_edit_allowed"));
+ return Err(LemmyErrorType::NoCommentEditAllowed)?;
}
- if data.distinguished.is_some() {
- // Verify that only a mod or admin can distinguish a comment
- is_mod_or_admin(
- context.pool(),
- local_user_view.person.id,
- orig_comment.community.id,
- )
- .await?;
- }
+ let language_id = self.language_id;
+ CommunityLanguage::is_allowed_community_language(
+ &mut context.pool(),
+ language_id,
+ orig_comment.community.id,
+ )
+ .await?;
// Update the Content
- let content_slurs_removed = data
+ let content = data
.content
.as_ref()
- .map(|c| remove_slurs(&c, &context.settings().slur_regex()));
+ .map(|c| remove_slurs(c, &local_site_to_slur_regex(&local_site)));
+ is_valid_body_field(&content, false)?;
+ let content = sanitize_html_opt(&content);
+
let comment_id = data.comment_id;
- let form = CommentForm {
- creator_id: orig_comment.comment.creator_id,
- post_id: orig_comment.comment.post_id,
- content: content_slurs_removed.unwrap_or(orig_comment.comment.content),
- distinguished: data.distinguished,
- language_id: data.language_id,
- ..Default::default()
- };
- let updated_comment = blocking(context.pool(), move |conn| {
- Comment::update(conn, comment_id, &form)
- })
- .await?
- .map_err(|e| LemmyError::from_error_message(e, "couldnt_update_comment"))?;
+ let form = CommentUpdateForm::builder()
+ .content(content)
+ .language_id(data.language_id)
+ .updated(Some(Some(naive_now())))
+ .build();
+ let updated_comment = Comment::update(&mut context.pool(), comment_id, &form)
+ .await
+ .with_lemmy_type(LemmyErrorType::CouldntUpdateComment)?;
// Do the mentions / recipients
- let updated_comment_content = updated_comment.content.to_owned();
+ let updated_comment_content = updated_comment.content.clone();
let mentions = scrape_text_for_mentions(&updated_comment_content);
let recipient_ids = send_local_notifs(
mentions,
)
.await?;
- // Send the apub update
- CreateOrUpdateComment::send(
- updated_comment.into(),
- &local_user_view.person.into(),
- CreateOrUpdateType::Update,
+ build_comment_response(
context,
- &mut 0,
- )
- .await?;
-
- send_comment_ws_message(
- data.comment_id,
- UserOperationCrud::EditComment,
- websocket_id,
- data.form_id.to_owned(),
- None,
+ updated_comment.id,
+ Some(local_user_view),
+ self.form_id.clone(),
recipient_ids,
- context,
)
.await
}