| Bot-Net documentation | Contained in the Bot-Net distribution. |
Bot::Net::Mixin::Bot::IRCD - mixin class for building IRC daemon bots
# Build an Eliza chatbot directly on the server
use strict;
use warnings;
package MyBotNet::Bot::ElizaOnServer;
use Bot::Net::Bot;
use Bot::Net::Mixin::Bot::IRCD;
use Chatbot::Eliza; # available separately on CPAN
on _start => run {
# Make it easy for other sessions to talk to us
get(KERNEL)->alias_set('eliza');
};
on bot startup => run {
remember eliza => Chatbot::Eliza->new;
};
on bot message_to_me => run {
my $event = getARG0;
my $reply = recall('eliza')->transform( $event->message );
yield reply_to_sender => $message, $reply;
};
on bot quit => run {
# Clear the session alias for clean shutdown
get(KERNEL)->alias_remove('eliza');
};
1;
Then in your server, you need something like this:
use strict;
use warnings;
package MyBotNet::Server::Main;
use Bot::Net::Server;
use Bot::Net::Mixin::Server::IRC;
use MyBotNet::Bot::ElizaOnServer;
on _start => run {
remember eliza => MyBotNet::Bot::ElizaOnServer->setup;
};
on server quit => run {
# Clean up for clean shutdown
forget 'eliza';
};
1;
Some bots are best run direclty on the server itself. This is generally useful for handling channel or nick management services and other administrative tasks. A typical bot should run as a separate entity whenever possible. See Bot::Net::Mixin::Bot::IRC.
Unlike stand-alone bots, an IRC daemon bot cannot be run using the botnet command. Instead, they must be installed by a specific server and then are run as part of server startup via the botnet command.
Returns a bas configuration for an IRC daemon bot.
The following states are available for your bot to implement.
This is emitted as soon as the nick spoofing and initial channel setup has been setup.
This state occurs when the bot observes a message sent to a channel it is in, but the message is not directed at the bot (i.e., is not prefixed with "MyNick:").
The EVENT is a Bot::Net::Message object setup to contain the group message.
This state occurs when either the bot observes a message sent to a channel that it is in that has been prefixed with "MyNick:" or a private message was sent directly to the bot.
The EVENT is a Bot::Net::Message object setup for either a group or private message depending on which occurred. If a public message the "MyNick:" part of the message will have been stripped.
Tells the server to spoof the bot's nick and registers to receive daemon events. It also sets up an alias for the session so that the server and other server bots may talk to your bot directly. The alias is set according to the "alias" configuration parameter.
It ends by firing the on bot connected state.
Handles IRC messasges sent directly to the bot. This then emits the additional bot state on bot message_to_me. It will be passed a single argument, the Bot::Net::Message containting the message received.
Handles any public messages stated in a channel the bot is in. It will emit either a on bot message_to_group state or on bot message_to_me state depending on whether or not the message was prefixed with "MyNick:".
Both events will receive a single argument, the Bot::Net::Message representing the message sent. The message will have the "MyNick:" prefix stripped on teh on bot message_to_me message.
This sends the given MESSAGE to the given DESTINATION.
The DESTINATION may be one of the following:
#channelThe name of a channel to send to. In this case, no special modifications will be made to the message.
nickThe name of a nick to send to. In this case, no special modifications will be made to the message.
[ #channel1, #channel2, nick1, nick2 ]A list of channels and nicks to send to. In this case, no special modifications will be made to the message.
{ #channel1 => nick1, #channel2 => [ nick1, nick2 ] }A hash of channels pointing to nicks. The nicks may be either a single nick or an array of nicks. In this case, the messages will have the given nick (or nicks) prepended to every message sent (except continuations).
If MESSAGE contains new lines (\n), then the message will be broken up and sent in pieces.
If any message that would be sent to the server approaches the 512 byte limit on IRC messages, the line will be broken. The broken line will have a backslash (\) appended at the break point of the message to signal the recipient that the line was broken. A line may be broken multiple times.
Sends the MESSAGE back to the nick that sent the given EVENT. The EVENT should be a Bot::Net::Message object. The MESSAGE may be broken up and chunked as specified in send_to.
This method will reply back to a user in all the channels it received the message in if the EVENT was sent to a channel (or group of channels) or will send privately back if the EVENT was private.
Sends the MESSAGE back to the nick that sent the given EVENT, but via a private message directly to their nick, even if the original EVENT took place as a channel message.
Sends the MESSAGE back to the nick that sent the given EVENT, but via the public group name in the CHANNEL argument.
This is similar to reply_to_sender, except that if sent to a channel, the sender will not be identified by nick.
This causes the IRC client to close down the connection and quit.
Andrew Sterling Hanenkamp <hanenkamp@cpan.org>
Copyright 2007 Boomer Consulting, Inc. All Rights Reserved.
This program is free software and may be modified and distributed under the same terms as Perl itself.
| Bot-Net documentation | Contained in the Bot-Net distribution. |
use strict; use warnings; package Bot::Net::Mixin::Bot::IRCD; use Bot::Net::Mixin; use Scalar::Util qw/ reftype /;
sub default_configuration { my $class = shift; my $package = shift; my $name = Bot::Net->short_name_for_bot($package); $name =~ s/\W+/_/g; my $default_channel = Bot::Net->config->net('ApplicationName'); $default_channel =~ s/\W+/_/g; return { alias => $name, spoofing => { nick => $name, ircname => $name, }, channels => [ '#'.$default_channel ], }; }
on _start => run { my $alias = recall [ config => 'alias' ]; my $spoofed = recall [ config => 'spoofing' ]; my $channels = recall [ config => 'channels' ]; get(KERNEL)->alias_set($alias) if $alias; recall('log')->info('Setting up nick spoofing as ' .$spoofed->{nick}); post ircd => register => 'all'; # TODO limit this to a subset # Make sure we are, in fact, ready when we report call ircd => add_spoofed_nick => $spoofed; recall('log')->info("BOT READY : nick $spoofed->{nick}"); for my $channel (@{ $channels || [] }) { recall('log')->info('Joining '.$channel); post ircd => daemon_cmd_join => $spoofed->{nick}, $channel; } yield 'bot_connected'; };
on ircd_daemon_privmsg => run { my $userhost = get ARG0; my $me = get ARG1; my $message = get ARG2; my $my_nick = recall [ config => spoofing => 'nick' ]; my ($nick, $host) = split /!/, $userhost; # Only respond to messages directly to me if ($me eq $my_nick) { my $event = Bot::Net::Message->new({ sender_nick => $nick, sender_host => $host, recipient_nicks => $me, message => $message, private => 1, }); yield bot_message_to_me => $event; } };
on ircd_daemon_public => run { my $userhost = get ARG0; my $channel = get ARG1; my $message = get ARG2; my $my_nick = recall [ config => spoofing => 'nick' ]; my $channels = recall [ config => 'channels' ]; my ($nick, $host) = split /!/, $userhost; # FIXME This isn't a very good mechanism for determining if this spoofed # nick is in the channel since it could have changed. # Am I in this channel? if (grep { $_ eq $channel } @{ $channels || [] }) { my $state; if ($message =~ s/^\Q$my_nick\E:\s*//) { $state = 'message_to_me'; } else { $state = 'message_to_group'; } my $event = Bot::Net::Message->new({ sender_nick => $nick, sender_host => $host, recipient_groups => $channel, message => $message, public => 1, }); yield 'bot_' . $state => $event; } };
on send_to => run { my $to = get ARG0; my $full_message = get ARG1; my $log = recall 'log'; my $my_nick = recall [ config => spoofing => 'nick' ]; # Split the message up by newlines my @messages = split /\n/, $full_message; # Normalize the destination by channels and nicks my (%group_destinations, %private_destinations); # Given a single channel or nick if (!defined reftype $to) { # Given a single channel if ($to =~ /\#/) { $group_destinations{$to} = [ ]; } # Given a single nick else { $private_destinations{$to}++; } } # Given an array of channels and/or nicks elsif (reftype($to) eq 'ARRAY') { # For each channel/nick in the array... for my $thing (@$to) { # Add channels to the group list if ($thing =~ /\#/) { $group_destinations{$thing} = [ ]; } # Add nicks to the private list else { $private_destinations{$thing}++; } } } # Given a hash of channels => nicks elsif (reftype($to) eq 'HASH') { # For each channel in the hash... for my $channel (keys %$to) { my $nicks = $to->{$channel}; # If a single nick, wrap it in an array if (!defined reftype($nicks)) { $group_destinations{$channel} = [ $nicks ]; } # If an array, assume an array of nicks elsif (reftype($nicks) eq 'ARRAY') { $group_destinations{$channel} = $nicks; } # Wha? else { $log->error("send_to: Don't know what to do with $channel => $nicks, ignoring."); } } } # Wha? else { $log->error("send_to: Don't know what to do with $to, ignore."); return; } # Internal function for chunking the message my $message_chunks = sub { my $message = shift; my @chunks; while (length $message > 400) { my $chunk = substr $message, 0, 400; $message = substr $message, 400; push @chunks, $chunk.'\\'; } push @chunks, $message; return @chunks; }; # Handle group messages for my $channel (keys %group_destinations) { my $nicks = $group_destinations{$channel}; # For each message line for my $message (@messages) { # Prepend any nicks we have if (@$nicks) { $message = join(',', @$nicks).': '.$message; } # Post the message by chunks post ircd => daemon_cmd_privmsg => $my_nick => $channel => $_ for $message_chunks->($message); } } # Send out the private messages for my $nick (keys %private_destinations) { # For each message line, post it to the nick in chunks for my $message (@messages) { post ircd => daemon_cmd_privmsg => $my_nick => $nick => $_ for $message_chunks->($message); } } };
on reply_to_sender => run { my $event = get ARG0; my $message = get ARG1; if ($event->public) { my %send_to = map { $_ => $event->sender_nick } @{ $event->recipient_groups }; yield send_to => \%send_to => $message; } else { yield send_to => $event->sender_nick => $message; } };
on reply_to_sender_privately => run { my $event = get ARG0; my $message = get ARG1; yield send_to => $event->sender_nick => $message; };
on reply_to_sender_group => run { my $event = get ARG0; my $channel = get ARG1; my $message = get ARG2; yield send_to => { $channel => $event->sender_nick } => $message; };
on reply_to_general => run { my $event = get ARG0; my $message = get ARG1; if ($event->public) { yield send_to => $event->recipient_groups => $message; } else { yield send_to => $event->sender_nick => $message; } };
on bot_quit => run { recall('log')->warn("Stopping spoofed server-side IRC bot."); my $alias = recall [ config => 'alias' ]; my $my_nick = recall [ config => spoofing => 'nick' ]; post ircd => del_spoofed_nick => $my_nick => 'Quitting.'; post ircd => unregister => 'all'; get(KERNEL)->alias_remove($alias) if $alias; };
1;