my ($self, $event) = @_;
my $msg = ($event->args)[0];
my $sock = ($event->to)[0];
- my $nick = $event->nick();
-
- $userHandle = ""; # reset.
-
- # who is set to bot's name, why?
+ my $nick = lc $event->nick();
- if (!exists $nuh{$who}) {
- &DEBUG("chat: nuh{$who} (nick => $nick) doesn't exist; trying WHOIS .");
- $self->whois($who);
+ if (!exists $nuh{$nick}) {
+ &DEBUG("chat: nuh{$nick} doesn't exist; trying WHOIS .");
$self->whois($nick);
return;
- } else {
- $message = $msg;
- $who = lc $nick;
- $orig{who} = $nick;
- $orig{message} = $msg;
- $nuh = $nuh{$who};
- $uh = (split /\!/, $nuh)[1];
- $addressed = 1;
- $msgType = 'chat';
}
+ ### set vars that would have been set in hookMsg.
+ $userHandle = ""; # reset.
+ $who = lc $nick;
+ $message = $msg;
+ $orig{who} = $nick;
+ $orig{message} = $msg;
+ $nuh = $nuh{$who};
+ $uh = (split /\!/, $nuh)[1];
+ $addressed = 1;
+ $msgType = 'chat';
+
if (!exists $dcc{'CHATvrfy'}{$nick}) {
$userHandle = &verifyUser($who, $nuh);
my $crypto = $users{$userHandle}{PASS};
### TODO: prevent users without CRYPT chatting.
if (!defined $crypto) {
- &DEBUG("chat: no pass required.");
-### $success++;
+ &DEBUG("todo: dcc close chat");
+ &msg($who, "nope, no guest logins allowed...");
+ return;
+ }
- } elsif (&ckpasswd($msg, $crypto)) {
+ if (&ckpasswd($msg, $crypto)) {
# stolen from eggdrop.
$self->privmsg($sock, "Connected to $ident");
$self->privmsg($sock, "Commands start with '.' (like '.quit' or '.help')");
# update IRCStats.
$ircstats{'ConnectTime'} = time();
$ircstats{'ConnectCount'}++;
+ $ircstats{'OffTime'} += time() - $ircstats{'DisconnectTime'}
+ if (defined $ircstats{'DisconnectTime'});
+
+ # first time run.
+ if (!exists $users{_default}) {
+ &status("First time run... adding _default user.");
+ $users{_default}{FLAGS} = "mrt";
+ $users{_default}{HOSTS} = "*!*@*";
+ }
+
+ if (scalar keys %users < 2) {
+ &status("Ok... now /msg $ident PASS <pass> to get master access through DCC CHAT.");
+ }
+ # end of first time run.
if (&IsChanConf("wingate")) {
my $file = "$bot_base_dir/$param{'ircUser'}.wingate";
sub on_dcc {
my ($self, $event) = @_;
my $type = uc( ($event->args)[1] );
- my $nick = $event->nick();
+ my $nick = lc $event->nick();
# pity Net::IRC doesn't store nuh. Here's a hack :)
- $self->whois($nick);
- $nuh{lc $nick} = "GETTING-NOW"; # trying.
+ if (!exists $nuh{lc $nick}) {
+ $self->whois($nick);
+ $nuh{$nick} = "GETTING-NOW"; # trying.
+ }
$type ||= "???";
- &DEBUG("on_dcc: nuh => GETTING-NOW.");
if ($type eq 'SEND') { # GET for us.
# incoming DCC SEND. we're receiving a file.
sub on_dcc_open {
my ($self, $event) = @_;
my $type = uc( ($event->args)[0] );
- my $nick = $event->nick();
+ my $nick = lc $event->nick();
my $sock = ($event->to)[0];
$msgType = 'chat';
# really custom sub to get NUH since Net::IRC doesn't appear to support
# it.
sub on_dcc_open_chat {
- my(undef, $nick,$sock) = @_;
+ my(undef, $nick, $sock) = @_;
- &DEBUG("nuh{$nick} => $nuh{$nick}.");
- if ($nuh{$nick} =~ /^(\S+)(\d+)$/) {
- my $i = $2;
- $i++;
- $nuh{$nick} = $1.$i;
+ if ($nuh{$nick} eq "GETTING-NOW") {
&DEBUG("getting nuh for $nick failed. FIXME.");
return;
}
$ircstats{'DisconnectTime'} = time();
$ircstats{'DisconnectReason'} = $what;
$ircstats{'DisconnectCount'}++;
+ $ircstats{'TotalTime'} += time() - $ircstats{'ConnectTime'};
# clear any variables on reconnection.
$nickserv = 0;
my ($self, $event) = @_;
my $chan = ($event->args)[1];
- if (exists $jointime{$chan}) {
- my $delta_time = sprintf("%.03f", &gettimeofday() - $jointime{$chan});
+ if (exists $cache{jointime}{$chan}) {
+ my $delta_time = sprintf("%.03f", &gettimeofday() - $cache{jointime}{$chan});
$delta_time = 0 if ($delta_time < 0);
&status("$b_blue$chan$ob: sync in ${delta_time}s.");
my $txt;
my @array;
foreach ("o","v","") {
- my $count = scalar(keys %{$channels{$chan}{$_}});
+ my $count = scalar(keys %{ $channels{$chan}{$_} });
next unless ($count);
$txt = "total" if ($_ eq "");
if (scalar @joinchan) { # remaining channels to join.
&joinNextChan();
+ } else {
+ &DEBUG("running ircCheck to get chanserv ops.");
+ &ircCheck();
}
return unless (&IsChanConf("chanServ_ops") > 0);
}
if ($netsplit and !$netsplittime) {
- &status("ok.... re-running chanlimitCheck in 60.");
+ &DEBUG("on_join: ok.... re-running chanlimitCheck in 60.");
$conn->schedule(60, sub {
&chanlimitCheck();
$netsplittime = undef;
&status(">>> join/$b_blue$chan$ob $b_cyan$who$ob $b_yellow($ob$user\@$host$b_yellow)$ob$netsplitstr");
$channels{$chan}{''}{$who}++;
- $nuh{lc $who} = $who."!".$user."\@".$host unless (exists $nuh{lc $who});
+ $nuh = $who."!".$user."\@".$host;
+ $nuh{lc $who} = $nuh unless (exists $nuh{lc $who});
+
+ ### on-join bans.
+ my @bans;
+ push(@bans, keys %{ $bans{$chan} }) if (exists $bans{$chan});
+ push(@bans, keys %{ $bans{"*"} }) if (exists $bans{"*"});
+ foreach (@bans) {
+ my $ban = $_;
+ s/\?/./g;
+ s/\*/\\S*/g;
+ my $mask = $_;
+ next unless ($nuh =~ /^$mask$/i);
+
+ ### TODO: check $channels{$chan}{'b'} if ban already exists.
+ foreach (keys %{ $channels{$chan}{'b'} }) {
+ &DEBUG(" bans_on_chan($chan) => $_");
+ }
- ### on-join ban. (TODO: kick)
- if (exists $bans{$chan}) {
- ### TODO: need to do $chan and _default
- foreach (keys %{ $bans{$chan} }) {
- s/\*/\\S*/g;
- next unless /^\Q$nuh\E$/i;
+ my $reason = "no reason";
+ foreach ($chan, "*") {
+ next unless (exists $bans{$_});
+ next unless (exists $bans{$_}{$ban});
- foreach (keys %{ $channels{$chan}{'b'} }) {
- &DEBUG(" bans_on_chan($chan) => $_");
- }
+ my @array = @{ $bans{$_}{$ban} };
- ### TODO: check $channels{$chan}{'b'} if ban already exists.
- &ban( "*!*@".&makeHostMask($host), $chan);
+ $reason = $array[4] if ($array[4]);
last;
}
+
+ &ban($ban, $chan);
+ &kick($who, $chan, $reason);
+
+ last;
}
### ROOTWARN:
$user =~ /^r(oo|ew|00)t$/i &&
$channels{$chan}{'o'}{$ident});
+ ### NEWS:
+ if (&IsChanConf("news") && &IsChanConf("newsKeepRead")) {
+ # todo: what if it hasn't been loaded?
+ &News::latest($chan);
+ }
+
+ ### chanlimit check.
+ &chanLimitVerify($chan);
+
# used to determine sync time.
if ($who =~ /^$ident$/i) {
- if (defined( my $whojoin = $joinverb{$chan} )) {
+ if (defined( my $whojoin = $cache{join}{$chan} )) {
&msg($chan, "Okay, I'm here. (courtesy of $whojoin)");
- delete $joinverb{$chan};
+ delete $cache{join}{$chan};
}
### TODO: move this to &joinchan()?
- $jointime{$chan} = &gettimeofday();
+ $cache{jointime}{$chan} = &gettimeofday();
rawout("WHO $chan");
} else {
### TODO: this may go wild on a netjoin :)
($user,$host) = split(/\@/, $event->userhost);
$uh = $event->userhost();
$nuh = $nick."!".$uh;
+ $msgtime = time();
+
+ if ($nick eq $ident) { # hopefully ourselves.
+ if ($msg eq "TEST") {
+ &status("IRCTEST: Yes, we're alive.");
+ delete $cache{connect};
+ return;
+ }
+ }
&hookMsg('private', undef, $nick, $msg);
}
if ($nick =~ /^NickServ$/i) { # nickserv.
&status("NickServ: <== '$args'");
- if ($args =~ /^This nickname is registered/i) {
+ my $check = 0;
+ $check++ if ($args =~ /^This nickname is registered/i);
+ $check++ if ($args =~ /nickname.*owned/i);
+
+ if ($check) {
&status("nickserv told us to register; doing it.");
if (&IsParam("nickServ_pass")) {
&status("NickServ: ==> Identifying.");
}
}
-
$msgtime = time();
$lastWho{$chan} = $nick;
### TODO: use $nick or lc $nick?
my $nick = $event->nick();
my $reason = ($event->args)[0];
+ my $count = 0;
foreach (keys %channels) {
# fixes inconsistent chanstats bug #1.
- next unless (&IsNickInChan($nick,$_));
+ if (!&IsNickInChan($nick,$_)) {
+ $count++;
+ next;
+ }
$chanstats{$_}{'SignOff'}++;
}
+
+ if ($count == scalar keys %channels) {
+ &DEBUG("on_quit: nick $nick was not found in any chan.");
+ }
+
&DeleteUserInfo($nick, keys %channels);
+
if (exists $nuh{lc $nick}) {
delete $nuh{lc $nick};
} else {
if ($reason=~/^($mask{host})\s($mask{host})$/) { # netsplit.
$reason = "NETSPLIT: $1 <=> $2";
+ if (&ChanConfList("chanlimitcheck") and !scalar keys %netsplit) {
+ &DEBUG("on_quit: netsplit detected; disabling chan limit.");
+ &rawout("MODE $chan -l");
+ }
+
$netsplit{lc $nick} = time();
if (!exists $netsplitservers{$1}{$2}) {
&status("netsplit detected between $1 and $2.");
my $nick = $event->nick();
my($me,$chan,$why) = $event->args();
+ ### TODO: incomplete.
### .* wait (\d+) second/) {
- &status("X1 $msg");
+ &status("on_ttf: X1 $msg") if (defined $msg);
my $sleep = 5;
&status("going to sleep for $sleep...");
sleep $sleep;
$nuh{lc $args[1]} = $args[1]."!".$args[2]."\@".$args[3];
}
-#######################################################################
-####### IRC HOOK HELPERS IRC HOOK HELPERS IRC HOOK HELPERS ########
-#######################################################################
-
-#####
-# Usage: &hookMode($chan, $modes, @targets);
-sub hookMode {
- my ($chan, $modes, @targets) = @_;
- my $parity = 0;
-
- $chan = lc $chan; # !!!.
-
- my $mode;
- foreach $mode (split(//, $modes)) {
- # sign.
- if ($mode =~ /[-+]/) {
- $parity = 1 if ($mode eq "+");
- $parity = 0 if ($mode eq "-");
- next;
- }
-
- # mode with target.
- if ($mode =~ /[bklov]/) {
- my $target = shift @targets;
-
- if ($parity) {
- $chanstats{$chan}{'Op'}++ if ($mode eq "o");
- $chanstats{$chan}{'Ban'}++ if ($mode eq "b");
- } else {
- $chanstats{$chan}{'Deop'}++ if ($mode eq "o");
- $chanstats{$chan}{'Unban'}++ if ($mode eq "b");
- }
-
- # modes w/ target affecting nick => cache it.
- if ($mode =~ /[bov]/) {
- $channels{$chan}{$mode}{$target}++ if $parity;
- delete $channels{$chan}{$mode}{$target} if !$parity;
- }
-
- if ($mode =~ /[l]/) {
- $channels{$chan}{$mode} = $target if $parity;
- delete $channels{$chan}{$mode} if !$parity;
- }
- }
-
- # important channel modes, targetless.
- if ($mode =~ /[mt]/) {
- $channels{$chan}{$mode}++ if $parity;
- delete $channels{$chan}{$mode} if !$parity;
- }
- }
-}
-
-sub hookMsg {
- ($msgType, $chan, $who, $message) = @_;
- my $skipmessage = 0;
- $addressed = 0;
- $addressedother = 0;
- $orig{message} = $message;
- $orig{who} = $who;
- $addrchar = 0;
-
- $message =~ s/[\cA-\c_]//ig; # strip control characters
- $message =~ s/^\s+//; # initial whitespaces.
- $who =~ tr/A-Z/a-z/; # lowercase.
-
- &showProc();
-
- # addressing.
- if ($msgType =~ /private/) {
- # private messages.
- $addressed = 1;
- } else {
- # public messages.
- # addressing revamped by the xk.
- ### below needs to be fixed...
- if (&IsParam("addressCharacter")) {
- if ($message =~ s/^$param{'addressCharacter'}//) {
- $addrchar = 1;
- $addressed = 1;
- }
- }
-
- if ($message =~ /^($mask{nick})([\;\:\>\, ]+) */) {
- my $newmessage = $';
- if ($1 =~ /^\Q$ident\E$/i) {
- $message = $newmessage;
- $addressed = 1;
- } else {
- # ignore messages addressed to other people or unaddressed.
- $skipmessage++ if ($2 ne "" and $2 !~ /^ /);
- }
- }
- }
-
- # Determine floodwho.
- my $c = "_default";
- if ($msgType =~ /public/i) { # public.
- $floodwho = $c = lc $chan;
- } elsif ($msgType =~ /private/i) { # private.
- $floodwho = lc $who;
- } else { # dcc?
- &DEBUG("FIXME: floodwho = ???");
- }
-
- my $val = &getChanConfDefault("floodRepeat", "2:10", $c);
- my ($count, $interval) = split /:/, $val;
-
- # flood repeat protection.
- if ($addressed) {
- my $time = $flood{$floodwho}{$message};
-
- if (defined $time and (time - $time < $interval)) {
- ### public != personal who so the below is kind of pointless.
- my @who;
- foreach (keys %flood) {
- next if (/^\Q$floodwho\E$/);
- next if (defined $chan and /^\Q$chan\E$/);
-
- push(@who, grep /^\Q$message\E$/i, keys %{$flood{$_}});
- }
-
- if (scalar @who) {
- &msg($who, "you already said what ".
- join(' ', @who)." have said.");
- } else {
- &msg($who,"Someone already said that ". (time - $time) ." seconds ago" );
- }
-
- ### TODO: delete old floodwarn{} keys.
- my $floodwarn = 0;
- if (!exists $floodwarn{$floodwho}) {
- $floodwarn++;
- } else {
- $floodwarn++ if (time() - $floodwarn{$floodwho} > $interval);
- }
-
- if ($floodwarn) {
- &status("FLOOD repetition detected from $floodwho.");
- $floodwarn{$floodwho} = time();
- }
-
- return;
- }
-
- if ($addrchar) {
- &status("$b_cyan$who$ob is short-addressing me");
- } elsif ($msgType eq "private") { # private.
- &status("$b_cyan$who$ob is /msg'ing me");
- } else { # public?
- &status("$b_cyan$who$ob is addressing me");
- }
-
- $flood{$floodwho}{$message} = time();
- }
-
- $val = &getChanConfDefault("floodMessages", "5:30", $c);
- ($count, $interval) = split /:/, $val;
-
- # flood overflow protection.
- if ($addressed) {
- foreach (keys %{$flood{$floodwho}}) {
- next unless (time() - $flood{$floodwho}{$_} > $interval);
- delete $flood{$floodwho}{$_};
- }
-
- my $i = scalar keys %{$flood{$floodwho}};
- if ($i > $count) {
- &msg($who,"overflow of messages ($i > $count)");
- &status("FLOOD overflow detected from $floodwho; ignoring");
-
- my $expire = $param{'ignoreAutoExpire'} || 5;
- &ignoreAdd("*!$uh", $chan, $expire, "flood overflow auto-detected.");
- return;
- }
-
- $flood{$floodwho}{$message} = time();
- }
-
- if ($msgType =~ /public/i) { # public.
- $talkchannel = $chan;
- &status("<$orig{who}/$chan> $orig{message}");
- } elsif ($msgType =~ /private/i) { # private.
- &status("[$orig{who}] $orig{message}");
- $talkchannel = undef;
- $chan = "_default";
- } else {
- &DEBUG("unknown msgType => $msgType.");
- }
-
- if ((!$skipmessage or &IsChanConf("seenStoreAll")) and
- &IsChanConf("seen") and
- $msgType =~ /public/
- ) {
- $seencache{$who}{'time'} = time();
- $seencache{$who}{'nick'} = $orig{who};
- $seencache{$who}{'host'} = $uh;
- $seencache{$who}{'chan'} = $talkchannel;
- $seencache{$who}{'msg'} = $orig{message};
- $seencache{$who}{'msgcount'}++;
- }
-
- return if ($skipmessage);
- return unless (&IsParam("minVolunteerLength") or $addressed);
-
- local $ignore = 0;
- if (exists $ignore{lc $chan}) {
- foreach (keys %{ $ignore{lc $chan} }) {
- s/\*/\\S*/g;
-
- next unless ($nuh =~ /^\Q$_\E$/i);
- $ignore++;
- last;
- }
- }
-
- if (defined $nuh) {
- if (!defined $userHandle) {
- &DEBUG("line 1074: need verifyUser?");
- &verifyUser($who, $nuh);
- }
- } else {
- &DEBUG("hookMsg: 'nuh' not defined?");
- }
-
-### For extra debugging purposes...
- if ($_ = &process()) {
-# &DEBUG("IrcHooks: process returned '$_'.");
- }
-
- return;
-}
-
1;