mirror of
https://github.com/discourse/discourse.git
synced 2024-11-23 23:54:16 +08:00
0d8ecab362
If a user was granted a trust level, joined a group that granted a trust level and left the group, the trust level was reset. This commit tries to restore the last known trust level before joining the group by looking into staff logs. This commit also migrates old :change_trust_level user history records to use previous_value and new_value fields.
209 lines
6.5 KiB
Ruby
209 lines
6.5 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
class GroupUser < ActiveRecord::Base
|
|
belongs_to :group, counter_cache: "user_count"
|
|
belongs_to :user
|
|
|
|
after_save :update_title
|
|
after_destroy :grant_other_available_title
|
|
|
|
after_save :set_primary_group
|
|
after_destroy :remove_primary_and_flair_group, :recalculate_trust_level
|
|
|
|
before_create :set_notification_level
|
|
after_save :grant_trust_level
|
|
after_save :set_category_notifications
|
|
after_save :set_tag_notifications
|
|
|
|
def self.notification_levels
|
|
NotificationLevels.all
|
|
end
|
|
|
|
def self.ensure_consistency!(last_seen = 1.hour.ago)
|
|
update_first_unread_pm(last_seen)
|
|
end
|
|
|
|
def self.update_first_unread_pm(last_seen, limit: 10_000)
|
|
whisperers_group_ids = SiteSetting.whispers_allowed_group_ids
|
|
|
|
DB.exec(<<~SQL, archetype: Archetype.private_message, last_seen: last_seen, limit: limit, now: 10.minutes.ago, whisperers_group_ids: whisperers_group_ids)
|
|
UPDATE group_users gu
|
|
SET first_unread_pm_at = Y.min_date
|
|
FROM (
|
|
SELECT
|
|
X.group_id,
|
|
X.user_id,
|
|
X.min_date
|
|
FROM (
|
|
SELECT
|
|
gu.group_id,
|
|
gu.user_id,
|
|
COALESCE(Z.min_date, :now) min_date
|
|
FROM group_users gu
|
|
LEFT JOIN (
|
|
SELECT
|
|
gu2.group_id,
|
|
gu2.user_id,
|
|
MIN(t.updated_at) min_date
|
|
FROM group_users gu2
|
|
INNER JOIN topic_allowed_groups tag ON tag.group_id = gu2.group_id
|
|
INNER JOIN topics t ON t.id = tag.topic_id
|
|
INNER JOIN users u ON u.id = gu2.user_id
|
|
LEFT JOIN topic_users tu ON t.id = tu.topic_id AND tu.user_id = gu2.user_id
|
|
WHERE t.deleted_at IS NULL
|
|
AND t.archetype = :archetype
|
|
AND tu.last_read_post_number < CASE
|
|
WHEN u.admin OR u.moderator #{whisperers_group_ids.present? ? 'OR gu2.group_id IN (:whisperers_group_ids)' : ''}
|
|
THEN t.highest_staff_post_number
|
|
ELSE t.highest_post_number
|
|
END
|
|
AND (COALESCE(tu.notification_level, 1) >= 2)
|
|
GROUP BY gu2.user_id, gu2.group_id
|
|
) AS Z ON Z.user_id = gu.user_id AND Z.group_id = gu.group_id
|
|
) AS X
|
|
WHERE X.user_id IN (
|
|
SELECT id
|
|
FROM users
|
|
WHERE last_seen_at IS NOT NULL
|
|
AND last_seen_at > :last_seen
|
|
ORDER BY last_seen_at DESC
|
|
LIMIT :limit
|
|
)
|
|
) Y
|
|
WHERE gu.user_id = Y.user_id AND gu.group_id = Y.group_id
|
|
SQL
|
|
end
|
|
|
|
protected
|
|
|
|
def set_notification_level
|
|
self.notification_level = group&.default_notification_level || 3
|
|
end
|
|
|
|
def set_primary_group
|
|
user.update!(primary_group: group) if group.primary_group
|
|
end
|
|
|
|
def remove_primary_and_flair_group
|
|
return if self.destroyed_by_association&.active_record == User # User is being destroyed, so don't try to update
|
|
|
|
updates = {}
|
|
updates[:primary_group_id] = nil if user.primary_group_id == group_id
|
|
updates[:flair_group_id] = nil if user.flair_group_id == group_id
|
|
|
|
user.update(updates) if updates.present?
|
|
end
|
|
|
|
def grant_other_available_title
|
|
if group.title.present? && group.title == user.title
|
|
user.update_attribute(:title, user.next_best_title)
|
|
end
|
|
end
|
|
|
|
def update_title
|
|
if group.title.present?
|
|
DB.exec("
|
|
UPDATE users SET title = :title
|
|
WHERE (title IS NULL OR title = '') AND id = :id",
|
|
id: user_id, title: group.title
|
|
)
|
|
end
|
|
end
|
|
|
|
def grant_trust_level
|
|
return if group.grant_trust_level.nil?
|
|
|
|
TrustLevelGranter.grant(group.grant_trust_level, user)
|
|
end
|
|
|
|
def recalculate_trust_level
|
|
return if group.grant_trust_level.nil?
|
|
return if self.destroyed_by_association&.active_record == User # User is being destroyed, so don't try to recalculate
|
|
|
|
Promotion.recalculate(user, use_previous_trust_level: true)
|
|
end
|
|
|
|
def set_category_notifications
|
|
self.class.set_category_notifications(group, user)
|
|
end
|
|
|
|
def self.set_category_notifications(group, user)
|
|
group_levels = group.group_category_notification_defaults.each_with_object({}) do |r, h|
|
|
h[r.notification_level] ||= []
|
|
h[r.notification_level] << r.category_id
|
|
end
|
|
|
|
return if group_levels.empty?
|
|
|
|
user_levels = CategoryUser.where(user_id: user.id).each_with_object({}) do |r, h|
|
|
h[r.notification_level] ||= []
|
|
h[r.notification_level] << r.category_id
|
|
end
|
|
|
|
higher_level_category_ids = user_levels.values.flatten
|
|
|
|
[:muted, :regular, :tracking, :watching_first_post, :watching].each do |level|
|
|
level_num = NotificationLevels.all[level]
|
|
higher_level_category_ids -= (user_levels[level_num] || [])
|
|
if group_category_ids = group_levels[level_num]
|
|
CategoryUser.batch_set(
|
|
user,
|
|
level,
|
|
group_category_ids + (user_levels[level_num] || []) - higher_level_category_ids
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
def set_tag_notifications
|
|
self.class.set_tag_notifications(group, user)
|
|
end
|
|
|
|
def self.set_tag_notifications(group, user)
|
|
group_levels = group.group_tag_notification_defaults.each_with_object({}) do |r, h|
|
|
h[r.notification_level] ||= []
|
|
h[r.notification_level] << r.tag_id
|
|
end
|
|
|
|
return if group_levels.empty?
|
|
|
|
user_levels = TagUser.where(user_id: user.id).each_with_object({}) do |r, h|
|
|
h[r.notification_level] ||= []
|
|
h[r.notification_level] << r.tag_id
|
|
end
|
|
|
|
higher_level_tag_ids = user_levels.values.flatten
|
|
|
|
[:muted, :regular, :tracking, :watching_first_post, :watching].each do |level|
|
|
level_num = NotificationLevels.all[level]
|
|
higher_level_tag_ids -= (user_levels[level_num] || [])
|
|
if group_tag_ids = group_levels[level_num]
|
|
TagUser.batch_set(
|
|
user,
|
|
level,
|
|
group_tag_ids + (user_levels[level_num] || []) - higher_level_tag_ids
|
|
)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
# == Schema Information
|
|
#
|
|
# Table name: group_users
|
|
#
|
|
# id :integer not null, primary key
|
|
# group_id :integer not null
|
|
# user_id :integer not null
|
|
# created_at :datetime not null
|
|
# updated_at :datetime not null
|
|
# owner :boolean default(FALSE), not null
|
|
# notification_level :integer default(2), not null
|
|
# first_unread_pm_at :datetime not null
|
|
#
|
|
# Indexes
|
|
#
|
|
# index_group_users_on_group_id_and_user_id (group_id,user_id) UNIQUE
|
|
# index_group_users_on_user_id_and_group_id (user_id,group_id) UNIQUE
|
|
#
|