discourse/app/models/post_action.rb

353 lines
12 KiB
Ruby
Raw Normal View History

2013-02-05 13:16:51 -06:00
require_dependency 'rate_limiter'
require_dependency 'system_message'
require_dependency 'trashable'
2013-02-05 13:16:51 -06:00
class PostAction < ActiveRecord::Base
2013-05-03 19:52:45 -05:00
class AlreadyActed < StandardError; end
2013-02-07 09:45:24 -06:00
2013-02-05 13:16:51 -06:00
include RateLimiter::OnCreateRecord
include Trashable
2013-02-05 13:16:51 -06:00
belongs_to :post
belongs_to :user
belongs_to :post_action_type
belongs_to :related_post, class_name: 'Post'
2014-02-05 16:54:16 -06:00
belongs_to :target_user, class_name: 'User'
2013-02-05 13:16:51 -06:00
rate_limit :post_action_rate_limiter
scope :spam_flags, -> { where(post_action_type_id: PostActionType.types[:spam]) }
2013-08-14 22:44:30 -05:00
after_save :update_counters
after_save :enforce_rules
after_commit :notify_subscribers
2013-08-14 22:44:30 -05:00
2013-02-05 13:16:51 -06:00
def self.update_flagged_posts_count
posts_flagged_count = PostAction.joins(post: :topic)
.where('defer = false or defer IS NULL')
.where('post_actions.post_action_type_id' => PostActionType.notify_flag_type_ids,
'posts.deleted_at' => nil,
2013-05-03 19:52:45 -05:00
'topics.deleted_at' => nil)
.count('DISTINCT posts.id')
$redis.set('posts_flagged_count', posts_flagged_count)
user_ids = User.staff.pluck(:id)
MessageBus.publish('/flagged_counts', { total: posts_flagged_count }, { user_ids: user_ids })
2013-02-05 13:16:51 -06:00
end
def self.flagged_posts_count
$redis.get('posts_flagged_count').to_i
end
def self.counts_for(collection, user)
return {} if collection.blank?
collection_ids = collection.map {|p| p.id}
2013-02-07 09:45:24 -06:00
user_id = user.present? ? user.id : 0
2013-02-05 13:16:51 -06:00
result = PostAction.where(post_id: collection_ids, user_id: user_id)
2013-02-05 13:16:51 -06:00
user_actions = {}
result.each do |r|
user_actions[r.post_id] ||= {}
user_actions[r.post_id][r.post_action_type_id] = r
end
2013-02-07 09:45:24 -06:00
2013-02-05 13:16:51 -06:00
user_actions
2013-02-07 09:45:24 -06:00
end
2013-02-05 13:16:51 -06:00
def self.count_per_day_for_type(sinceDaysAgo = 30, post_action_type)
unscoped.where(post_action_type_id: post_action_type).where('created_at > ?', sinceDaysAgo.days.ago).group('date(created_at)').order('date(created_at)').count
2013-03-17 12:53:00 -05:00
end
2013-02-06 22:15:48 -06:00
def self.clear_flags!(post, moderator_id, action_type_id = nil)
2013-02-05 13:16:51 -06:00
# -1 is the automatic system cleary
2013-02-06 22:15:48 -06:00
actions = if action_type_id
2013-02-07 09:45:24 -06:00
[action_type_id]
2013-02-06 22:15:48 -06:00
else
2013-03-01 06:07:44 -06:00
moderator_id == -1 ? PostActionType.auto_action_flag_types.values : PostActionType.flag_types.values
2013-02-06 22:15:48 -06:00
end
2013-02-05 13:16:51 -06:00
2013-07-09 14:20:18 -05:00
PostAction.where({ post_id: post.id, post_action_type_id: actions }).update_all({ deleted_at: Time.zone.now, deleted_by_id: moderator_id })
2013-03-01 06:07:44 -06:00
f = actions.map{|t| ["#{PostActionType.types[t]}_count", 0]}
Post.where(id: post.id).with_deleted.update_all(Hash[*f.flatten])
2013-02-05 13:16:51 -06:00
update_flagged_posts_count
end
def self.defer_flags!(post, moderator_id)
actions = PostAction.where(
defer: nil,
post_id: post.id,
post_action_type_id:
PostActionType.flag_types.values,
deleted_at: nil
)
actions.each do |a|
a.defer = true
a.defer_by = moderator_id
# so callback is called
a.save
end
update_flagged_posts_count
end
def self.create_message_for_post_action(user, post, post_action_type_id, opts)
post_action_type = PostActionType.types[post_action_type_id]
return unless opts[:message] && [:notify_moderators, :notify_user].include?(post_action_type)
# this is a hack to allow a PM with no reciepients, we should think through
# a cleaner technique, a PM with myself is valid for flagging
target_usernames = post_action_type == :notify_user ? post.user.username : "x"
title = I18n.t("post_action_types.#{post_action_type}.email_title",
title: post.topic.title)
body = I18n.t("post_action_types.#{post_action_type}.email_body",
message: opts[:message],
link: "#{Discourse.base_url}#{post.url}")
subtype = post_action_type == :notify_moderators ? TopicSubtype.notify_moderators : TopicSubtype.notify_user
if target_usernames.present?
PostCreator.new(user,
target_usernames: target_usernames,
archetype: Archetype.private_message,
subtype: subtype,
title: title,
raw: body
).create.id
2013-02-05 13:16:51 -06:00
end
end
def self.act(user, post, post_action_type_id, opts={})
related_post_id = create_message_for_post_action(user,post,post_action_type_id,opts)
targets_topic = if opts[:flag_topic] and post.topic
post.topic.reload
post.topic.posts_count != 1
end
create( post_id: post.id,
user_id: user.id,
post_action_type_id: post_action_type_id,
message: opts[:message],
staff_took_action: opts[:take_action] || false,
2014-02-05 16:54:16 -06:00
related_post_id: related_post_id,
targets_topic: !!targets_topic )
2014-03-23 21:22:03 -05:00
rescue ActiveRecord::RecordNotUnique
# can happen despite being .create
# since already bookmarked
true
end
2013-02-05 13:16:51 -06:00
def self.remove_act(user, post, post_action_type_id)
if action = where(post_id: post.id,
user_id: user.id,
2013-11-05 12:04:47 -06:00
post_action_type_id: post_action_type_id).first
2013-08-14 22:44:30 -05:00
action.remove_act!(user)
2013-02-05 13:16:51 -06:00
end
end
def remove_act!(user)
2013-07-09 14:20:18 -05:00
trash!(user)
# NOTE: save is called to ensure all callbacks are called
# trash will not trigger callbacks, and triggering after_commit
# is not trivial
save
end
2013-02-07 09:45:24 -06:00
def is_bookmark?
2013-03-01 06:07:44 -06:00
post_action_type_id == PostActionType.types[:bookmark]
2013-02-05 13:16:51 -06:00
end
2013-02-07 09:45:24 -06:00
def is_like?
2013-03-01 06:07:44 -06:00
post_action_type_id == PostActionType.types[:like]
2013-02-05 13:16:51 -06:00
end
def is_flag?
2013-03-01 06:07:44 -06:00
PostActionType.flag_types.values.include?(post_action_type_id)
2013-02-07 09:45:24 -06:00
end
2013-02-05 13:16:51 -06:00
def is_private_message?
post_action_type_id == PostActionType.types[:notify_user] ||
post_action_type_id == PostActionType.types[:notify_moderators]
end
2013-02-05 13:16:51 -06:00
# A custom rate limiter for this model
def post_action_rate_limiter
return unless is_flag? || is_bookmark? || is_like?
2013-02-05 13:16:51 -06:00
return @rate_limiter if @rate_limiter.present?
%w(like flag bookmark).each do |type|
if send("is_#{type}?")
2013-02-07 09:45:24 -06:00
@rate_limiter = RateLimiter.new(user, "create_#{type}:#{Date.today.to_s}", SiteSetting.send("max_#{type}s_per_day"), 1.day.to_i)
2013-02-05 13:16:51 -06:00
return @rate_limiter
end
end
end
2013-02-08 15:55:40 -06:00
2013-02-07 09:45:24 -06:00
before_create do
2013-05-03 19:52:45 -05:00
post_action_type_ids = is_flag? ? PostActionType.flag_types.values : post_action_type_id
raise AlreadyActed if PostAction.where(user_id: user_id,
post_id: post_id,
post_action_type_id: post_action_type_ids,
deleted_at: nil)
.exists?
end
# Returns the flag counts for a post, taking into account that some users
# can weigh flags differently.
def self.flag_counts_for(post_id)
flag_counts = exec_sql("SELECT SUM(CASE
WHEN pa.deleted_at IS NULL AND (pa.staff_took_action) THEN :flags_required_to_hide_post
WHEN pa.deleted_at IS NULL AND (NOT pa.staff_took_action) THEN 1
ELSE 0
END) AS new_flags,
SUM(CASE
WHEN pa.deleted_at IS NOT NULL AND (pa.staff_took_action) THEN :flags_required_to_hide_post
WHEN pa.deleted_at IS NOT NULL AND (NOT pa.staff_took_action) THEN 1
ELSE 0
END) AS old_flags
FROM post_actions AS pa
INNER JOIN users AS u ON u.id = pa.user_id
WHERE pa.post_id = :post_id AND
pa.post_action_type_id IN (:post_action_types)",
post_id: post_id,
post_action_types: PostActionType.auto_action_flag_types.values,
flags_required_to_hide_post: SiteSetting.flags_required_to_hide_post).first
[flag_counts['old_flags'].to_i, flag_counts['new_flags'].to_i]
end
2013-08-14 22:44:30 -05:00
def post_action_type_key
PostActionType.types[post_action_type_id]
end
def update_counters
2013-02-05 13:16:51 -06:00
# Update denormalized counts
2013-08-14 22:44:30 -05:00
column = "#{post_action_type_key.to_s}_count"
2013-02-05 13:16:51 -06:00
delta = deleted_at.nil? ? 1 : -1
# We probably want to refactor this method to something cleaner.
2013-08-14 22:44:30 -05:00
case post_action_type_key
when :vote
# Voting also changes the sort_order
Post.where(id: post_id).update_all ["vote_count = vote_count + :delta, sort_order = :max - (vote_count + :delta)",
delta: delta,
max: Topic.max_sort_order]
when :like
# `like_score` is weighted higher for staff accounts
Post.where(id: post_id).update_all ["like_count = like_count + :delta, like_score = like_score + :score_delta",
delta: delta,
score_delta: user.staff? ? delta * SiteSetting.staff_like_weight : delta]
2013-02-05 13:16:51 -06:00
else
Post.where(id: post_id).update_all ["#{column} = #{column} + ?", delta]
2013-02-05 13:16:51 -06:00
end
Topic.where(id: post.topic_id).update_all ["#{column} = #{column} + ?", delta]
2013-02-05 13:16:51 -06:00
if PostActionType.notify_flag_type_ids.include?(post_action_type_id)
2013-02-05 13:16:51 -06:00
PostAction.update_flagged_posts_count
end
2013-08-14 22:44:30 -05:00
end
2013-08-14 22:44:30 -05:00
def enforce_rules
PostAction.auto_hide_if_needed(post, post_action_type_key)
SpamRulesEnforcer.enforce!(post.user) if post_action_type_key == :spam
end
2014-03-23 21:22:03 -05:00
def notify_subscribers
if (is_like? || is_flag?) && post
MessageBus.publish("/topic/#{post.topic_id}",{
id: post.id,
post_number: post.post_number,
type: "acted"
},
group_ids: post.topic.secure_group_ids
)
end
end
def self.auto_hide_if_needed(post, post_action_type)
return if post.hidden
if PostActionType.auto_action_flag_types.include?(post_action_type) &&
SiteSetting.flags_required_to_hide_post > 0
old_flags, new_flags = PostAction.flag_counts_for(post.id)
2013-02-05 13:16:51 -06:00
if new_flags >= SiteSetting.flags_required_to_hide_post
hide_post!(post, guess_hide_reason(old_flags))
2013-02-05 13:16:51 -06:00
end
end
end
def self.hide_post!(post, reason=nil)
return if post.hidden
unless reason
old_flags,_ = PostAction.flag_counts_for(post.id)
reason = guess_hide_reason(old_flags)
end
Post.where(id: post.id).update_all(["hidden = true, hidden_reason_id = COALESCE(hidden_reason_id, ?)", reason])
Topic.where(["id = :topic_id AND NOT EXISTS(SELECT 1 FROM POSTS WHERE topic_id = :topic_id AND NOT hidden)",
topic_id: post.topic_id]).update_all({ visible: false })
# inform user
if post.user
SystemMessage.create(post.user,
:post_hidden,
url: post.url,
edit_delay: SiteSetting.cooldown_minutes_after_hiding_posts)
end
end
def self.guess_hide_reason(old_flags)
old_flags > 0 ?
Post.hidden_reasons[:flag_threshold_reached_again] :
Post.hidden_reasons[:flag_threshold_reached]
2013-02-05 13:16:51 -06:00
end
2013-04-14 22:09:52 -05:00
protected
def self.target_moderators
Group[:moderators].name
end
2013-02-05 13:16:51 -06:00
end
# == Schema Information
#
# Table name: post_actions
#
# id :integer not null, primary key
# post_id :integer not null
# user_id :integer not null
# post_action_type_id :integer not null
# deleted_at :datetime
2014-04-08 10:35:44 -05:00
# created_at :datetime
# updated_at :datetime
2013-07-09 14:20:18 -05:00
# deleted_by_id :integer
# message :text
# related_post_id :integer
2013-06-16 19:48:58 -05:00
# staff_took_action :boolean default(FALSE), not null
# defer :boolean
# defer_by :integer
2014-02-05 16:54:16 -06:00
# targets_topic :boolean default(FALSE)
#
# Indexes
#
# idx_unique_actions (user_id,post_action_type_id,post_id,deleted_at) UNIQUE
# index_post_actions_on_post_id (post_id)
#