Autoban dmail spambots (#3408).
If a user sends spam to more than 10 users within a 24 hour window, automatically ban them for 3 days.
This commit is contained in:
@@ -1,6 +1,11 @@
|
|||||||
require 'digest/sha1'
|
require 'digest/sha1'
|
||||||
|
|
||||||
class Dmail < ApplicationRecord
|
class Dmail < ApplicationRecord
|
||||||
|
# if a person sends spam to more than 10 users within a 24 hour window, automatically ban them for 3 days.
|
||||||
|
AUTOBAN_THRESHOLD = 10
|
||||||
|
AUTOBAN_WINDOW = 24.hours
|
||||||
|
AUTOBAN_DURATION = 3
|
||||||
|
|
||||||
include Rakismet::Model
|
include Rakismet::Model
|
||||||
|
|
||||||
with_options on: :create do
|
with_options on: :create do
|
||||||
@@ -23,6 +28,23 @@ class Dmail < ApplicationRecord
|
|||||||
rakismet_attrs author: :from_name, author_email: :from_email, content: :title_and_body, user_ip: :creator_ip_addr_str
|
rakismet_attrs author: :from_name, author_email: :from_email, content: :title_and_body, user_ip: :creator_ip_addr_str
|
||||||
|
|
||||||
concerning :SpamMethods do
|
concerning :SpamMethods do
|
||||||
|
class_methods do
|
||||||
|
def is_spammer?(user)
|
||||||
|
return false if user.is_gold?
|
||||||
|
|
||||||
|
spammed_users = sent_by(user).where(is_spam: true).where("created_at > ?", AUTOBAN_WINDOW.ago).distinct.count(:to_id)
|
||||||
|
spammed_users >= AUTOBAN_THRESHOLD
|
||||||
|
end
|
||||||
|
|
||||||
|
def ban_spammer(spammer)
|
||||||
|
spammer.bans.create! do |ban|
|
||||||
|
ban.banner = User.system
|
||||||
|
ban.reason = "Spambot."
|
||||||
|
ban.duration = AUTOBAN_DURATION
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def title_and_body
|
def title_and_body
|
||||||
"#{title}\n\n#{body}"
|
"#{title}\n\n#{body}"
|
||||||
end
|
end
|
||||||
@@ -80,6 +102,8 @@ class Dmail < ApplicationRecord
|
|||||||
copy.owner_id = copy.from_id
|
copy.owner_id = copy.from_id
|
||||||
copy.is_read = true
|
copy.is_read = true
|
||||||
copy.save
|
copy.save
|
||||||
|
|
||||||
|
Dmail.ban_spammer(copy.from) if Dmail.is_spammer?(copy.from)
|
||||||
end
|
end
|
||||||
|
|
||||||
copy
|
copy
|
||||||
@@ -119,6 +143,10 @@ class Dmail < ApplicationRecord
|
|||||||
end
|
end
|
||||||
|
|
||||||
module SearchMethods
|
module SearchMethods
|
||||||
|
def sent_by(user)
|
||||||
|
where("dmails.from_id = ? AND dmails.owner_id != ?", user.id, user.id)
|
||||||
|
end
|
||||||
|
|
||||||
def active
|
def active
|
||||||
where("is_deleted = ?", false)
|
where("is_deleted = ?", false)
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ require 'test_helper'
|
|||||||
class DmailTest < ActiveSupport::TestCase
|
class DmailTest < ActiveSupport::TestCase
|
||||||
context "A dmail" do
|
context "A dmail" do
|
||||||
setup do
|
setup do
|
||||||
|
User.any_instance.stubs(:validate_sock_puppets).returns(true)
|
||||||
@user = FactoryGirl.create(:user)
|
@user = FactoryGirl.create(:user)
|
||||||
CurrentUser.user = @user
|
CurrentUser.user = @user
|
||||||
CurrentUser.ip_addr = "1.2.3.4"
|
CurrentUser.ip_addr = "1.2.3.4"
|
||||||
@@ -27,6 +28,20 @@ class DmailTest < ActiveSupport::TestCase
|
|||||||
assert(@recipient.dmails.last.is_spam?)
|
assert(@recipient.dmails.last.is_spam?)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
should "autoban spammers after sending spam to N distinct users" do
|
||||||
|
Dmail.any_instance.expects(:spam?).returns(true)
|
||||||
|
|
||||||
|
users = FactoryGirl.create_list(:user, Dmail::AUTOBAN_THRESHOLD)
|
||||||
|
users.each do |user|
|
||||||
|
Dmail.create_split(from: @user, to: user, title: "spam", body: "wonderful spam")
|
||||||
|
end
|
||||||
|
|
||||||
|
assert_equal(true, Dmail.is_spammer?(@user))
|
||||||
|
assert_equal(true, @user.reload.is_banned)
|
||||||
|
assert_equal(1, @user.bans.count)
|
||||||
|
assert_match(/Spambot./, @user.bans.last.reason)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
context "filter" do
|
context "filter" do
|
||||||
|
|||||||
Reference in New Issue
Block a user