2010-03-17 07:36:54 +01:00
|
|
|
# File: PBot.pm
|
|
|
|
# Author: pragma_
|
|
|
|
#
|
|
|
|
# Purpose: IRC Bot (3rd generation)
|
|
|
|
|
|
|
|
package PBot::PBot;
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings;
|
|
|
|
|
2010-03-23 19:24:02 +01:00
|
|
|
use vars qw($VERSION);
|
2010-03-26 09:58:25 +01:00
|
|
|
$VERSION = "0.6.3-beta";
|
2010-03-17 07:36:54 +01:00
|
|
|
|
|
|
|
# unbuffer stdout
|
|
|
|
STDOUT->autoflush(1);
|
|
|
|
|
|
|
|
use Carp ();
|
|
|
|
use PBot::Logger;
|
2010-03-22 08:33:44 +01:00
|
|
|
|
2010-03-17 07:36:54 +01:00
|
|
|
use PBot::StdinReader;
|
2010-03-22 08:33:44 +01:00
|
|
|
|
|
|
|
use Net::IRC;
|
|
|
|
use PBot::IRCHandlers;
|
2010-03-23 19:24:02 +01:00
|
|
|
use PBot::Channels;
|
2010-03-22 08:33:44 +01:00
|
|
|
|
2010-03-17 07:36:54 +01:00
|
|
|
use PBot::AntiFlood;
|
2010-03-22 08:33:44 +01:00
|
|
|
use PBot::Interpreter;
|
|
|
|
use PBot::Commands;
|
|
|
|
|
|
|
|
use PBot::ChanOps;
|
2010-03-23 19:24:02 +01:00
|
|
|
use PBot::ChanOpCommands;
|
2010-03-22 08:33:44 +01:00
|
|
|
|
|
|
|
use PBot::Factoids;
|
|
|
|
use PBot::FactoidCommands;
|
2010-03-17 07:36:54 +01:00
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
use PBot::BotAdmins;
|
|
|
|
use PBot::BotAdminCommands;
|
2010-03-17 07:36:54 +01:00
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
use PBot::IgnoreList;
|
|
|
|
use PBot::IgnoreListCommands;
|
|
|
|
|
2010-03-23 19:24:02 +01:00
|
|
|
use PBot::Quotegrabs;
|
|
|
|
# no PBot::QuotegrabsCommands (bundled inside PBot::Quotegrabs for a change)
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
use PBot::Timer;
|
2010-03-17 07:36:54 +01:00
|
|
|
|
|
|
|
sub new {
|
|
|
|
if(ref($_[1]) eq 'HASH') {
|
|
|
|
Carp::croak("Options to Logger should be key/value pairs, not hash reference");
|
|
|
|
}
|
|
|
|
|
|
|
|
my ($class, %conf) = @_;
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
my $self = bless {}, $class;
|
|
|
|
$self->initialize(%conf);
|
|
|
|
return $self;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub initialize {
|
|
|
|
my ($self, %conf) = @_;
|
|
|
|
|
2010-03-17 07:36:54 +01:00
|
|
|
my $log_file = delete $conf{log_file};
|
2010-03-22 08:33:44 +01:00
|
|
|
my $channels_file = delete $conf{channels_file};
|
|
|
|
my $admins_file = delete $conf{admins_file};
|
|
|
|
|
|
|
|
my $botnick = delete $conf{botnick};
|
2010-04-10 00:52:24 +02:00
|
|
|
my $username = delete $conf{username};
|
|
|
|
my $ircname = delete $conf{ircname};
|
2010-03-22 08:33:44 +01:00
|
|
|
my $identify_password = delete $conf{identify_password};
|
|
|
|
|
|
|
|
my $ircserver = delete $conf{ircserver};
|
|
|
|
my $max_msg_len = delete $conf{max_msg_len};
|
|
|
|
my $MAX_FLOOD_MESSAGES = delete $conf{MAX_FLOOD_MESSAGES};
|
|
|
|
my $MAX_NICK_MESSAGES = delete $conf{MAX_NICK_MESSAGES};
|
|
|
|
|
2010-04-06 20:38:27 +02:00
|
|
|
my $ignorelist_file = delete $conf{ignorelist_file};
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
my $factoids_file = delete $conf{factoids_file};
|
|
|
|
my $export_factoids_path = delete $conf{export_factoids_path};
|
|
|
|
my $export_factoids_site = delete $conf{export_factoids_site};
|
|
|
|
my $module_dir = delete $conf{module_dir};
|
|
|
|
|
|
|
|
my $quotegrabs_file = delete $conf{quotegrabs_file};
|
|
|
|
my $export_quotegrabs_path = delete $conf{export_quotegrabs_path};
|
|
|
|
my $export_quotegrabs_site = delete $conf{export_quotegrabs_site};
|
2010-03-17 07:36:54 +01:00
|
|
|
|
|
|
|
$ircserver = "irc.freenode.net" unless defined $ircserver;
|
2010-03-22 08:33:44 +01:00
|
|
|
$botnick = "pbot2" unless defined $botnick;
|
|
|
|
$identify_password = "" unless defined $identify_password;
|
2010-03-17 07:36:54 +01:00
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
$max_msg_len = 460 unless defined $max_msg_len;
|
|
|
|
$MAX_FLOOD_MESSAGES = 4 unless defined $MAX_FLOOD_MESSAGES;
|
|
|
|
$MAX_NICK_MESSAGES = 8 unless defined $MAX_NICK_MESSAGES;
|
2010-03-17 07:36:54 +01:00
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
$self->{botnick} = $botnick;
|
2010-04-10 00:52:24 +02:00
|
|
|
$self->{username} = $username;
|
|
|
|
$self->{ircname} = $ircname;
|
2010-03-22 08:33:44 +01:00
|
|
|
$self->{identify_password} = $identify_password;
|
|
|
|
|
|
|
|
$self->{max_msg_len} = $max_msg_len;
|
|
|
|
$self->{MAX_FLOOD_MESSAGES} = $MAX_FLOOD_MESSAGES;
|
|
|
|
$self->{MAX_NICK_MESSAGES} = $MAX_NICK_MESSAGES;
|
|
|
|
$self->{FLOOD_CHAT} = 0;
|
|
|
|
|
|
|
|
my $logger = PBot::Logger->new(log_file => $log_file);
|
|
|
|
$self->{logger} = $logger;
|
|
|
|
|
2010-03-23 19:24:02 +01:00
|
|
|
$self->{commands} = PBot::Commands->new(pbot => $self);
|
|
|
|
$self->{timer} = PBot::Timer->new(timeout => 10);
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
$self->{admins} = PBot::BotAdmins->new(
|
|
|
|
pbot => $self,
|
|
|
|
filename => $admins_file,
|
|
|
|
);
|
|
|
|
|
|
|
|
$self->admins->load_admins();
|
2010-03-29 14:30:35 +02:00
|
|
|
$self->admins->add_admin($botnick, '.*', "$botnick!stdin\@localhost", 60, 'admin');
|
2010-03-30 08:34:08 +02:00
|
|
|
$self->admins->login($botnick, "$botnick!stdin\@localhost", 'admin');
|
2010-03-22 08:33:44 +01:00
|
|
|
|
|
|
|
$self->{factoids} = PBot::Factoids->new(
|
|
|
|
pbot => $self,
|
|
|
|
filename => $factoids_file,
|
|
|
|
export_path => $export_factoids_path,
|
|
|
|
export_site => $export_factoids_site,
|
|
|
|
);
|
|
|
|
|
|
|
|
$self->factoids->add_factoid('text', '.*', $botnick, 'version', "/say pbot2 version $VERSION");
|
|
|
|
$self->factoids->load_factoids() if defined $factoids_file;
|
|
|
|
|
|
|
|
$self->module_dir($module_dir);
|
|
|
|
|
|
|
|
$self->{antiflood} = PBot::AntiFlood->new(pbot => $self);
|
2010-04-06 20:38:27 +02:00
|
|
|
|
|
|
|
$self->{ignorelist} = PBot::IgnoreList->new(pbot => $self, filename => $ignorelist_file);
|
|
|
|
$self->{ignorelist}->load_ignores() if defined $ignorelist_file;
|
2010-03-22 08:33:44 +01:00
|
|
|
|
|
|
|
$self->interpreter(PBot::Interpreter->new(pbot => $self));
|
|
|
|
$self->interpreter->register(sub { return $self->commands->interpreter(@_); });
|
|
|
|
$self->interpreter->register(sub { return $self->factoids->interpreter(@_); });
|
|
|
|
|
|
|
|
$self->{botadmincmds} = PBot::BotAdminCommands->new(pbot => $self);
|
|
|
|
$self->{factoidcmds} = PBot::FactoidCommands->new(pbot => $self);
|
|
|
|
$self->{ignorelistcmds} = PBot::IgnoreListCommands->new(pbot => $self);
|
|
|
|
|
|
|
|
$self->{irc} = Net::IRC->new();
|
|
|
|
$self->{ircserver} = $ircserver;
|
|
|
|
$self->{irchandlers} = PBot::IRCHandlers->new(pbot => $self);
|
|
|
|
|
|
|
|
$self->{channels} = PBot::Channels->new(pbot => $self, filename => $channels_file);
|
|
|
|
$self->channels->load_channels() if defined $channels_file;
|
|
|
|
|
2010-03-23 19:24:02 +01:00
|
|
|
$self->{chanops} = PBot::ChanOps->new(pbot => $self);
|
|
|
|
$self->{chanopcmds} = PBot::ChanOpCommands->new(pbot => $self);
|
|
|
|
|
|
|
|
$self->{quotegrabs} = PBot::Quotegrabs->new(
|
|
|
|
pbot => $self,
|
|
|
|
filename => $quotegrabs_file,
|
|
|
|
export_path => $export_quotegrabs_path,
|
|
|
|
export_site => $export_quotegrabs_site,
|
|
|
|
);
|
|
|
|
|
|
|
|
$self->quotegrabs->add_quotegrab($botnick, "#pbot2", 0, "pragma_", "Who's a bot?");
|
|
|
|
$self->quotegrabs->load_quotegrabs() if defined $quotegrabs_file;
|
2010-03-22 08:33:44 +01:00
|
|
|
|
|
|
|
$self->timer->start();
|
2010-03-17 07:36:54 +01:00
|
|
|
}
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
# TODO: add disconnect subroutine
|
2010-03-17 07:36:54 +01:00
|
|
|
|
|
|
|
sub connect {
|
|
|
|
my ($self, $server) = @_;
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
$server = $self->ircserver if not defined $server;
|
|
|
|
|
|
|
|
if($self->{connected}) {
|
|
|
|
# TODO: disconnect, clean-up, etc
|
|
|
|
}
|
|
|
|
|
|
|
|
$self->logger->log("Connecting to $server ...\n");
|
2010-03-17 07:36:54 +01:00
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
$self->conn($self->irc->newconn(
|
|
|
|
Nick => $self->{botnick},
|
|
|
|
Username => $self->{username},
|
|
|
|
Ircname => $self->{ircname},
|
2010-03-17 07:36:54 +01:00
|
|
|
Server => $server,
|
2010-03-22 08:33:44 +01:00
|
|
|
Port => $self->{port}))
|
|
|
|
or Carp::croak "$0: Can't connect to IRC server.\n";
|
|
|
|
|
|
|
|
$self->{connected} = 1;
|
|
|
|
|
|
|
|
#set up default handlers for the IRC engine
|
|
|
|
$self->conn->add_handler([ 251,252,253,254,302,255 ], sub { $self->irchandlers->on_init(@_) });
|
|
|
|
$self->conn->add_handler(376 , sub { $self->irchandlers->on_connect(@_) });
|
|
|
|
$self->conn->add_handler('disconnect' , sub { $self->irchandlers->on_disconnect(@_) });
|
|
|
|
$self->conn->add_handler('notice' , sub { $self->irchandlers->on_notice(@_) });
|
|
|
|
$self->conn->add_handler('caction' , sub { $self->irchandlers->on_action(@_) });
|
|
|
|
$self->conn->add_handler('public' , sub { $self->irchandlers->on_public(@_) });
|
|
|
|
$self->conn->add_handler('msg' , sub { $self->irchandlers->on_msg(@_) });
|
|
|
|
$self->conn->add_handler('mode' , sub { $self->irchandlers->on_mode(@_) });
|
|
|
|
$self->conn->add_handler('part' , sub { $self->irchandlers->on_departure(@_) });
|
|
|
|
$self->conn->add_handler('join' , sub { $self->irchandlers->on_join(@_) });
|
|
|
|
$self->conn->add_handler('quit' , sub { $self->irchandlers->on_departure(@_) });
|
2010-03-17 07:36:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
#main loop
|
|
|
|
sub do_one_loop {
|
2010-03-22 08:33:44 +01:00
|
|
|
my $self = shift;
|
|
|
|
|
2010-03-17 07:36:54 +01:00
|
|
|
# process IRC events
|
2010-03-22 08:33:44 +01:00
|
|
|
$self->irc->do_one_loop();
|
2010-03-17 07:36:54 +01:00
|
|
|
|
|
|
|
# process STDIN events
|
2010-03-22 08:33:44 +01:00
|
|
|
$self->check_stdin();
|
|
|
|
}
|
|
|
|
|
|
|
|
sub start {
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
if(not defined $self->{connected} or $self->{connected} == 0) {
|
|
|
|
$self->connect();
|
|
|
|
}
|
|
|
|
|
|
|
|
while(1) {
|
|
|
|
$self->do_one_loop();
|
|
|
|
}
|
2010-03-17 07:36:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
sub check_stdin {
|
2010-03-22 08:33:44 +01:00
|
|
|
my $self = shift;
|
|
|
|
|
2010-03-17 07:36:54 +01:00
|
|
|
my $input = PBot::StdinReader::check_stdin();
|
|
|
|
|
|
|
|
return if not defined $input;
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
$self->logger->log("---------------------------------------------\n");
|
|
|
|
$self->logger->log("Read '$input' from STDIN\n");
|
2010-03-17 07:36:54 +01:00
|
|
|
|
|
|
|
my ($from, $text);
|
|
|
|
|
|
|
|
if($input =~ m/^~([^ ]+)\s+(.*)/) {
|
|
|
|
$from = $1;
|
|
|
|
$text = "!$2";
|
|
|
|
} else {
|
|
|
|
$from = undef;
|
|
|
|
$text = "!$input";
|
|
|
|
}
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
return $self->interpreter->process_line($from, $self->{botnick}, "stdin", "localhost", $text);
|
|
|
|
}
|
|
|
|
|
|
|
|
sub irc {
|
|
|
|
my $self = shift;
|
|
|
|
return $self->{irc};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub logger {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{logger} = shift; }
|
|
|
|
return $self->{logger};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub channels {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{channels} = shift; }
|
|
|
|
return $self->{channels};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub factoids {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{factoids} = shift; }
|
|
|
|
return $self->{factoids};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub timer {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{timer} = shift; }
|
|
|
|
return $self->{timer};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub conn {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{conn} = shift; }
|
|
|
|
return $self->{conn};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub irchandlers {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{irchandlers} = shift; }
|
|
|
|
return $self->{irchandlers};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub interpreter {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{interpreter} = shift; }
|
|
|
|
return $self->{interpreter};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub admins {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{admins} = shift; }
|
|
|
|
return $self->{admins};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub commands {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{commands} = shift; }
|
|
|
|
return $self->{commands};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub botnick {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{botnick} = shift; }
|
|
|
|
return $self->{botnick};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub identify_password {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{identify_password} = shift; }
|
|
|
|
return $self->{identify_password};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub max_msg_len {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{max_msg_len} = shift; }
|
|
|
|
return $self->{max_msg_len};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub module_dir {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{module_dir} = shift; }
|
|
|
|
return $self->{module_dir};
|
2010-03-17 07:36:54 +01:00
|
|
|
}
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
sub ignorelist {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{ignorelist} = shift; }
|
|
|
|
return $self->{ignorelist};
|
2010-03-17 07:36:54 +01:00
|
|
|
}
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
sub antiflood {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{antiflood} = shift; }
|
|
|
|
return $self->{antiflood};
|
2010-03-17 07:36:54 +01:00
|
|
|
}
|
|
|
|
|
2010-03-23 19:24:02 +01:00
|
|
|
sub quotegrabs {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{quotegrabs} = shift; }
|
|
|
|
return $self->{quotegrabs};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub chanops {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{chanops} = shift; }
|
|
|
|
return $self->{chanops};
|
|
|
|
}
|
|
|
|
|
2010-03-22 08:33:44 +01:00
|
|
|
sub ircserver {
|
|
|
|
my $self = shift;
|
|
|
|
if(@_) { $self->{ircserver} = shift; }
|
|
|
|
return $self->{ircserver};
|
2010-03-17 07:36:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
1;
|