mirror of
				https://github.com/pragma-/pbot.git
				synced 2025-10-25 20:47:32 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			152 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			152 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| # File: AntiRepeat.pm
 | |
| #
 | |
| # Purpose: Stops flooders/spammers from excessively repeating similiar messages.
 | |
| 
 | |
| # SPDX-FileCopyrightText: 2016-2023 Pragmatic Software <pragma78@gmail.com>
 | |
| # SPDX-License-Identifier: MIT
 | |
| 
 | |
| package PBot::Plugin::AntiRepeat;
 | |
| use parent 'PBot::Plugin::Base';
 | |
| 
 | |
| use PBot::Imports;
 | |
| 
 | |
| use String::LCSS qw/lcss/;
 | |
| use Time::HiRes qw/gettimeofday/;
 | |
| use POSIX qw/strftime/;
 | |
| 
 | |
| sub initialize($self, %conf) {
 | |
|     $self->{pbot}->{registry}->add_default('text', 'antiflood', 'antirepeat',           $conf{antirepeat}           // 1);
 | |
|     $self->{pbot}->{registry}->add_default('text', 'antiflood', 'antirepeat_threshold', $conf{antirepeat_threshold} // 3);
 | |
|     $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(@_) });
 | |
|     $self->{pbot}->{event_dispatcher}->register_handler('irc.caction', sub { $self->on_public(@_) });
 | |
| 
 | |
|     $self->{offenses} = {};
 | |
| }
 | |
| 
 | |
| sub unload($self) {
 | |
|     $self->{pbot}->{event_queue}->dequeue_event('antirepeat .*');
 | |
|     $self->{pbot}->{event_dispatcher}->remove_handler('irc.public');
 | |
|     $self->{pbot}->{event_dispatcher}->remove_handler('irc.caction');
 | |
| }
 | |
| 
 | |
| sub on_public($self, $event_type, $event) {
 | |
|     my ($nick, $user, $host, $msg) = (
 | |
|         $event->nick,
 | |
|         $event->user,
 | |
|         $event->host,
 | |
|         $event->args,
 | |
|     );
 | |
| 
 | |
|     my $channel = lc $event->{to}[0];
 | |
| 
 | |
|     ($nick, $user, $host) = $self->{pbot}->{irchandlers}->normalize_hostmask($nick, $user, $host);
 | |
| 
 | |
|     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;
 | |
| 
 | |
|     return 0 if $self->{pbot}->{registry}->get_value($channel, 'dont_enforce_antiflood');
 | |
| 
 | |
|     return 0 if $channel !~ m/^#/;
 | |
|     return 0 if $event->{interpreted};
 | |
| 
 | |
|     my $u = $self->{pbot}->{users}->loggedin($channel, "$nick!$user\@$host");
 | |
|     return 0 if $self->{pbot}->{capabilities}->userhas($u, 'is-whitelisted');
 | |
| 
 | |
|     my $account = $self->{pbot}->{messagehistory}->{database}->get_message_account($nick, $user, $host);
 | |
| 
 | |
|     # don't enforce anti-repeat for unreg spam
 | |
|     my $chanmodes = $self->{pbot}->{channels}->get_meta($channel, 'MODE');
 | |
|     if (defined $chanmodes and $chanmodes =~ m/z/ and $self->{pbot}->{banlist}->{quietlist}->exists($channel, '$~a')) {
 | |
|         my $nickserv = $self->{pbot}->{messagehistory}->{database}->get_current_nickserv_account($account);
 | |
|         return 0 if not defined $nickserv or not length $nickserv;
 | |
|     }
 | |
| 
 | |
|     my $messages = $self->{pbot}->{messagehistory}->{database}->get_recent_messages($account, $channel, 6, $self->{pbot}->{messagehistory}->{MSG_CHAT});
 | |
| 
 | |
|     my $botnick = $self->{pbot}->{conn}->nick;
 | |
| 
 | |
|     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');
 | |
| 
 | |
|     my %matches;
 | |
|     my $now = gettimeofday;
 | |
| 
 | |
|     foreach my $string1 (@$messages) {
 | |
|         next if $now - $string1->{timestamp} > 60 * 60 * 2;
 | |
|         next if $allow_bot and $string1->{msg} =~ m/^(?:$bot_trigger|$botnick.?)/;
 | |
|         $string1->{msg} =~ s/^[^;,:]{1,20}[;,:]//;    # remove nick-like prefix if one exists
 | |
|         next if length $string1->{msg} <= 10;          # allow really short messages since "yep" "ok" "thanks" etc are so common
 | |
| 
 | |
|         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 $now - $string2->{timestamp} > 60 * 60 * 2;
 | |
|             next if $allow_bot and $string2->{msg} =~ m/^(?:$bot_trigger|$botnick.?)/;
 | |
|             $string2->{msg} =~ s/^[^;,:]{1,20}[;,:]//;    # remove nick-like prefix if one exists
 | |
|             next if length $string2->{msg} <= 10;          # allow really short messages since "yep" "ok" "thanks" etc are so common
 | |
| 
 | |
|             if (exists $self->{offenses}->{$account} and exists $self->{offenses}->{$account}->{$channel}) {
 | |
|                 next if $self->{offenses}->{$account}->{$channel}->{last_offense} >= $string2->{timestamp};
 | |
|             }
 | |
| 
 | |
|             my $string = lcss(lc $string1->{msg}, lc $string2->{msg});
 | |
| 
 | |
|             if (defined $string) {
 | |
|                 my $length  = length $string;
 | |
|                 my $length1 = $length / length $string1->{msg};
 | |
|                 my $length2 = $length / length $string2->{msg};
 | |
| 
 | |
|                 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');
 | |
| 
 | |
|     foreach my $match (keys %matches) {
 | |
|         if (sqrt $matches{$match} > $threshold) {
 | |
|             $self->{offenses}->{$account}->{$channel}->{last_offense} = gettimeofday;
 | |
|             $self->{offenses}->{$account}->{$channel}->{offenses}++;
 | |
| 
 | |
|             $self->{pbot}->{event_queue}->enqueue_event(sub {
 | |
|                     my ($event) = @_;
 | |
|                     $self->{offenses}->{$account}->{$channel}->{offenses}--;
 | |
| 
 | |
|                     if ($self->{offenses}->{$account}->{$channel}->{offenses} <= 0) {
 | |
|                         $event->{repeating} = 0;
 | |
|                         delete $self->{offenses}->{$account}->{$channel};
 | |
|                         if (keys %{$self->{offenses}->{$account}} == 0) { delete $self->{offenses}->{$account}; }
 | |
|                     }
 | |
| 
 | |
|                 }, 60 * 60 * 2, "antirepeat offense-- $account $channel", 1
 | |
|             );
 | |
| 
 | |
|             $self->{pbot}->{logger}->log("$nick!$user\@$host triggered anti-repeat; offense $self->{offenses}->{$account}->{$channel}->{offenses}\n");
 | |
| 
 | |
|             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}->{banlist}->ban_user_timed($channel, 'b', "*!*\@$host", 30,      $botnick, 'repeating messages'); }
 | |
|                 when (3) { $self->{pbot}->{banlist}->ban_user_timed($channel, 'b', "*!*\@$host", 60 * 5,  $botnick, 'repeating messages'); }
 | |
|                 default  { $self->{pbot}->{banlist}->ban_user_timed($channel, 'b', "*!*\@$host", 60 * 60, $botnick, 'repeating messages'); }
 | |
|             }
 | |
|             return 0;
 | |
|         }
 | |
|     }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 1;
 | 
