2010-03-23 19:24:02 +01:00
|
|
|
# File: ChanOpCommands.pm
|
2010-03-24 07:47:40 +01:00
|
|
|
# Author: pragma_
|
2010-03-23 19:24:02 +01:00
|
|
|
#
|
|
|
|
# Purpose: Channel operator command subroutines.
|
|
|
|
|
2017-03-05 22:33:31 +01:00
|
|
|
# This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
# License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
2010-03-23 19:24:02 +01:00
|
|
|
package PBot::ChanOpCommands;
|
|
|
|
|
|
|
|
use warnings;
|
|
|
|
use strict;
|
|
|
|
|
2019-07-11 03:40:53 +02:00
|
|
|
use feature 'unicode_strings';
|
|
|
|
|
2010-03-23 19:24:02 +01:00
|
|
|
use Carp ();
|
2015-04-12 01:00:20 +02:00
|
|
|
use Time::Duration;
|
2019-07-23 17:55:24 +02:00
|
|
|
use Time::HiRes qw/gettimeofday/;
|
2015-04-14 00:43:19 +02:00
|
|
|
|
2010-03-23 19:24:02 +01:00
|
|
|
sub new {
|
2017-11-10 04:26:05 +01:00
|
|
|
if (ref($_[1]) eq 'HASH') {
|
2010-03-23 19:24:02 +01:00
|
|
|
Carp::croak("Options to ChanOpCommands should be key/value pairs, not hash reference");
|
|
|
|
}
|
|
|
|
|
|
|
|
my ($class, %conf) = @_;
|
|
|
|
|
|
|
|
my $self = bless {}, $class;
|
|
|
|
$self->initialize(%conf);
|
|
|
|
return $self;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub initialize {
|
|
|
|
my ($self, %conf) = @_;
|
|
|
|
|
|
|
|
my $pbot = delete $conf{pbot};
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $pbot) {
|
2010-03-23 19:24:02 +01:00
|
|
|
Carp::croak("Missing pbot reference to ChanOpCommands");
|
|
|
|
}
|
|
|
|
|
|
|
|
$self->{pbot} = $pbot;
|
2017-11-10 04:26:05 +01:00
|
|
|
|
2014-05-18 22:09:05 +02:00
|
|
|
$pbot->{commands}->register(sub { return $self->ban_user(@_) }, "ban", 10);
|
|
|
|
$pbot->{commands}->register(sub { return $self->unban_user(@_) }, "unban", 10);
|
2015-05-27 19:45:43 +02:00
|
|
|
$pbot->{commands}->register(sub { return $self->mute_user(@_) }, "mute", 10);
|
|
|
|
$pbot->{commands}->register(sub { return $self->unmute_user(@_) }, "unmute", 10);
|
2014-05-18 22:09:05 +02:00
|
|
|
$pbot->{commands}->register(sub { return $self->kick_user(@_) }, "kick", 10);
|
2019-07-23 17:55:24 +02:00
|
|
|
$pbot->{commands}->register(sub { return $self->checkban(@_) }, "checkban", 0);
|
|
|
|
$pbot->{commands}->register(sub { return $self->checkmute(@_) }, "checkmute", 0);
|
2019-12-26 15:08:39 +01:00
|
|
|
$pbot->{commands}->register(sub { return $self->mode(@_) }, "mode", 40);
|
2019-12-29 08:17:15 +01:00
|
|
|
$pbot->{commands}->register(sub { return $self->invite(@_) }, "invite", 10);
|
|
|
|
}
|
|
|
|
|
|
|
|
sub invite {
|
|
|
|
my ($self, $from, $nick, $user, $host, $arguments, $stuff) = @_;
|
|
|
|
|
|
|
|
if (not length $arguments) {
|
|
|
|
return "Usage: invite [channel] <nick>";
|
|
|
|
}
|
|
|
|
|
|
|
|
# add current channel as default channel
|
|
|
|
if ($stuff->{arglist}[0] !~ m/^#/) {
|
|
|
|
if ($from =~ m/^#/) {
|
|
|
|
unshift @{$stuff->{arglist}}, $from;
|
|
|
|
} else {
|
|
|
|
return "Usage from private message: invite <channel> <nick>";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
my ($channel, $target) = $self->{pbot}->{interpreter}->split_args($stuff->{arglist}, 2);
|
|
|
|
$self->{pbot}->{chanops}->add_op_command($channel, "sl invite $target $channel");
|
|
|
|
$self->{pbot}->{chanops}->gain_ops($channel);
|
|
|
|
return "";
|
2019-12-26 15:08:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
sub mode {
|
|
|
|
my ($self, $from, $nick, $user, $host, $arguments, $stuff) = @_;
|
|
|
|
|
|
|
|
if (not length $arguments) {
|
|
|
|
return "Usage: mode [channel] <arguments>";
|
|
|
|
}
|
|
|
|
|
|
|
|
# add current channel as default channel
|
|
|
|
if ($stuff->{arglist}[0] !~ m/^#/) {
|
|
|
|
if ($from =~ m/^#/) {
|
|
|
|
unshift @{$stuff->{arglist}}, $from;
|
|
|
|
} else {
|
|
|
|
return "Usage from private message: mode <channel> <arguments>";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-27 04:24:35 +01:00
|
|
|
my ($channel, $modes, $args) = $self->{pbot}->{interpreter}->split_args($stuff->{arglist}, 3);
|
|
|
|
my @targets = split /\s+/, $args;
|
|
|
|
my $modifier;
|
|
|
|
my $i = 0;
|
|
|
|
my $arg = 0;
|
|
|
|
|
|
|
|
my ($new_modes, $new_targets) = ("", "");
|
2019-12-27 08:00:14 +01:00
|
|
|
my $max_modes = $self->{pbot}->{ircd}->{MODES} // 1;
|
2019-12-27 04:24:35 +01:00
|
|
|
|
|
|
|
while ($modes =~ m/(.)/g) {
|
|
|
|
my $mode = $1;
|
|
|
|
|
|
|
|
if ($mode eq '-' or $mode eq '+') {
|
|
|
|
$modifier = $mode;
|
|
|
|
$new_modes .= $mode;
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $target = $targets[$arg++] // "";
|
|
|
|
|
|
|
|
if (($mode eq 'v' or $mode eq 'o') and $target =~ m/\*/) {
|
2019-12-31 00:59:56 +01:00
|
|
|
# wildcard used; find all matching nicks; test against whitelist, etc
|
2019-12-27 04:24:35 +01:00
|
|
|
my $q_target = lc quotemeta $target;
|
|
|
|
$q_target =~ s/\\\*/.*/g;
|
|
|
|
$channel = lc $channel;
|
|
|
|
|
|
|
|
if (not exists $self->{pbot}->{nicklist}->{nicklist}->{$channel}) {
|
2019-12-31 00:59:56 +01:00
|
|
|
return "I have no nicklist for channel $channel; cannot use wildcard.";
|
2019-12-27 04:24:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach my $n (keys %{$self->{pbot}->{nicklist}->{nicklist}->{$channel}}) {
|
|
|
|
if ($n =~ m/^$q_target$/) {
|
2019-12-31 00:59:56 +01:00
|
|
|
my $nick_data = $self->{pbot}->{nicklist}->{nicklist}->{$channel}->{$n};
|
|
|
|
|
|
|
|
if ($modifier eq '-') {
|
|
|
|
# removing mode -- check against whitelist, etc
|
2019-12-31 03:17:35 +01:00
|
|
|
next if $nick_data->{nick} eq $self->{pbot}->{registry}->get_value('irc', 'botnick');
|
2019-12-31 00:59:56 +01:00
|
|
|
next if $self->{pbot}->{antiflood}->whitelisted($channel, $nick_data->{hostmask});
|
|
|
|
next if $self->{pbot}->{admins}->loggedin($channel, $nick_data->{hostmask});
|
|
|
|
}
|
|
|
|
|
|
|
|
# skip nick if already has mode set/unset
|
|
|
|
if ($modifier eq '+') {
|
|
|
|
next if exists $nick_data->{"+$mode"};
|
|
|
|
} else {
|
|
|
|
next unless exists $nick_data->{"+$mode"};
|
|
|
|
}
|
|
|
|
|
2019-12-27 04:24:35 +01:00
|
|
|
$new_modes = $modifier if not length $new_modes;
|
|
|
|
$new_modes .= $mode;
|
|
|
|
$new_targets .= "$self->{pbot}->{nicklist}->{nicklist}->{$channel}->{$n}->{nick} ";
|
|
|
|
$i++;
|
|
|
|
|
2019-12-27 08:00:14 +01:00
|
|
|
if ($i == $max_modes) {
|
2019-12-27 04:24:35 +01:00
|
|
|
$self->{pbot}->{chanops}->add_op_command($channel, "mode $channel $new_modes $new_targets");
|
|
|
|
$new_modes = "";
|
|
|
|
$new_targets = "";
|
|
|
|
$i = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2019-12-31 00:59:56 +01:00
|
|
|
# no wildcard used; explicit mode requested - no whitelist checking
|
2019-12-27 04:24:35 +01:00
|
|
|
$new_modes .= $mode;
|
|
|
|
$new_targets .= "$target " if length $target;
|
|
|
|
$i++;
|
|
|
|
|
2019-12-27 08:00:14 +01:00
|
|
|
if ($i == $max_modes) {
|
2019-12-27 04:24:35 +01:00
|
|
|
$self->{pbot}->{chanops}->add_op_command($channel, "mode $channel $new_modes $new_targets");
|
|
|
|
$new_modes = "";
|
|
|
|
$new_targets = "";
|
|
|
|
$i = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($i) {
|
|
|
|
$self->{pbot}->{chanops}->add_op_command($channel, "mode $channel $new_modes $new_targets");
|
|
|
|
}
|
|
|
|
|
|
|
|
$self->{pbot}->{chanops}->gain_ops($channel);
|
2019-12-26 15:08:39 +01:00
|
|
|
|
|
|
|
if ($from !~ m/^#/) {
|
|
|
|
return "Done.";
|
|
|
|
}
|
2019-07-23 17:55:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
sub checkban {
|
|
|
|
my ($self, $from, $nick, $user, $host, $arguments, $stuff) = @_;
|
|
|
|
my ($target, $channel) = $self->{pbot}->{interpreter}->split_args($stuff->{arglist}, 2);
|
|
|
|
|
|
|
|
if (not defined $target) {
|
|
|
|
return "Usage: checkban <mask> [channel]";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not defined $channel) {
|
|
|
|
$channel = $from;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($channel !~ /^#/) {
|
|
|
|
return "Please specify a channel.";
|
|
|
|
}
|
|
|
|
|
|
|
|
$channel = lc $channel;
|
|
|
|
$target = lc $target;
|
|
|
|
|
|
|
|
my $mask = $self->{pbot}->{chanops}->nick_to_banmask($target);
|
|
|
|
|
|
|
|
if (exists $self->{pbot}->{chanops}->{unban_timeout}->hash->{$channel}
|
|
|
|
&& exists $self->{pbot}->{chanops}->{unban_timeout}->hash->{$channel}->{$mask}) {
|
|
|
|
my $timeout = $self->{pbot}->{chanops}->{unban_timeout}->hash->{$channel}->{$mask}{timeout};
|
2019-07-26 23:17:06 +02:00
|
|
|
my $owner = $self->{pbot}->{chanops}->{unban_timeout}->hash->{$channel}->{$mask}{owner};
|
|
|
|
my $reason = $self->{pbot}->{chanops}->{unban_timeout}->hash->{$channel}->{$mask}{reason};
|
|
|
|
my $duration = concise duration($timeout - gettimeofday);
|
|
|
|
|
|
|
|
my $result = "$mask banned in $channel ";
|
|
|
|
$result .= "by $owner " if defined $owner;
|
|
|
|
$result .= "for $reason " if defined $reason;
|
|
|
|
$result .= "($duration remaining)";
|
|
|
|
return $result;
|
2019-07-23 17:55:24 +02:00
|
|
|
} else {
|
|
|
|
return "$mask has no ban timeout";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub checkmute {
|
|
|
|
my ($self, $from, $nick, $user, $host, $arguments, $stuff) = @_;
|
|
|
|
my ($target, $channel) = $self->{pbot}->{interpreter}->split_args($stuff->{arglist}, 2);
|
|
|
|
|
|
|
|
if (not defined $target) {
|
|
|
|
return "Usage: checkmute <mask> [channel]";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not defined $channel) {
|
|
|
|
$channel = $from;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($channel !~ /^#/) {
|
|
|
|
return "Please specify a channel.";
|
|
|
|
}
|
|
|
|
|
|
|
|
$channel = lc $channel;
|
|
|
|
$target = lc $target;
|
|
|
|
|
|
|
|
my $mask = $self->{pbot}->{chanops}->nick_to_banmask($target);
|
|
|
|
|
|
|
|
if (exists $self->{pbot}->{chanops}->{unmute_timeout}->hash->{$channel}
|
|
|
|
&& exists $self->{pbot}->{chanops}->{unmute_timeout}->hash->{$channel}->{$mask}) {
|
|
|
|
my $timeout = $self->{pbot}->{chanops}->{unmute_timeout}->hash->{$channel}->{$mask}{timeout};
|
2019-07-26 23:17:06 +02:00
|
|
|
my $owner = $self->{pbot}->{chanops}->{unmute_timeout}->hash->{$channel}->{$mask}{owner};
|
|
|
|
my $reason = $self->{pbot}->{chanops}->{unmute_timeout}->hash->{$channel}->{$mask}{reason};
|
|
|
|
my $duration = concise duration($timeout - gettimeofday);
|
|
|
|
|
|
|
|
my $result = "$mask muted in $channel ";
|
|
|
|
$result .= "by $owner " if defined $owner;
|
|
|
|
$result .= "for $reason " if defined $reason;
|
|
|
|
$result .= "($duration remaining)";
|
2019-07-23 17:55:24 +02:00
|
|
|
|
|
|
|
return "$mask has $duration remaining on their $channel mute";
|
|
|
|
} else {
|
|
|
|
return "$mask has no mute timeout";
|
|
|
|
}
|
2010-03-23 19:24:02 +01:00
|
|
|
}
|
|
|
|
|
2010-06-18 09:03:16 +02:00
|
|
|
sub ban_user {
|
2019-11-25 22:57:37 +01:00
|
|
|
my ($self, $from, $nick, $user, $host, $arguments, $stuff) = @_;
|
2018-08-09 02:38:57 +02:00
|
|
|
my ($target, $channel, $length) = $self->{pbot}->{interpreter}->split_args($stuff->{arglist}, 3);
|
2010-03-23 19:24:02 +01:00
|
|
|
|
2017-04-11 04:13:56 +02:00
|
|
|
$channel = '' if not defined $channel;
|
|
|
|
$length = '' if not defined $length;
|
|
|
|
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $from) {
|
2014-05-18 22:09:05 +02:00
|
|
|
$self->{pbot}->{logger}->log("Command missing ~from parameter!\n");
|
2010-03-23 19:24:02 +01:00
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:45:43 +02:00
|
|
|
if ($channel !~ m/^#/) {
|
|
|
|
$length = "$channel $length";
|
2015-06-10 11:27:13 +02:00
|
|
|
$length = undef if $length eq ' ';
|
2018-02-22 03:41:56 +01:00
|
|
|
$channel = exists $stuff->{admin_channel_override} ? $stuff->{admin_channel_override} : $from;
|
2015-05-27 19:45:43 +02:00
|
|
|
}
|
|
|
|
|
2018-02-22 03:41:56 +01:00
|
|
|
$channel = exists $stuff->{admin_channel_override} ? $stuff->{admin_channel_override} : $from if not defined $channel or not length $channel;
|
2010-03-23 19:24:02 +01:00
|
|
|
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $target) {
|
2015-05-27 19:45:43 +02:00
|
|
|
return "/msg $nick Usage: ban <mask> [channel [timeout (default: 24 hours)]]";
|
2010-03-23 19:24:02 +01:00
|
|
|
}
|
|
|
|
|
2019-07-23 17:55:24 +02:00
|
|
|
my $no_length = 0;
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $length) {
|
2019-11-25 22:57:37 +01:00
|
|
|
$length = $self->{pbot}->{registry}->get_value($channel, 'default_ban_timeout', 0, $stuff) //
|
|
|
|
$self->{pbot}->{registry}->get_value('general', 'default_ban_timeout', 0, $stuff) // 60 * 60 * 24; # 24 hours
|
2019-07-23 17:55:24 +02:00
|
|
|
$no_length = 1;
|
2015-04-12 01:00:20 +02:00
|
|
|
} else {
|
2015-04-14 00:43:19 +02:00
|
|
|
my $error;
|
2019-06-07 06:46:00 +02:00
|
|
|
($length, $error) = $self->{pbot}->{parsedate}->parsedate($length);
|
2015-04-14 00:43:19 +02:00
|
|
|
return $error if defined $error;
|
2010-03-23 19:24:02 +01:00
|
|
|
}
|
|
|
|
|
2019-07-23 17:55:24 +02:00
|
|
|
$channel = lc $channel;
|
|
|
|
$target = lc $target;
|
|
|
|
|
2014-05-17 22:08:19 +02:00
|
|
|
my $botnick = $self->{pbot}->{registry}->get_value('irc', 'botnick');
|
2015-05-07 06:13:39 +02:00
|
|
|
return "I don't think so." if $target =~ /^\Q$botnick\E!/i;
|
2010-03-23 19:24:02 +01:00
|
|
|
|
2019-06-23 19:34:41 +02:00
|
|
|
if (not $stuff->{'effective-level'} and not $self->{pbot}->{admins}->loggedin($channel, "$nick!$user\@$host")) {
|
2017-11-03 20:28:41 +01:00
|
|
|
return "/msg $nick You are not an admin for $channel.";
|
|
|
|
}
|
|
|
|
|
2019-07-23 17:55:24 +02:00
|
|
|
my $result = '';
|
|
|
|
my $sep = '';
|
2018-06-06 03:28:03 +02:00
|
|
|
my @targets = split /,/, $target;
|
2018-06-06 07:59:33 +02:00
|
|
|
my $immediately = @targets > 1 ? 0 : 1;
|
2018-06-06 03:28:03 +02:00
|
|
|
foreach my $t (@targets) {
|
2019-07-23 17:55:24 +02:00
|
|
|
my $mask = $self->{pbot}->{chanops}->nick_to_banmask($t);
|
|
|
|
|
|
|
|
if ($no_length && exists $self->{pbot}->{chanops}->{unban_timeout}->hash->{$channel}
|
|
|
|
&& exists $self->{pbot}->{chanops}->{unban_timeout}->hash->{$channel}->{$mask}) {
|
|
|
|
my $timeout = $self->{pbot}->{chanops}->{unban_timeout}->hash->{$channel}->{$mask}{timeout};
|
|
|
|
my $duration = duration($timeout - gettimeofday);
|
|
|
|
|
|
|
|
$result .= "$sep$mask has $duration remaining on their $channel ban";
|
|
|
|
$sep = '; ';
|
|
|
|
} else {
|
2019-07-26 23:17:06 +02:00
|
|
|
$self->{pbot}->{chanops}->ban_user_timed("$nick!$user\@$host", undef, $mask, $channel, $length, $immediately);
|
2019-07-23 17:55:24 +02:00
|
|
|
|
|
|
|
my $duration;
|
|
|
|
if ($length > 0) {
|
|
|
|
$duration = duration($length);
|
|
|
|
} else {
|
|
|
|
$duration = 'all eternity';
|
|
|
|
}
|
|
|
|
|
|
|
|
$result .= "$sep$mask banned in $channel for $duration";
|
|
|
|
$sep = '; ';
|
|
|
|
}
|
2018-06-06 07:59:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (not $immediately) {
|
|
|
|
$self->{pbot}->{chanops}->check_ban_queue;
|
2018-06-06 03:28:03 +02:00
|
|
|
}
|
2015-05-07 06:13:39 +02:00
|
|
|
|
2019-07-23 17:55:24 +02:00
|
|
|
$result = "/msg $nick $result" if $result !~ m/remaining on their/;
|
|
|
|
return $result;
|
2010-03-23 19:24:02 +01:00
|
|
|
}
|
|
|
|
|
2010-06-18 09:03:16 +02:00
|
|
|
sub unban_user {
|
2010-03-23 19:24:02 +01:00
|
|
|
my $self = shift;
|
2018-02-22 02:21:38 +01:00
|
|
|
my ($from, $nick, $user, $host, $arguments, $stuff) = @_;
|
2010-03-23 19:24:02 +01:00
|
|
|
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $from) {
|
2014-05-18 22:09:05 +02:00
|
|
|
$self->{pbot}->{logger}->log("Command missing ~from parameter!\n");
|
2010-03-23 19:24:02 +01:00
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2018-08-09 02:38:57 +02:00
|
|
|
my ($target, $channel, $immediately) = $self->{pbot}->{interpreter}->split_args($stuff->{arglist}, 3);
|
2010-03-23 19:24:02 +01:00
|
|
|
|
2018-06-06 02:55:59 +02:00
|
|
|
if (defined $target and defined $channel and $channel !~ /^#/) {
|
|
|
|
my $temp = $target;
|
|
|
|
$target = $channel;
|
|
|
|
$channel = $temp;
|
|
|
|
}
|
|
|
|
|
2019-05-28 18:19:42 +02:00
|
|
|
if (not defined $target) {
|
2018-06-06 03:28:03 +02:00
|
|
|
return "/msg $nick Usage: unban <nick/mask> [[channel] [false value to use unban queue]]";
|
2010-03-23 19:24:02 +01:00
|
|
|
}
|
|
|
|
|
2018-02-22 03:41:56 +01:00
|
|
|
$channel = exists $stuff->{admin_channel_override} ? $stuff->{admin_channel_override} : $from if not defined $channel;
|
2017-11-08 20:11:43 +01:00
|
|
|
$immediately = 1 if not defined $immediately;
|
2017-11-10 04:26:05 +01:00
|
|
|
|
2018-06-06 03:28:03 +02:00
|
|
|
return "/msg $nick Usage for /msg: unban <nick/mask> <channel> [false value to use unban queue]" if $channel !~ /^#/;
|
2010-03-23 19:24:02 +01:00
|
|
|
|
2019-06-23 19:34:41 +02:00
|
|
|
if (not $stuff->{'effective-level'} and not $self->{pbot}->{admins}->loggedin($channel, "$nick!$user\@$host")) {
|
2017-11-03 20:28:41 +01:00
|
|
|
return "/msg $nick You are not an admin for $channel.";
|
|
|
|
}
|
|
|
|
|
2017-11-08 20:11:43 +01:00
|
|
|
my @targets = split /,/, $target;
|
2018-06-06 07:59:33 +02:00
|
|
|
$immediately = 0 if @targets > 1;
|
2017-11-08 20:11:43 +01:00
|
|
|
|
|
|
|
foreach my $t (@targets) {
|
|
|
|
$self->{pbot}->{chanops}->unban_user($t, $channel, $immediately);
|
|
|
|
}
|
|
|
|
|
2018-06-06 07:59:33 +02:00
|
|
|
if (@targets > 1) {
|
|
|
|
$self->{pbot}->{chanops}->check_unban_queue;
|
|
|
|
}
|
|
|
|
|
2011-10-31 20:56:08 +01:00
|
|
|
return "/msg $nick $target has been unbanned from $channel.";
|
2010-03-23 19:24:02 +01:00
|
|
|
}
|
|
|
|
|
2015-05-27 19:45:43 +02:00
|
|
|
sub mute_user {
|
2019-11-25 22:57:37 +01:00
|
|
|
my ($self, $from, $nick, $user, $host, $arguments, $stuff) = @_;
|
2018-08-09 02:38:57 +02:00
|
|
|
my ($target, $channel, $length) = $self->{pbot}->{interpreter}->split_args($stuff->{arglist}, 3);
|
2015-05-27 19:45:43 +02:00
|
|
|
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $from) {
|
2015-05-27 19:45:43 +02:00
|
|
|
$self->{pbot}->{logger}->log("Command missing ~from parameter!\n");
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2016-11-17 04:14:39 +01:00
|
|
|
if (not defined $channel and $from !~ m/^#/) {
|
|
|
|
return "/msg $nick Usage from private message: mute <mask> <channel> [timeout (default: 24 hours)]";
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:45:43 +02:00
|
|
|
if ($channel !~ m/^#/) {
|
|
|
|
$length = "$channel $length";
|
2015-06-10 11:27:13 +02:00
|
|
|
$length = undef if $length eq ' ';
|
2018-02-22 03:41:56 +01:00
|
|
|
$channel = exists $stuff->{admin_channel_override} ? $stuff->{admin_channel_override} : $from;
|
2015-05-27 19:45:43 +02:00
|
|
|
}
|
|
|
|
|
2018-02-22 03:41:56 +01:00
|
|
|
$channel = exists $stuff->{admin_channel_override} ? $stuff->{admin_channel_override} : $from if not defined $channel;
|
2015-05-27 19:45:43 +02:00
|
|
|
|
2016-11-17 04:14:39 +01:00
|
|
|
if ($channel !~ m/^#/) {
|
|
|
|
return "/msg $nick Please specify a channel.";
|
|
|
|
}
|
|
|
|
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $target) {
|
2015-05-27 19:45:43 +02:00
|
|
|
return "/msg $nick Usage: mute <mask> [channel [timeout (default: 24 hours)]]";
|
|
|
|
}
|
|
|
|
|
2019-07-23 17:55:24 +02:00
|
|
|
my $no_length = 0;
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $length) {
|
2019-11-25 22:57:37 +01:00
|
|
|
$length = $self->{pbot}->{registry}->get_value($channel, 'default_mute_timeout', 0, $stuff) //
|
|
|
|
$self->{pbot}->{registry}->get_value('general', 'default_mute_timeout', 0, $stuff) // 60 * 60 * 24; # 24 hours
|
2019-07-23 17:55:24 +02:00
|
|
|
$no_length = 1;
|
2015-05-27 19:45:43 +02:00
|
|
|
} else {
|
|
|
|
my $error;
|
2019-06-07 06:46:00 +02:00
|
|
|
($length, $error) = $self->{pbot}->{parsedate}->parsedate($length);
|
2015-05-27 19:45:43 +02:00
|
|
|
return $error if defined $error;
|
|
|
|
}
|
|
|
|
|
2019-07-23 17:55:24 +02:00
|
|
|
$channel = lc $channel;
|
|
|
|
$target = lc $target;
|
|
|
|
|
2015-05-27 19:45:43 +02:00
|
|
|
my $botnick = $self->{pbot}->{registry}->get_value('irc', 'botnick');
|
|
|
|
return "I don't think so." if $target =~ /^\Q$botnick\E!/i;
|
|
|
|
|
2019-06-23 19:34:41 +02:00
|
|
|
if (not $stuff->{'effective-level'} and not $self->{pbot}->{admins}->loggedin($channel, "$nick!$user\@$host")) {
|
2017-11-03 20:28:41 +01:00
|
|
|
return "/msg $nick You are not an admin for $channel.";
|
|
|
|
}
|
|
|
|
|
2019-07-23 17:55:24 +02:00
|
|
|
my $result = '';
|
|
|
|
my $sep = '';
|
2018-06-06 03:28:03 +02:00
|
|
|
my @targets = split /,/, $target;
|
2018-06-06 07:59:33 +02:00
|
|
|
my $immediately = @targets > 1 ? 0 : 1;
|
2018-06-06 03:28:03 +02:00
|
|
|
foreach my $t (@targets) {
|
2019-07-23 17:55:24 +02:00
|
|
|
my $mask = $self->{pbot}->{chanops}->nick_to_banmask($t);
|
|
|
|
|
|
|
|
if ($no_length && exists $self->{pbot}->{chanops}->{unmute_timeout}->hash->{$channel}
|
|
|
|
&& exists $self->{pbot}->{chanops}->{unmute_timeout}->hash->{$channel}->{$mask}) {
|
|
|
|
my $timeout = $self->{pbot}->{chanops}->{unmute_timeout}->hash->{$channel}->{$mask}{timeout};
|
|
|
|
my $duration = duration($timeout - gettimeofday);
|
|
|
|
|
|
|
|
$result .= "$sep$mask has $duration remaining on their $channel mute";
|
|
|
|
$sep = '; ';
|
|
|
|
} else {
|
2019-07-26 23:17:06 +02:00
|
|
|
$self->{pbot}->{chanops}->mute_user_timed("$nick!$user\@$host", undef, $t, $channel, $length, $immediately);
|
2019-07-23 17:55:24 +02:00
|
|
|
|
|
|
|
my $duration;
|
|
|
|
if ($length > 0) {
|
|
|
|
$duration = duration($length);
|
|
|
|
} else {
|
|
|
|
$duration = 'all eternity';
|
|
|
|
}
|
|
|
|
|
|
|
|
$result .= "$sep$mask muted in $channel for $duration";
|
|
|
|
$sep = '; ';
|
|
|
|
}
|
2018-06-06 07:59:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (not $immediately) {
|
|
|
|
$self->{pbot}->{chanops}->check_ban_queue;
|
2018-06-06 03:28:03 +02:00
|
|
|
}
|
2015-05-27 19:45:43 +02:00
|
|
|
|
2019-07-23 17:55:24 +02:00
|
|
|
$result = "/msg $nick $result" if $result !~ m/remaining on their/;
|
|
|
|
return $result;
|
2015-05-27 19:45:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
sub unmute_user {
|
|
|
|
my $self = shift;
|
2018-02-22 02:21:38 +01:00
|
|
|
my ($from, $nick, $user, $host, $arguments, $stuff) = @_;
|
2015-05-27 19:45:43 +02:00
|
|
|
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $from) {
|
2015-05-27 19:45:43 +02:00
|
|
|
$self->{pbot}->{logger}->log("Command missing ~from parameter!\n");
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2018-08-09 02:38:57 +02:00
|
|
|
my ($target, $channel, $immediately) = $self->{pbot}->{interpreter}->split_args($stuff->{arglist}, 3);
|
2015-05-27 19:45:43 +02:00
|
|
|
|
2018-06-06 02:55:59 +02:00
|
|
|
if (defined $target and defined $channel and $channel !~ /^#/) {
|
|
|
|
my $temp = $target;
|
|
|
|
$target = $channel;
|
|
|
|
$channel = $temp;
|
|
|
|
}
|
|
|
|
|
2019-05-28 18:19:42 +02:00
|
|
|
if (not defined $target) {
|
2018-06-06 03:28:03 +02:00
|
|
|
return "/msg $nick Usage: unmute <nick/mask> [[channel] [false value to use unban queue]]";
|
2015-05-27 19:45:43 +02:00
|
|
|
}
|
|
|
|
|
2018-02-22 03:41:56 +01:00
|
|
|
$channel = exists $stuff->{admin_channel_override} ? $stuff->{admin_channel_override} : $from if not defined $channel;
|
2018-06-06 03:28:03 +02:00
|
|
|
$immediately = 1 if not defined $immediately;
|
2015-05-27 19:45:43 +02:00
|
|
|
|
2018-06-06 03:28:03 +02:00
|
|
|
return "/msg $nick Usage for /msg: unmute <nick/mask> <channel> [false value to use unban queue]" if $channel !~ /^#/;
|
2015-05-27 19:45:43 +02:00
|
|
|
|
2019-06-23 19:34:41 +02:00
|
|
|
if (not $stuff->{'effective-level'} and not $self->{pbot}->{admins}->loggedin($channel, "$nick!$user\@$host")) {
|
2017-11-03 20:28:41 +01:00
|
|
|
return "/msg $nick You are not an admin for $channel.";
|
|
|
|
}
|
|
|
|
|
2018-06-06 03:28:03 +02:00
|
|
|
my @targets = split /,/, $target;
|
2018-06-06 07:59:33 +02:00
|
|
|
$immediately = 0 if @targets > 1;
|
2018-06-06 03:28:03 +02:00
|
|
|
|
|
|
|
foreach my $t (@targets) {
|
|
|
|
$self->{pbot}->{chanops}->unmute_user($t, $channel, $immediately);
|
|
|
|
}
|
2018-06-06 07:59:33 +02:00
|
|
|
|
|
|
|
if (@targets > 1) {
|
|
|
|
$self->{pbot}->{chanops}->check_unban_queue;
|
|
|
|
}
|
|
|
|
|
2015-05-27 19:45:43 +02:00
|
|
|
return "/msg $nick $target has been unmuted in $channel.";
|
|
|
|
}
|
|
|
|
|
2010-03-23 19:24:02 +01:00
|
|
|
sub kick_user {
|
|
|
|
my $self = shift;
|
2018-02-22 02:21:38 +01:00
|
|
|
my ($from, $nick, $user, $host, $arguments, $stuff) = @_;
|
2010-03-23 19:24:02 +01:00
|
|
|
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not defined $from) {
|
2014-05-18 22:09:05 +02:00
|
|
|
$self->{pbot}->{logger}->log("Command missing ~from parameter!\n");
|
2010-03-23 19:24:02 +01:00
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2016-08-29 07:36:46 +02:00
|
|
|
my ($channel, $victim, $reason);
|
|
|
|
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not $from =~ /^#/) {
|
2016-08-29 07:36:46 +02:00
|
|
|
# used in private message
|
2017-11-10 04:26:05 +01:00
|
|
|
if (not $arguments =~ s/^(^#\S+) (\S+)\s*//) {
|
2016-08-29 07:36:46 +02:00
|
|
|
return "/msg $nick Usage from private message: kick <channel> <nick> [reason]";
|
|
|
|
}
|
|
|
|
($channel, $victim) = ($1, $2);
|
|
|
|
} else {
|
|
|
|
# used in channel
|
2017-11-10 04:26:05 +01:00
|
|
|
if ($arguments =~ s/^(#\S+)\s+(\S+)\s*//) {
|
|
|
|
($channel, $victim) = ($1, $2);
|
|
|
|
} elsif ($arguments =~ s/^(\S+)\s*//) {
|
2018-02-22 03:41:56 +01:00
|
|
|
($victim, $channel) = ($1, exists $stuff->{admin_channel_override} ? $stuff->{admin_channel_override} : $from);
|
2017-11-10 04:26:05 +01:00
|
|
|
} else {
|
|
|
|
return "/msg $nick Usage: kick [channel] <nick> [reason]";
|
2014-04-19 12:35:27 +02:00
|
|
|
}
|
2010-03-23 19:24:02 +01:00
|
|
|
}
|
2014-04-19 12:35:27 +02:00
|
|
|
|
2016-08-29 07:36:46 +02:00
|
|
|
$reason = $arguments;
|
|
|
|
|
2017-11-10 04:26:05 +01:00
|
|
|
# If the user is too stupid to remember the order of the arguments,
|
|
|
|
# we can help them out by seeing if they put the channel in the reason.
|
|
|
|
if ($reason =~ s/^(#\S+)\s*//) {
|
|
|
|
$channel = $1;
|
|
|
|
}
|
|
|
|
|
2019-06-23 19:34:41 +02:00
|
|
|
if (not $stuff->{'effective-level'} and not $self->{pbot}->{admins}->loggedin($channel, "$nick!$user\@$host")) {
|
2017-11-11 21:59:27 +01:00
|
|
|
return "/msg $nick You are not an admin for $channel.";
|
|
|
|
}
|
|
|
|
|
|
|
|
my @insults;
|
2016-08-29 07:36:46 +02:00
|
|
|
if (not length $reason) {
|
|
|
|
if (open my $fh, '<', $self->{pbot}->{registry}->get_value('general', 'module_dir') . '/insults.txt') {
|
2017-11-11 21:59:27 +01:00
|
|
|
@insults = <$fh>;
|
2016-08-29 07:36:46 +02:00
|
|
|
close $fh;
|
|
|
|
$reason = $insults[rand @insults];
|
2017-11-11 21:59:27 +01:00
|
|
|
$reason =~ s/\s+$//;
|
2016-08-29 07:36:46 +02:00
|
|
|
} else {
|
|
|
|
$reason = 'Bye!';
|
|
|
|
}
|
2010-03-23 19:24:02 +01:00
|
|
|
}
|
2014-04-19 12:35:27 +02:00
|
|
|
|
2017-11-11 21:59:27 +01:00
|
|
|
my @nicks = split /,/, $victim;
|
|
|
|
foreach my $n (@nicks) {
|
2019-12-31 03:17:35 +01:00
|
|
|
if ($n =~ m/\*/) {
|
|
|
|
# wildcard used; find all matching nicks; test against whitelist, etc
|
|
|
|
my $q_target = lc quotemeta $n;
|
|
|
|
$q_target =~ s/\\\*/.*/g;
|
|
|
|
$channel = lc $channel;
|
|
|
|
|
|
|
|
if (not exists $self->{pbot}->{nicklist}->{nicklist}->{$channel}) {
|
|
|
|
return "I have no nicklist for channel $channel; cannot use wildcard.";
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach my $nl (keys %{$self->{pbot}->{nicklist}->{nicklist}->{$channel}}) {
|
|
|
|
if ($nl =~ m/^$q_target$/) {
|
|
|
|
my $nick_data = $self->{pbot}->{nicklist}->{nicklist}->{$channel}->{$nl};
|
|
|
|
|
|
|
|
next if $nick_data->{nick} eq $self->{pbot}->{registry}->get_value('irc', 'botnick');
|
|
|
|
next if $self->{pbot}->{antiflood}->whitelisted($channel, $nick_data->{hostmask});
|
|
|
|
next if $self->{pbot}->{admins}->loggedin($channel, $nick_data->{hostmask});
|
|
|
|
|
|
|
|
$self->{pbot}->{chanops}->add_op_command($channel, "kick $channel $nl $reason");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
# no wildcard used, explicit kick
|
|
|
|
$self->{pbot}->{chanops}->add_op_command($channel, "kick $channel $n $reason");
|
|
|
|
}
|
|
|
|
|
|
|
|
# randomize next kick reason
|
2017-11-11 21:59:27 +01:00
|
|
|
if (@insults) {
|
|
|
|
$reason = $insults[rand @insults];
|
|
|
|
$reason =~ s/\s+$//;
|
|
|
|
}
|
2017-11-03 20:28:41 +01:00
|
|
|
}
|
|
|
|
|
2016-08-29 07:36:46 +02:00
|
|
|
$self->{pbot}->{chanops}->gain_ops($channel);
|
2017-09-19 06:24:30 +02:00
|
|
|
return "";
|
2010-03-23 19:24:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
1;
|