#!/usr/bonsaitools/bin/perl5 -w # -*- Mode: perl; indent-tabs-mode: nil -*- # # The contents of this file are subject to the Mozilla Public License # Version 1.0 (the "License"); you may not use this file except in # compliance with the License. You may obtain a copy of the License at # http://www.mozilla.org/MPL/ # # Software distributed under the License is distributed on an "AS IS" # basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the # License for the specific language governing rights and limitations # under the License. # # The Original Code is the Bugzilla Bug Tracking System. # # The Initial Developer of the Original Code is Netscape Communications # Corporation. Portions created by Netscape are Copyright (C) 1998 # Netscape Communications Corporation. All Rights Reserved. # # Contributor(s): Harrison Page # Terry Weissman # # mozbot.pl harrison@netscape.com 10/14/98 # "irc bot for the gang on #mozilla" # # features: reports tinderbox status upon request. # remembers urls. tells you the phase of the moon. # grabs mozillaZine headlines. bot will auto-op # based on nick and remote host. # # hack on me! required reading: # # Net::IRC web page: # http://netirc.betterbox.net/ # (free software) # or get it from CPAN @ http://www.perl.com/CPAN # # RFC 1459 (Internet Relay Chat Protocol): # http://sunsite.cnlab-switch.ch/ftp/doc/standard/rfc/14xx/1459 $SIG{'INT'} = 'killed'; $SIG{'KILL'} = 'killed'; $SIG{'TERM'} = 'killed'; use strict; use diagnostics; use lib "."; use Net::IRC; use LWP::Simple; use Tinderbox; use Carp; $|++; my $VERSION = "1.7"; # keep me in sync with the mozilla.org cvs repository my $debug = 1; # debug output also includes warnings, errors my %cmds = ( "about" => \&bot_about, "hi" => \&bot_hi, "moon" => \&bot_moon, "up" => \&bot_up, "trees" => \&bot_tinderbox, "url" => \&bot_urls, ); @::origargv = @ARGV; my $server = shift; my $port = shift; my $nick = shift; my $channel = shift; $server = $server || "irc.mozilla.org"; $port = $port || "6667"; $nick = $nick || "mozbot"; $channel = $channel || "#mozilla"; &debug ("mozbot $VERSION starting up"); &create_pid_file; # read admin list my %admins = ( "sar" => "netscape.com", "terry" => "netscape.com", "harrison" => "(censor.com|netscape.com)"); my $adminf = ".mozbot-admins"; &fetch_admin_conf (\%admins); my $uptime = 0; $::moon = "./moon"; $::moon = (-f $::moon) ? $::moon : ""; delete $cmds{'moon'} if (! $::moon); my $phase; my $last_moon = 0; # leave @trees empty if you don't want tinderbox details my @trees = qw (Mozilla Mozilla-External raptor); my $trees; my $status; my $last_tree; my %broken; my @urls; my $greet = 0; my @greetings = ( "g'day", "bonjour", "guten tag", "moshi, moshi", "hello", "hola", "hi", "buon giorno", "aloha", "hey", "'sup", "lo", "howdy", "saluton", "hei", "hallo" ); # leave $mozillazine undef'd if you don't want headlines # checked every eight hours and available via # "mozilla, mozillazine" (or "zine" or "mz") my $mozillazine = "http://www.mozillazine.org/home.html"; my @headlines; my $irc = new Net::IRC or confess "$0: duh?"; my $bot = $irc->newconn ( Server => $server, Port => $port, Nick => $nick, Ircname => "mozilla.org bot/thing $VERSION", Username => $nick, ) or die "$0: can't connect to $server, port $port"; &debug ("adding global handlers"); $bot->add_global_handler ([ 251,252,253,254,302,255 ], \&on_startup); $bot->add_global_handler (376, \&on_connect); $bot->add_global_handler (433, \&on_nick_taken); $bot->add_global_handler ([ 'disconnect', 'kill', 474, 465 ], \&on_boot); &debug ("adding more handlers"); $bot->add_handler ('msg', \&on_msg); $bot->add_handler ('public', \&on_public); $bot->add_handler ('join', \&on_join); &debug ("scheduling stuff"); $bot->schedule (0, \&tinderbox); $bot->schedule (0, \&checksourcechange); $bot->schedule (0, \&mozillazine); &debug ("connecting to $server $port as $nick on $channel"); $irc->start; # and done. ################################ # Net::IRC handler subroutines # ################################ sub on_startup { my ($self, $event) = @_; my (@args) = ($event->args); shift (@args); &debug ("@args\n"); } sub on_connect { my $self = shift; &debug ("startup took " . (time - $^T) . " seconds"); $self->join ($channel); $uptime = time; } # on_nick_taken: or do something smarter sub on_nick_taken { die "hey! somebody took my nick!"; } # on_msg: private message received via /msg sub on_msg { my ($self, $event) = @_; my ($nick) = $event->nick; my ($arg) = $event->args; my ($cmd, $rest) = split ' ', $arg; # hoo dee boo if (exists $admins{$nick}) { if ((my ($who, $where) = $arg =~ /^bless\s+(\S+)\s+(\S+)$/i) or $arg =~ /^bless/i) { if (! $who or ! $where) { $self->privmsg ($nick, "usage: bless [ user ] [ host ] " . "(example: bless marca netscape.com)"); return; } $admins{$who} = $where; &debug ("$nick blessed $who ($where)"); &store_admin_conf (\%admins); $self->privmsg ($nick, "mozbot admins: " . join ' ', (sort keys %admins)); return; } elsif ($arg =~ /^unbless (\S+)$/i && exists ($admins{$1})) { delete $admins{$1}; &debug ("$nick unblessed $1"); &store_admin_conf (\%admins); $self->privmsg ($nick, "mozbot admins: " . join ' ', (sort keys %admins)); return; } elsif ($arg eq "shutdown yes") { &debug ("forced shutdown from $nick"); $::dontQuitOnSignal++; $self->quit ("$nick told me to shutdown"); exit (0); } elsif ($arg =~ /^shutdown/) { $self->privmsg ($nick, "usage: shutdown yes"); return; } elsif ($arg =~ /^say (.*)/) { my $text = $1; if ($text =~ m@^/me (.*)@) { $self->me($channel, $1); } else { $self->privmsg ($channel, $text); } return; } elsif ($arg =~ /^list$/) { foreach (sort keys %admins) { $self->privmsg ($nick, "$_ $admins{$_}"); } return; } elsif ($arg =~ /^help/) { $self->privmsg($nick, "mozbot admin commands: bless help list say shutdown unbless"); return; } } else { &debug ("msg from $nick: $arg") if ($arg); } if ($cmd ne "help") { foreach (sort keys %cmds) { if ($_ =~ /^$cmd/i) # matched a command { &debug ("command: $_"); # see %cmds hash defined at the top my $msg = &{$cmds{$_}} ($nick, $cmd, $rest); # bot functions return either a reference to an # array or just a scalar if (ref ($msg) eq "ARRAY") { foreach (@$msg) { $self->privmsg ($nick, $_); } } else { $self->privmsg ($nick, $msg); } return; } } } # print help message $self->privmsg ($nick, "unknown command \"$cmd\". " . &bot_hi ($nick)); } sub on_public { my ($self, $event) = @_; my ($to) = $event->to; my ($arg) = $event->args; my ($nick, $me) = ($event->nick, $self->nick); if (my ($cmd, $rest) = $arg =~ /^$me[:,]?\s+(\S+)(?:\s+(.*))?$/i) { # if this gets any larger, break this # out into a command/subroutine hash if ($cmd =~ /tree/i) { my $t = &bot_tinderbox (undef, undef, undef, # it's all about making perl happy (defined $rest && $rest eq "all") ? 0 : 1); foreach (@$t) { next if ($_ eq "."); $self->privmsg ($channel, "$_"); } } elsif ($cmd =~ /^(zine|mozillazine|mz)/) { $self->privmsg ($channel, "Headlines from mozillaZine (http://www.mozillazine.org/)"); if ($#headlines == -1) { $self->privmsg ($channel, "- sorry, no headlines -"); } else { foreach (@headlines) { $self->privmsg ($channel, $_); } } } elsif ($cmd =~ /^(hi|hello|lo|sup)/) { $self->privmsg ($channel, $greetings[$greet++] . " $nick"); $greet = 0 if ($greet > $#greetings); } } # catch urls, stick them in a list for mozbot's url command if ($arg =~ /(http|ftp|gopher):/i && $nick ne $me) { push @urls, "$arg (" . &logdate() . ")"; while ($#urls > 10) { shift @urls; } } } sub on_join { my ($self, $event) = @_; my ($channel) = ($event->to)[0]; my $nick = $event->nick; my $userhost = $event->userhost; # auto-op if user is a mozbot admin and coming in from # the right host if (exists $admins{$nick} && $userhost =~ /$admins{$nick}$/i) { $self->mode ($channel, "+o", $nick); &debug ("auto-op for $nick on $channel"); } } $::dontQuitOnSignal = 0; sub on_boot { if (!$::dontQuitOnSignal) { die "$0: disconnected from network"; } } ################ # bot commands # ################ # bot_about: it's either an about box or the # address of the guy to blame when the bot # breaks sub bot_about { return "i am mozbot version $VERSION. hack on me! " . "harrison\@netscape.com 10/16/98. " . "connected to $server since " . &bot_up . ". " . "see http://cvs-mirror.mozilla.org/webtools/bonsai/cvsquery.cgi?branch=HEAD&file=mozilla/webtools/mozbot/&date=week " . "for a changelog."; } # bot_hi: list commands, also default function for # unknown commands sub bot_hi { my ($nick, $cmd, $rest) = @_; carp "bot_hi wants nick and optional first word of line, rest of line" unless $nick; my @cmds = sort keys %cmds; return "i am mozbot. i know these commands: @cmds"; } # bot_moon: goodnight moon sub bot_moon { return "- no moon -" if (! defined $::moon); return $phase if ($phase && (time - $last_moon > (60 * 60 * 24))); # we only want to run this once/day $phase = `$::moon`; $last_moon = time; return $phase; } # bot_up: report uptime sub bot_up { return &logdate ($uptime) . " (" . &days ($uptime) . ")"; } # bot_urls: show last ten urls caught by mozbot sub bot_urls { return ($#urls == -1 ? "- mozbot has seen no URLs yet -" : \@urls); } # show tinderbox status # # this is a messy little function but it works. sub bot_tinderbox { my ($nick, $cmd, $rest, $terse) = @_; my $bustage; my $buf; my @buf; my @tree; # user can supply a list of trees separated # by whitespace, default is all trees push @tree, $rest ? (split /\s+/, $rest) : @trees; # loop through requested trees push @buf, "Tinderbox status from http://cvs-mirror.mozilla.org/webtools/tinderbox/showbuilds.cgi"; foreach my $t (@tree) { $bustage = 0; $buf = "$t " . ($$status{$t} ? "<$$status{$t}> " : "") . ": "; # politely report failures if (! exists $$trees{$t}) { $buf .= "unknown tree \"$t\", trees include @trees. "; } else { foreach my $e (sort keys %{$$trees{$t}}) { next if ($terse && $$trees{$t}{$e} ne "Horked"); $buf .= "[$e: $$trees{$t}{$e}] "; $bustage++; } } $buf .= "- no known bustage -" if (! $bustage); push @buf, $buf, "."; } $buf = $buf || "something broke. report a bug here: " . "http://cvs-mirror.mozilla.org/webtools/bugzilla/enter_bug.cgi " . "with product of Webtools and component set to Mozbot"; push @buf, "last update: " . &logdate ($last_tree) . " (" . &days ($last_tree) . " ago)"; return \@buf; } ############# # utilities # ############# sub debug { return if (! $debug); foreach (@_) { chomp; print &logdate() . " $_ [$$]\n"; } } # logdate: return nice looking date (10/16/98 18:29) sub logdate { my $t = shift; $t = time unless ($t); my ($sec,$min,$hour,$mday,$mon,$year) = localtime ($t); return sprintf ("%02d/%02d/%02d %02d:%02d", $mon + 1, $mday, $year, $hour, $min); } # days: how long ago was that? sub days { my ($then) = shift; my $seconds = time - $then; my $minutes = int ($seconds / 60); my $hours = int ($minutes / 60); my $days = int ($hours / 24); if ($seconds < 60) { return (sprintf "%d second%s", $seconds, $seconds == 1 ? "" : "s"); } elsif ($minutes < 60) { return (sprintf "%d minute%s", $minutes, $minutes == 1 ? "" : "s"); } elsif ($hours < 24) { return (sprintf "%d hour%s", $hours, $hours == 1 ? "" : "s"); } else { return (sprintf "%d day%s", $days, $days == 1 ? "" : "s"); } } # signal handler sub killed { confess "i have received a signal of some manner. good night.\n\n"; } # write admin list sub store_admin_conf { my $admins = shift; my $when = localtime (time) . " by $$"; if (open ADMINS, ">$adminf") { print ADMINS <) { chomp; next if ($_ =~ /^#/ or ! $_); my ($user, $host) = split /\s+/, $_; $$admins{$user} = $host; } &debug ("admins: " . keys %$admins); } else { &debug ("&fetch_admin_conf $adminf: $!"); } close ADMINS; } # create a pid file if we can sub create_pid_file { my $pid = ".mozbot-pid"; if (open PID, ">$pid") { print PID "$$\n"; close PID; } else { &debug ("warning: problem creating pid file: $pid, $!"); } } # fetches headlines from mozillaZine # # this should be a more general feature, to grab # content. if you feel like it, implement a # grabber for slashdot headlines: # # http://slashdot.org/ultramode.txt sub mozillazine { &debug ("fetching mozillazine headers"); return if (! defined $mozillazine); my $output = get $mozillazine; return if (! $output); my @mz = split /\n/, $output; @headlines = (); foreach (@mz) { if (my ($h) = $_ =~ /COLOR="#FEFEFE">([^<>]+)/) { $h =~ s/ //g; push @headlines, $h; } } $bot->schedule (60 * 60 * 8, \&mozillazine); } # fetch tinderbox details sub tinderbox { &debug ("fetching tinderbox status"); my ($newtrees, $newstatus) = Tinderbox::status (\@trees); if (! $newtrees) { $bot->schedule (90, \&tinderbox); &debug ("hmm, couldn't get tinderbox status"); return; } $last_tree = time; if (defined $status) { foreach my $s (keys %$newstatus) { if (defined $$newstatus{$s} && $$status{$s} ne $$newstatus{$s}) { $bot->privmsg ($channel, "$s changed state from $$status{$s} to $$newstatus{$s}"); } } } if (defined $trees) { foreach my $t (@trees) { foreach my $e (sort keys %{$$newtrees{$t}}) { if (!defined $$trees{$t}{$e}) { $bot->privmsg($channel, "$t: A new column '$e' has appeared ($$newtrees{$t}{$e})"); } else { if ($$trees{$t}{$e} ne $$newtrees{$t}{$e}) { $bot->privmsg($channel, "$t: '$e' has changed state from $$trees{$t}{$e} to $$newtrees{$t}{$e}"); } } } } } $trees = $newtrees; $status = $newstatus; $bot->schedule (360, \&tinderbox); } # See if someone has changed our source. $::ourdate = 0; $::tinderboxdate = 0; sub checksourcechange { my ($self) = @_; my $lastourdate = $::ourdate; my $lasttinderboxdate = $::tinderboxdate; my ($dev,$ino,$mode,$nlink,$uid,$gid,$rdev,$size, $atime,$mtime,$ctime,$blksize,$blocks) = stat("./mozbot.pl"); $::ourdate = $mtime; ($dev,$ino,$mode,$nlink,$uid,$gid,$rdev,$size, $atime,$mtime,$ctime,$blksize,$blocks) = stat("./Tinderbox.pm"); $::tinderboxdate = $mtime; if (defined $lastourdate && ($::ourdate > $lastourdate || $::tinderboxdate > $lasttinderboxdate)) { $::dontQuitOnSignal = 1; $self->quit("someone seems to have changed my source code. Be right back"); &debug ("restarting self"); exec "$0 @::origargv"; } $bot->schedule (60, \&checksourcechange); }