X-Git-Url: http://git.annexia.org/?p=xavierbot.git;a=blobdiff_plain;f=xavierbot.pl.in;h=8f01f0b80ab25c6ef4a5c0a081e417446b5c6359;hp=8b97671c9a8a3da458bbbf2c563a80fe310189af;hb=cf77e53fed9a4870a5774825127c670f51a645b7;hpb=d165f93ed79c62a1fa76f391d87ffc4d215b9efe diff --git a/xavierbot.pl.in b/xavierbot.pl.in index 8b97671..8f01f0b 100755 --- a/xavierbot.pl.in +++ b/xavierbot.pl.in @@ -2,44 +2,42 @@ # xavierbot : an OCaml interpreter IRC bot. # By Richard W.M. Jones . # This code is in the Public Domain. -# $Id: xavierbot.pl.in,v 1.1 2007/06/28 19:47:26 rjones Exp $ +# $Id: xavierbot.pl.in,v 1.7 2007/06/29 21:44:02 rjones Exp $ use strict; - -use POE qw(Component::IRC); +use POE qw(Component::IRC Wheel::Run); #---------------------------------------------------------------------- # Start of configuration. my $nick = "xavierbot"; my $ircname = "Xavierbot"; # Printable name. -#my $server = "chat.freenode.net"; -my $server = "devserv.devel.redhat.com"; +my $server = "chat.freenode.net"; +#my $server = "devserv.devel.redhat.com"; my $port = 6667; - -my @channels = ("#ocaml"); +my $channel = "#ocaml"; # End of configuration. #---------------------------------------------------------------------- +# Current state. + +# Simple flood protection. This counts number of lines received from +# the toplevel, and is reset when we send a line. If this exceeds +# some value, then we just eat lines. +# XXX This ought to count characters, not lines. +my $flood_lim = 0; -my $irc = POE::Component::IRC->spawn ( - nick => $nick, - ircname => $ircname, - server => $server, - port => $port, - ) or die "POE::Component::IRC->spawn failed: $!"; +# Are we awake or sleeping? +my $sleeping = 0; + +#---------------------------------------------------------------------- + +$ENV{PATH} = "/usr/bin:/bin"; POE::Session->create ( package_states => [ - main => [ qw(_default _start irc_001 irc_public) ], + main => [ qw(_default _start irc_001 irc_public got_stdout got_sigchld) ], ], - heap => { irc => $irc }, -); - -POE::Wheel::Run->new ( - Program => [ "@WRAPPER@" ], - StdoutEvent => 'stdout_event', - StderrEvent => 'stdout_event', ); POE::Kernel->run (); @@ -49,6 +47,21 @@ sub _start { my ($kernel, $heap) = @_[KERNEL,HEAP]; + my $irc = POE::Component::IRC->spawn + ( + nick => $nick, + ircname => $ircname, + server => $server, + port => $port, + ) or die "POE::Component::IRC->spawn failed: $!"; + + my $ocaml = start_toplevel (); + + $kernel->sig(CHLD => qw(got_sigchld)); + + $heap->{irc} = $irc; + $heap->{ocaml} = $ocaml; + my $irc_session = $heap->{irc}->session_id (); $kernel->post ($irc_session => register => "all"); $kernel->post ($irc_session => connect => { }); @@ -63,21 +76,56 @@ sub irc_001 my $poco_object = $sender->get_heap (); print "Connected to ", $poco_object->server_name (), "\n"; - $kernel->post ($sender => join => $_ ) for @channels; + $kernel->post ($sender => join => $channel); undef; } sub irc_public { - my ($kernel, $sender, $who, $where, $what) = - @_[KERNEL,SENDER,ARG0,ARG1,ARG2]; + my ($kernel, $sender, $who, $where, $what, $heap) = + @_[KERNEL,SENDER,ARG0,ARG1,ARG2,HEAP]; my $nick = (split /!/, $who)[0]; my $channel = $where->[0]; + my @usage = + ( + "expr ;; evaluate expr in toplevel and print result", + "help help message", + "restart restart the toplevel", + "sleep go to sleep", + "wake wake me up from sleep", + ); + print "got: $what\n"; - if (my ($rot13) = $what =~ /^rot13 (.+)/) { - $rot13 =~ tr[a-zA-Z][n-za-mN-ZA-M]; - $kernel->post ($sender => privmsg => $channel => "$nick: $rot13"); + # XXX How to interpolate $nick into the patterns? + if ($what =~ /^\s*xavierbot\b.*\bhelp\b/) { + my $nick = (split /!/, $who)[0]; + $kernel->post ($sender => privmsg => $channel => + "hello $nick, I am xavierbot @VERSION@, an OCaml toplevel"); + $kernel->post ($sender => privmsg => $channel => $_) + foreach (@usage); + } + elsif (!$sleeping) { + if (my ($stmt) = $what =~ m/^\s*([^\#].*;;)\s*$/) { + $heap->{ocaml}->put ("$stmt\n"); + $flood_lim = 0; + } + elsif ($what =~ /^\s*xavierbot\b.*\b(sleep|shut|quiet)\b/) { + $sleeping = 1; + $kernel->post ($sender => privmsg => $channel => + "xavierbot goes to sleep (do 'xavierbot wake' to wake)"); + } + } else { # sleeping + if ($what =~ /^\s*xavierbot\b.*\bwake\b/) { + $sleeping = 0; + $kernel->post ($sender => privmsg => $channel => + "xavierbot wakes up"); + } + elsif ($what =~ /^\s*xavierbot\b.*\brestart\b/) { + $sleeping = 0; + print STDOUT "got instruction to restart ...\n"; + restart_toplevel ($heap->{ocaml}); + } } undef; } @@ -100,10 +148,49 @@ sub _default #---------------------------------------------------------------------- -# Start the bot. +# Toplevel wrote something. -sub stdout_event +sub got_stdout { - my ($heap, $input, $wheel_id) = @_[HEAP,ARG0,ARG1]; + my ($kernel,$heap, $input, $wheel_id) = @_[KERNEL,HEAP,ARG0,ARG1]; print "Child said: $input\n"; + if ($flood_lim < 10) { + $kernel->post ($heap->{irc} => privmsg => $channel => "$input"); + } + $flood_lim++; +} + +# Got a SIGCHLD, so start the bot up again. + +sub got_sigchld +{ + my ($kernel, $heap) = @_[KERNEL,HEAP]; + my $ocaml = start_toplevel (); + $heap->{ocaml} = $ocaml; +} + +# Start up the toplevel (assumes it's not running). + +sub start_toplevel +{ + return POE::Wheel::Run->new + ( + Program => "@WRAPPER@", + StdoutEvent => "got_stdout", + StderrEvent => "got_stdout", + ) or die "POE::Wheel::Run->new @WRAPPER@ failed: $!"; +} + +# Restart the toplevel - kill the old one, and a new one +# will be spawned after we get the SIGCHLD signal. +# +# XXX Can't send signal to setuid child, so instead just close +# stdin. + +sub restart_toplevel +{ + my $ocaml = shift; + + $ocaml->kill (9); + $ocaml->shutdown_stdin; }