2013-09-11 09:21:16 +08:00
|
|
|
# UserHistory stores information about actions that users have taken,
|
|
|
|
# like deleting users, changing site settings, dimissing notifications, etc.
|
|
|
|
# Use other classes, like StaffActionLogger, to log records to this table.
|
|
|
|
class UserHistory < ActiveRecord::Base
|
|
|
|
belongs_to :acting_user, class_name: 'User'
|
|
|
|
belongs_to :target_user, class_name: 'User'
|
2013-04-12 04:04:20 +08:00
|
|
|
|
2014-10-01 23:40:13 +08:00
|
|
|
belongs_to :post
|
|
|
|
belongs_to :topic
|
2015-09-17 15:51:32 +08:00
|
|
|
belongs_to :category
|
2014-10-01 23:40:13 +08:00
|
|
|
|
2013-04-12 04:04:20 +08:00
|
|
|
validates_presence_of :action
|
|
|
|
|
2013-09-11 09:21:16 +08:00
|
|
|
scope :only_staff_actions, ->{ where("action IN (?)", UserHistory.staff_action_ids) }
|
|
|
|
|
2014-03-01 05:30:45 +08:00
|
|
|
before_save :set_admin_only
|
|
|
|
|
2013-04-12 04:04:20 +08:00
|
|
|
def self.actions
|
2016-01-08 18:53:52 +08:00
|
|
|
@actions ||= Enum.new(delete_user: 1,
|
|
|
|
change_trust_level: 2,
|
|
|
|
change_site_setting: 3,
|
|
|
|
change_site_customization: 4,
|
|
|
|
delete_site_customization: 5,
|
|
|
|
checked_for_custom_avatar: 6, # not used anymore
|
|
|
|
notified_about_avatar: 7,
|
|
|
|
notified_about_sequential_replies: 8,
|
|
|
|
notified_about_dominating_topic: 9,
|
|
|
|
suspend_user: 10,
|
|
|
|
unsuspend_user: 11,
|
|
|
|
facebook_no_email: 12,
|
|
|
|
grant_badge: 13,
|
|
|
|
revoke_badge: 14,
|
|
|
|
auto_trust_level_change: 15,
|
|
|
|
check_email: 16,
|
|
|
|
delete_post: 17,
|
|
|
|
delete_topic: 18,
|
|
|
|
impersonate: 19,
|
|
|
|
roll_up: 20,
|
|
|
|
change_username: 21,
|
|
|
|
custom: 22,
|
|
|
|
custom_staff: 23,
|
|
|
|
anonymize_user: 24,
|
|
|
|
reviewed_post: 25,
|
|
|
|
change_category_settings: 26,
|
|
|
|
delete_category: 27,
|
|
|
|
create_category: 28,
|
2016-01-15 04:05:11 +08:00
|
|
|
change_site_text: 29,
|
|
|
|
block_user: 30,
|
2016-01-27 17:38:16 +08:00
|
|
|
unblock_user: 31,
|
|
|
|
grant_admin: 32,
|
|
|
|
revoke_admin: 33,
|
|
|
|
grant_moderation: 34,
|
2016-02-28 01:38:24 +08:00
|
|
|
revoke_moderation: 35,
|
2016-03-16 04:06:50 +08:00
|
|
|
backup_operation: 36,
|
2016-05-03 05:15:32 +08:00
|
|
|
rate_limited_like: 37, # not used anymore
|
|
|
|
revoke_email: 38
|
2016-03-18 02:41:00 +08:00
|
|
|
)
|
2013-04-12 04:04:20 +08:00
|
|
|
end
|
2013-08-09 22:06:02 +08:00
|
|
|
|
2013-09-11 09:21:16 +08:00
|
|
|
# Staff actions is a subset of all actions, used to audit actions taken by staff users.
|
|
|
|
def self.staff_actions
|
|
|
|
@staff_actions ||= [:delete_user,
|
|
|
|
:change_trust_level,
|
|
|
|
:change_site_setting,
|
|
|
|
:change_site_customization,
|
2013-11-01 22:47:03 +08:00
|
|
|
:delete_site_customization,
|
2015-12-18 21:31:04 +08:00
|
|
|
:change_site_text,
|
2013-11-08 02:53:32 +08:00
|
|
|
:suspend_user,
|
2014-03-20 03:30:12 +08:00
|
|
|
:unsuspend_user,
|
|
|
|
:grant_badge,
|
2014-09-30 04:31:05 +08:00
|
|
|
:revoke_badge,
|
2014-10-01 23:40:13 +08:00
|
|
|
:check_email,
|
|
|
|
:delete_post,
|
2014-11-06 17:58:47 +08:00
|
|
|
:delete_topic,
|
2014-11-25 02:48:54 +08:00
|
|
|
:impersonate,
|
2015-01-17 06:30:46 +08:00
|
|
|
:roll_up,
|
2015-02-06 03:34:57 +08:00
|
|
|
:change_username,
|
2015-03-07 05:44:54 +08:00
|
|
|
:custom_staff,
|
2015-04-16 03:29:37 +08:00
|
|
|
:anonymize_user,
|
2015-09-17 15:51:32 +08:00
|
|
|
:reviewed_post,
|
|
|
|
:change_category_settings,
|
|
|
|
:delete_category,
|
2016-01-15 04:05:11 +08:00
|
|
|
:create_category,
|
|
|
|
:block_user,
|
2016-01-27 17:38:16 +08:00
|
|
|
:unblock_user,
|
|
|
|
:grant_admin,
|
|
|
|
:revoke_admin,
|
|
|
|
:grant_moderation,
|
2016-02-28 01:38:24 +08:00
|
|
|
:revoke_moderation,
|
2016-06-07 04:58:09 +08:00
|
|
|
:backup_operation,
|
|
|
|
:revoke_email]
|
2013-09-11 09:21:16 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.staff_action_ids
|
|
|
|
@staff_action_ids ||= staff_actions.map { |a| actions[a] }
|
|
|
|
end
|
|
|
|
|
2014-03-01 05:30:45 +08:00
|
|
|
def self.admin_only_action_ids
|
|
|
|
@admin_only_action_ids ||= [actions[:change_site_setting]]
|
|
|
|
end
|
|
|
|
|
2013-08-09 22:06:02 +08:00
|
|
|
def self.with_filters(filters)
|
|
|
|
query = self
|
2015-02-06 03:34:57 +08:00
|
|
|
query = query.where(action: filters[:action_id]) if filters[:action_id].present?
|
|
|
|
query = query.where(custom_type: filters[:custom_type]) if filters[:custom_type].present?
|
|
|
|
|
2013-09-11 09:21:16 +08:00
|
|
|
[:acting_user, :target_user].each do |key|
|
2013-08-10 04:58:57 +08:00
|
|
|
if filters[key] and obj_id = User.where(username_lower: filters[key].downcase).pluck(:id)
|
2014-08-15 02:20:52 +08:00
|
|
|
query = query.where("#{key}_id = ?", obj_id)
|
2013-08-10 04:58:57 +08:00
|
|
|
end
|
|
|
|
end
|
2013-08-21 01:50:51 +08:00
|
|
|
query = query.where("subject = ?", filters[:subject]) if filters[:subject]
|
2013-08-09 22:06:02 +08:00
|
|
|
query
|
|
|
|
end
|
2013-08-21 22:49:35 +08:00
|
|
|
|
2013-11-01 22:47:03 +08:00
|
|
|
def self.for(user, action_type)
|
|
|
|
self.where(target_user_id: user.id, action: UserHistory.actions[action_type])
|
|
|
|
end
|
|
|
|
|
2013-09-18 02:38:39 +08:00
|
|
|
def self.exists_for_user?(user, action_type, opts=nil)
|
|
|
|
opts = opts || {}
|
|
|
|
result = self.where(target_user_id: user.id, action: UserHistory.actions[action_type])
|
|
|
|
result = result.where(topic_id: opts[:topic_id]) if opts[:topic_id]
|
|
|
|
result.exists?
|
2013-09-13 05:46:43 +08:00
|
|
|
end
|
|
|
|
|
2015-02-06 03:34:57 +08:00
|
|
|
def self.staff_filters
|
|
|
|
[:action_id, :custom_type, :acting_user, :target_user, :subject]
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.staff_action_records(viewer, opts=nil)
|
|
|
|
opts ||= {}
|
|
|
|
query = self.with_filters(opts.slice(*staff_filters)).only_staff_actions.limit(200).order('id DESC').includes(:acting_user, :target_user)
|
2014-03-01 05:30:45 +08:00
|
|
|
query = query.where(admin_only: false) unless viewer && viewer.admin?
|
|
|
|
query
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
def set_admin_only
|
|
|
|
self.admin_only = UserHistory.admin_only_action_ids.include?(self.action)
|
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2013-08-21 22:49:35 +08:00
|
|
|
def new_value_is_json?
|
2013-09-11 09:21:16 +08:00
|
|
|
[UserHistory.actions[:change_site_customization], UserHistory.actions[:delete_site_customization]].include?(action)
|
2013-08-21 22:49:35 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def previous_value_is_json?
|
|
|
|
new_value_is_json?
|
|
|
|
end
|
2013-04-12 04:04:20 +08:00
|
|
|
end
|
2013-05-24 10:48:32 +08:00
|
|
|
|
|
|
|
# == Schema Information
|
|
|
|
#
|
2013-10-04 11:28:49 +08:00
|
|
|
# Table name: user_histories
|
2013-05-24 10:48:32 +08:00
|
|
|
#
|
|
|
|
# id :integer not null, primary key
|
|
|
|
# action :integer not null
|
2013-10-04 11:28:49 +08:00
|
|
|
# acting_user_id :integer
|
2013-05-24 10:48:32 +08:00
|
|
|
# target_user_id :integer
|
|
|
|
# details :text
|
2014-08-27 13:19:25 +08:00
|
|
|
# created_at :datetime not null
|
|
|
|
# updated_at :datetime not null
|
2016-02-23 07:33:53 +08:00
|
|
|
# context :string
|
|
|
|
# ip_address :string
|
|
|
|
# email :string
|
2013-08-28 08:42:58 +08:00
|
|
|
# subject :text
|
|
|
|
# previous_value :text
|
|
|
|
# new_value :text
|
2013-10-04 11:28:49 +08:00
|
|
|
# topic_id :integer
|
2014-03-20 12:35:51 +08:00
|
|
|
# admin_only :boolean default(FALSE)
|
2014-11-20 11:53:15 +08:00
|
|
|
# post_id :integer
|
2016-02-23 07:33:53 +08:00
|
|
|
# custom_type :string
|
2015-09-17 15:51:32 +08:00
|
|
|
# category_id :integer
|
2013-08-14 04:09:27 +08:00
|
|
|
#
|
|
|
|
# Indexes
|
|
|
|
#
|
2013-10-04 11:28:49 +08:00
|
|
|
# index_user_histories_on_acting_user_id_and_action_and_id (acting_user_id,action,id)
|
2014-05-28 09:49:50 +08:00
|
|
|
# index_user_histories_on_action_and_id (action,id)
|
2015-09-17 15:51:32 +08:00
|
|
|
# index_user_histories_on_category_id (category_id)
|
2014-05-28 09:49:50 +08:00
|
|
|
# index_user_histories_on_subject_and_id (subject,id)
|
|
|
|
# index_user_histories_on_target_user_id_and_id (target_user_id,id)
|
2013-05-24 10:48:32 +08:00
|
|
|
#
|