# frozen_string_literal: true class ReviewableFlaggedPost < Reviewable scope :pending_and_default_visible, -> { pending.default_visible } # Penalties are handled by the modal after the action is performed def self.action_aliases { agree_and_keep_hidden: :agree_and_keep, agree_and_silence: :agree_and_keep, agree_and_suspend: :agree_and_keep, agree_and_edit: :agree_and_keep, disagree_and_restore: :disagree, ignore_and_do_nothing: :ignore, } end def self.counts_for(posts) result = {} counts = DB.query(<<~SQL, pending: statuses[:pending]) SELECT r.target_id AS post_id, rs.reviewable_score_type, count(*) as total FROM reviewables AS r INNER JOIN reviewable_scores AS rs ON rs.reviewable_id = r.id WHERE r.type = 'ReviewableFlaggedPost' AND r.status = :pending GROUP BY r.target_id, rs.reviewable_score_type SQL counts.each do |c| result[c.post_id] ||= {} result[c.post_id][c.reviewable_score_type] = c.total end result end def post @post ||= (target || Post.with_deleted.find_by(id: target_id)) end def build_actions(actions, guardian, args) return unless pending? return if post.blank? agree_bundle = actions.add_bundle("#{id}-agree", icon: "thumbs-up", label: "reviewables.actions.agree.title") if potential_spam? && guardian.can_delete_user?(target_created_by) delete_user_actions(actions, agree_bundle) end if !post.user_deleted? && !post.hidden? build_action(actions, :agree_and_hide, icon: "far-eye-slash", bundle: agree_bundle) end if post.hidden? build_action(actions, :agree_and_keep_hidden, icon: "thumbs-up", bundle: agree_bundle) else build_action(actions, :agree_and_keep, icon: "thumbs-up", bundle: agree_bundle) build_action( actions, :agree_and_edit, icon: "pencil-alt", bundle: agree_bundle, client_action: "edit", ) end if guardian.can_delete_post_or_topic?(post) build_action(actions, :delete_and_agree, icon: "far-trash-alt", bundle: agree_bundle) if post.reply_count > 0 build_action( actions, :delete_and_agree_replies, icon: "far-trash-alt", bundle: agree_bundle, confirm: true, ) end end if guardian.can_suspend?(target_created_by) build_action( actions, :agree_and_suspend, icon: "ban", bundle: agree_bundle, client_action: "suspend", ) build_action( actions, :agree_and_silence, icon: "microphone-slash", bundle: agree_bundle, client_action: "silence", ) end if post.user_deleted? build_action(actions, :agree_and_restore, icon: "far-eye", bundle: agree_bundle) end if post.hidden? build_action(actions, :disagree_and_restore, icon: "thumbs-down") else build_action(actions, :disagree, icon: "thumbs-down") end ignore = actions.add_bundle( "#{id}-ignore", icon: "thumbs-up", label: "reviewables.actions.ignore.title", ) if !post.hidden? || guardian.user.is_system_user? build_action(actions, :ignore_and_do_nothing, icon: "external-link-alt", bundle: ignore) end if guardian.can_delete_post_or_topic?(post) build_action(actions, :delete_and_ignore, icon: "far-trash-alt", bundle: ignore) if post.reply_count > 0 build_action( actions, :delete_and_ignore_replies, icon: "far-trash-alt", confirm: true, bundle: ignore, ) end end end def perform_ignore(performed_by, args) perform_ignore_and_do_nothing(performed_by, args) end def post_action_type_view @post_action_type_view ||= PostActionTypeView.new end def perform_ignore_and_do_nothing(performed_by, args) actions = PostAction .active .where(post_id: target_id) .where(post_action_type_id: post_action_type_view.notify_flag_type_ids) actions.each do |action| action.deferred_at = Time.zone.now action.deferred_by_id = performed_by.id # so callback is called action.save unless args[:expired] action.add_moderator_post_if_needed(performed_by, :ignored, args[:post_was_deleted]) end end if actions.first.present? unassign_topic performed_by, post DiscourseEvent.trigger(:flag_reviewed, post) DiscourseEvent.trigger(:flag_deferred, actions.first) end create_result(:success, :ignored) do |result| result.update_flag_stats = { status: :ignored, user_ids: actions.map(&:user_id) } end end def perform_agree_and_keep(performed_by, args) agree(performed_by, args) end def perform_delete_user(performed_by, args) delete_options = delete_opts UserDestroyer.new(performed_by).destroy(post.user, delete_options) agree(performed_by, args) end def perform_delete_user_block(performed_by, args) delete_options = delete_opts delete_options.merge!(block_email: true, block_ip: true) if Rails.env.production? UserDestroyer.new(performed_by).destroy(post.user, delete_options) agree(performed_by, args) end def perform_agree_and_hide(performed_by, args) agree(performed_by, args) { |pa| post.hide!(pa.post_action_type_id) } end def perform_agree_and_restore(performed_by, args) agree(performed_by, args) { PostDestroyer.new(performed_by, post).recover } end def perform_disagree(performed_by, args) # -1 is the automatic system clear action_type_ids = if performed_by.id == Discourse::SYSTEM_USER_ID post_action_type_view.auto_action_flag_types.values else post_action_type_view.notify_flag_type_ids end actions = PostAction.active.where(post_id: target_id).where(post_action_type_id: action_type_ids) actions.each do |action| action.disagreed_at = Time.zone.now action.disagreed_by_id = performed_by.id # so callback is called action.save action.add_moderator_post_if_needed(performed_by, :disagreed) end # reset all cached counters cached = {} action_type_ids.each do |atid| column = "#{post_action_type_view.types[atid]}_count" cached[column] = 0 if ActiveRecord::Base.connection.column_exists?(:posts, column) end Post.with_deleted.where(id: target_id).update_all(cached) if actions.first.present? unassign_topic performed_by, post DiscourseEvent.trigger(:flag_reviewed, post) DiscourseEvent.trigger(:flag_disagreed, actions.first) end # Undo hide/silence if applicable if post&.hidden? notify_poster(performed_by) post.unhide! UserSilencer.unsilence(post.user) if UserSilencer.was_silenced_for?(post) end create_result(:success, :rejected) do |result| result.update_flag_stats = { status: :disagreed, user_ids: actions.map(&:user_id) } end end def perform_delete_and_ignore(performed_by, args) result = perform_ignore_and_do_nothing(performed_by, args) destroyer(performed_by, post).destroy result end def perform_delete_and_ignore_replies(performed_by, args) result = perform_ignore_and_do_nothing(performed_by, args) PostDestroyer.delete_with_replies(performed_by, post, self) result end def perform_delete_and_agree(performed_by, args) result = agree(performed_by, args) destroyer(performed_by, post).destroy result end def perform_delete_and_agree_replies(performed_by, args) result = agree(performed_by, args) PostDestroyer.delete_with_replies(performed_by, post, self) result end protected def agree(performed_by, args) actions = PostAction .active .where(post_id: target_id) .where(post_action_type_id: post_action_type_view.notify_flag_types.values) trigger_spam = false actions.each do |action| ActiveRecord::Base.transaction do action.agreed_at = Time.zone.now action.agreed_by_id = performed_by.id # so callback is called action.save DB.after_commit do action.add_moderator_post_if_needed(performed_by, :agreed, args[:post_was_deleted]) trigger_spam = true if action.post_action_type_id == post_action_type_view.types[:spam] end end end DiscourseEvent.trigger(:confirmed_spam_post, post) if trigger_spam if actions.first.present? unassign_topic performed_by, post DiscourseEvent.trigger(:flag_reviewed, post) DiscourseEvent.trigger(:flag_agreed, actions.first) yield(actions.first) if block_given? end create_result(:success, :approved) do |result| result.update_flag_stats = { status: :agreed, user_ids: actions.map(&:user_id) } result.recalculate_score = true end end def build_action( actions, id, icon:, button_class: nil, bundle: nil, client_action: nil, confirm: false ) actions.add(id, bundle: bundle) do |action| prefix = "reviewables.actions.#{id}" action.icon = icon action.button_class = button_class action.label = "#{prefix}.title" action.description = "#{prefix}.description" action.client_action = client_action action.confirm_message = "#{prefix}.confirm" if confirm end end def unassign_topic(performed_by, post) topic = post.topic return unless topic && performed_by && SiteSetting.reviewable_claiming != "disabled" ReviewableClaimedTopic.where(topic_id: topic.id).delete_all topic.reviewables.find_each { |reviewable| reviewable.log_history(:unclaimed, performed_by) } user_ids = User.staff.pluck(:id) if SiteSetting.enable_category_group_moderation? && group_id = topic.category&.reviewable_by_group_id.presence user_ids.concat(GroupUser.where(group_id: group_id).pluck(:user_id)) user_ids.uniq! end data = { topic_id: topic.id } MessageBus.publish("/reviewable_claimed", data, user_ids: user_ids) end private def delete_opts { delete_posts: true, prepare_for_destroy: true, block_urls: true, delete_as_spammer: true, context: "review", } end def destroyer(performed_by, post) PostDestroyer.new(performed_by, post, reviewable: self) end def notify_poster(performed_by) return unless performed_by.human? && performed_by.staff? Jobs.enqueue( :send_system_message, user_id: post.user_id, message_type: "flags_disagreed", message_options: { flagged_post_raw_content: post.raw, url: post.url, }, ) end end # == Schema Information # # Table name: reviewables # # id :bigint not null, primary key # type :string not null # status :integer default("pending"), not null # created_by_id :integer not null # reviewable_by_moderator :boolean default(FALSE), not null # reviewable_by_group_id :integer # category_id :integer # topic_id :integer # score :float default(0.0), not null # potential_spam :boolean default(FALSE), not null # target_id :integer # target_type :string # target_created_by_id :integer # payload :json # version :integer default(0), not null # latest_score :datetime # created_at :datetime not null # updated_at :datetime not null # force_review :boolean default(FALSE), not null # reject_reason :text # # Indexes # # idx_reviewables_score_desc_created_at_desc (score,created_at) # index_reviewables_on_reviewable_by_group_id (reviewable_by_group_id) # index_reviewables_on_status_and_created_at (status,created_at) # index_reviewables_on_status_and_score (status,score) # index_reviewables_on_status_and_type (status,type) # index_reviewables_on_target_id_where_post_type_eq_post (target_id) WHERE ((target_type)::text = 'Post'::text) # index_reviewables_on_topic_id_and_status_and_created_by_id (topic_id,status,created_by_id) # index_reviewables_on_type_and_target_id (type,target_id) UNIQUE #