3
0
mirror of https://github.com/pragma-/pbot.git synced 2025-01-11 20:42:38 +01:00
pbot/Plugins/AntiRepeat.pm.bak

202 lines
7.0 KiB
Perl
Raw Normal View History

2016-01-30 04:56:29 +01:00
package PBot::Plugins::AntiRepeat;
use warnings;
use strict;
use feature 'switch';
no if $] >= 5.018, warnings => "experimental::smartmatch";
use Carp ();
use String::LCSS qw/lcss/;
use Time::HiRes qw/gettimeofday/;
use POSIX qw/strftime/;
sub new {
Carp::croak("Options to " . __FILE__ . " should be key/value pairs, not hash reference") if ref $_[1] eq 'HASH';
my ($class, %conf) = @_;
my $self = bless {}, $class;
$self->initialize(%conf);
return $self;
}
sub initialize {
my ($self, %conf) = @_;
$self->{pbot} = delete $conf{pbot} // Carp::croak("Missing pbot reference to " . __FILE__);
$self->{pbot}->{registry}->add_default('text', 'antiflood', 'antirepeat', $conf{antirepeat} // 1);
$self->{pbot}->{registry}->add_default('text', 'antiflood', 'antirepeat_threshold', $conf{antirepeat_threshold} // 2.5);
$self->{pbot}->{registry}->add_default('text', 'antiflood', 'antirepeat_match', $conf{antirepeat_match} // 0.5);
$self->{pbot}->{registry}->add_default('text', 'antiflood', 'antirepeat_allow_bot', $conf{antirepeat_allow_bot} // 1);
$self->{pbot}->{event_dispatcher}->register_handler('irc.public', sub { $self->on_public(@_) });
2016-01-30 04:56:29 +01:00
$self->{pbot}->{timer}->register(sub { $self->adjust_offenses }, 60 * 60 * 1, 'antirepeat');
$self->{offenses} = {};
}
sub unload {
my $self = shift;
# perform plugin clean-up here
# normally we'd unregister the 'irc.public' event handler; however, the
# event dispatcher will do this automatically for us when it sees there
# is no longer an existing sub.
$self->{pbot}->{timer}->unregister('antirepeat');
}
sub on_public {
my ($self, $event_type, $event) = @_;
my ($nick, $user, $host, $msg) = ($event->{event}->nick, $event->{event}->user, $event->{event}->host, $event->{event}->args);
my $channel = $event->{event}->{to}[0];
2016-01-30 04:56:29 +01:00
return 0 if not $self->{pbot}->{registry}->get_value('antiflood', 'antirepeat');
my $antirepeat = $self->{pbot}->{registry}->get_value($channel, 'antirepeat');
return 0 if defined $antirepeat and not $antirepeat;
2016-01-30 04:56:29 +01:00
return 0 if $channel !~ m/^#/;
return 0 if $event->{interpreted};
return 0 if $self->{pbot}->{antiflood}->whitelisted($channel, "$nick!$user\@$host", 'antiflood');
my $account = $self->{pbot}->{messagehistory}->{database}->get_message_account($nick, $user, $host);
my $messages = $self->{pbot}->{messagehistory}->{database}->get_recent_messages($account, $channel, 6, $self->{pbot}->{messagehistory}->{MSG_CHAT});
my %matches;
my $log = '';
my $now = gettimeofday;
2016-01-30 04:56:29 +01:00
my $botnick = $self->{pbot}->{registry}->get_value('irc', 'botnick');
my $bot_trigger = $self->{pbot}->{registry}->get_value($channel, 'trigger')
// $self->{pbot}->{registry}->get_value('general', 'trigger');
my $allow_bot = $self->{pbot}->{registry}->get_value($channel, 'antirepeat_allow_bot')
// $self->{pbot}->{registry}->get_value('antiflood', 'antirepeat_allow_bot');
my $match = $self->{pbot}->{registry}->get_value($channel, 'antirepeat_match')
// $self->{pbot}->{registry}->get_value('antiflood', 'antirepeat_match');
foreach my $string1 (@$messages) {
next if length $string1->{msg} < 10;
2016-01-30 04:56:29 +01:00
next if $now - $string1->{timestamp} > 60 * 60 * 2;
next if $allow_bot and $string1->{msg} =~ m/^(?:$bot_trigger|$botnick.?)/;
if (exists $self->{offenses}->{$account} and exists $self->{offenses}->{$account}->{$channel}) {
next if $self->{offenses}->{$account}->{$channel}->{last_offense} >= $string1->{timestamp};
}
foreach my $string2 (@$messages) {
next if length $string2->{msg} < 10;
2016-01-30 04:56:29 +01:00
next if $now - $string2->{timestamp} > 60 * 60 * 2;
next if $allow_bot and $string2->{msg} =~ m/^(?:$bot_trigger|$botnick.?)/;
if (exists $self->{offenses}->{$account} and exists $self->{offenses}->{$account}->{$channel}) {
next if $self->{offenses}->{$account}->{$channel}->{last_offense} >= $string2->{timestamp};
}
my $string = lcss($string1->{msg}, $string2->{msg});
2016-01-30 04:56:29 +01:00
if (defined $string) {
my $length = length $string;
my $length1 = $length / length $string1->{msg};
my $length2 = $length / length $string2->{msg};
$log .= " $string -- ";
$log .= " 1: [$string1->{msg}] -- ";
$log .= " 2: [$string2->{msg}]";
$log .= " $length1 - $length2\n";
2016-01-30 04:56:29 +01:00
if ($length1 >= $match && $length2 >= $match) {
$matches{$string}++;
}
}
}
}
my $threshold = $self->{pbot}->{registry}->get_value($channel, 'antirepeat_threshold')
// $self->{pbot}->{registry}->get_value('antiflood', 'antirepeat_threshold');
my $ban = 0;
2016-01-30 04:56:29 +01:00
foreach my $match (keys %matches) {
my $sqrt = sqrt $matches{$match};
$log .= "- $matches{$match} (sqrt: $sqrt) matches: $match\n";
2016-01-30 04:56:29 +01:00
if (sqrt $matches{$match} > $threshold) {
$log .= " +++ BAN!\n";
$ban = 1;
}
}
if ($ban) {
print "------------\n";
print $log;
open my $fh, '>> banlog.txt';
print $fh "-------------\n";
print $fh "$channel <$nick!$user\@$host> $msg\n";
foreach my $string (@$messages) {
my $time = strftime "%a %b %e %H:%M:%S %Z %Y", localtime $string->{timestamp};
print $fh "-> $time $string->{msg}\n";
}
print $fh $log;
close $fh;
if ($ban && ($channel eq '##c')) {
2016-01-30 04:56:29 +01:00
$self->{offenses}->{$account}->{$channel}->{last_offense} = gettimeofday;
$self->{offenses}->{$account}->{$channel}->{last_adjustment} = gettimeofday;
$self->{offenses}->{$account}->{$channel}->{offenses}++;
$self->{pbot}->{logger}->log("antirepeat: offenses for $account/$channel = $self->{offenses}->{$account}->{$channel}->{offenses}\n");
2016-01-30 04:56:29 +01:00
given ($self->{offenses}->{$account}->{$channel}->{offenses}) {
when (1) {
$self->{pbot}->{chanops}->add_op_command($channel, "kick $channel $nick Stop repeating yourself");
$self->{pbot}->{chanops}->gain_ops($channel);
}
when (2) {
$self->{pbot}->{chanops}->ban_user_timed("*!*\@$host", $channel, 60 * 15);
2016-01-30 04:56:29 +01:00
}
when (3) {
$self->{pbot}->{chanops}->ban_user_timed("*!*\@$host", $channel, 60 * 60 * 24 * 7);
2016-01-30 04:56:29 +01:00
}
default {
$self->{pbot}->{chanops}->ban_user_timed("*!*\@$host", $channel, 60 * 60 * 24 * 31);
2016-01-30 04:56:29 +01:00
}
}
}
}
return 0;
}
sub adjust_offenses {
my $self = shift;
my $now = gettimeofday;
foreach my $account (keys %{ $self->{offenses} }) {
foreach my $channel (keys %{ $self->{offenses}->{$account} }) {
if ($self->{offenses}->{$account}->{$channel}->{offenses} > 0 and $now - $self->{offenses}->{$account}->{$channel}->{last_adjustment} > 60 * 60 * 3) {
$self->{offenses}->{$account}->{$channel}->{offenses}--;
if ($self->{offenses}->{$account}->{$channel}->{offenses} <= 0) {
delete $self->{offenses}->{$account}->{$channel};
if (keys $self->{offenses}->{$account} == 0) {
2016-01-30 04:56:29 +01:00
delete $self->{offenses}->{$account};
}
} else {
$self->{offenses}->{$account}->{$channel}->{last_adjustment} = $now;
}
}
}
}
}
1;