eBooru/app/models/ban.rb
Earlopain fc7d84affd
[RuboCop] Enable Style/FrozenStringLiteralComment
This reduces allocations on the posts page by about 5%, from basic testing
2024-02-25 18:15:55 +01:00

151 lines
3.8 KiB
Ruby

# frozen_string_literal: true
class Ban < ApplicationRecord
attr_accessor :is_permaban
after_create :create_feedback
after_create :update_user_on_create
after_create :create_ban_mod_action
after_update :create_ban_update_mod_action
after_destroy :update_user_on_destroy
after_destroy :create_unban_mod_action
belongs_to :user
belongs_to :banner, :class_name => "User"
validate :user_is_inferior
validates :user_id, :reason, :duration, presence: true
before_validation :initialize_banner_id, :on => :create
before_validation :initialize_permaban, on: [:update, :create]
scope :unexpired, -> { where("bans.expires_at > ? OR bans.expires_at IS NULL", Time.now) }
scope :expired, -> { where("bans.expires_at IS NOT NULL").where("bans.expires_at <= ?", Time.now) }
def self.is_banned?(user)
exists?(["user_id = ? AND (expires_at > ? OR expires_at IS NULL)", user.id, Time.now])
end
def self.search(params)
q = super
q = q.where_user(:banner_id, :banner, params)
q = q.where_user(:user_id, :user, params)
q = q.attribute_matches(:reason, params[:reason_matches])
q = q.expired if params[:expired].to_s.truthy?
q = q.unexpired if params[:expired].to_s.falsy?
case params[:order]
when "expires_at_desc"
q = q.order("bans.expires_at desc")
else
q = q.apply_basic_order(params)
end
q
end
def self.prune!
expired.includes(:user).find_each do |ban|
ban.user.unban! if ban.user.ban_expired?
end
end
def initialize_banner_id
self.banner_id = CurrentUser.id if self.banner_id.blank?
end
def initialize_permaban
if is_permaban == "1"
self.duration = -1
end
end
def user_is_inferior
if user
if user.is_admin?
errors.add(:base, "You can never ban an admin.")
false
elsif user.is_moderator? && banner.is_admin?
true
elsif user.is_moderator?
errors.add(:base, "Only admins can ban moderators.")
false
elsif banner.is_admin? || banner.is_moderator?
true
else
errors.add(:base, "No one else can ban.")
false
end
end
end
def update_user_on_create
user.is_banned = true
user.level = User::Levels::BLOCKED
# Don't validate in order for deleted users to be bannable
user.save(validate: false)
end
def update_user_on_destroy
user.is_banned = false
user.level = User::Levels::MEMBER
user.save(validate: false)
end
def user_name
user ? user.name : nil
end
def user_name=(username)
self.user_id = User.name_to_id(username)
end
def duration=(dur)
dur = dur.to_i
if dur < 0
self.expires_at = nil
else
self.expires_at = dur.days.from_now
end
@duration = dur if dur != 0
end
def duration
@duration
end
def humanized_duration
return 'permanent' if expires_at == nil
ApplicationController.helpers.distance_of_time_in_words(created_at, expires_at)
end
def humanized_expiration
return 'never' if expires_at == nil
ApplicationController.helpers.compact_time expires_at
end
def expire_days
return 'never' if expires_at == nil
ApplicationController.helpers.time_ago_in_words(expires_at)
end
def expired?
expires_at != nil && expires_at < Time.now
end
def create_feedback
user.feedback.create(category: "negative", body: "Banned for #{humanized_duration}: #{reason}")
end
def create_ban_mod_action
ModAction.log(:user_ban, {duration: duration, reason: reason, user_id: user_id})
end
def create_ban_update_mod_action
ModAction.log(:user_ban_update, { user_id: user_id, ban_id: id, expires_at: expires_at, expires_at_was: expires_at_before_last_save, reason: reason, reason_was: reason_before_last_save })
end
def create_unban_mod_action
ModAction.log(:user_unban, {user_id: user_id})
end
end