From Code to Community: Sponsoring The Perl and Raku Conference 2025 Learn more

use strict;
use warnings FATAL => 'all';
use lib 't/inc';
use File::Temp qw(tempdir);
use File::Spec::Functions qw(catfile);
use POE;
my $log_dir = tempdir(CLEANUP => 1);
my $bot1 = POE::Component::IRC::State->spawn(
Flood => 1,
plugin_debug => 1,
);
my $bot2 = POE::Component::IRC::State->spawn(
Flood => 1,
plugin_debug => 1,
);
my $ircd = POE::Component::Server::IRC->spawn(
Auth => 0,
AntiFlood => 0,
);
$bot2->plugin_add(Logger => POE::Component::IRC::Plugin::Logger->new(
Path => $log_dir,
));
my $file = catfile($log_dir, '=testbot1.log');
unlink $file if -e $file;
my @correct = (
qr/^--> Opened DCC chat connection with TestBot1 \(\S+:\d+\)$/,
'<TestBot1> Oh hi',
'* TestBot1 does something',
'<TestBot2> Hi yourself',
'* TestBot2 does something as well',
qr/^<-- Closed DCC chat connection with TestBot1 \(\S+:\d+\)$/,
);
plan tests => 7 + @correct;
POE::Session->create(
package_states => [
main => [qw(
_start
ircd_listener_add
ircd_listener_failure
_shutdown
irc_001
irc_dcc_request
irc_dcc_start
irc_dcc_chat
irc_disconnected
)],
],
);
$poe_kernel->run();
sub _start {
my ($kernel) = $_[KERNEL];
$ircd->yield('register', 'all');
$ircd->yield('add_listener');
$kernel->delay(_shutdown => 60, 'Timed out');
}
sub ircd_listener_failure {
my ($kernel, $op, $reason) = @_[KERNEL, ARG1, ARG3];
$kernel->yield('_shutdown', "$op: $reason");
}
sub ircd_listener_add {
my ($kernel, $port) = @_[KERNEL, ARG0];
$bot1->yield(register => 'all');
$bot1->yield(connect => {
nick => 'TestBot1',
server => '127.0.0.1',
port => $port,
});
$bot2->yield(register => 'all');
$bot2->yield(connect => {
nick => 'TestBot2',
server => '127.0.0.1',
port => $port,
});
}
sub _shutdown {
my ($kernel, $error) = @_[KERNEL, ARG0];
fail($error) if defined $error;
$kernel->alarm_remove_all();
$ircd->yield('shutdown');
$bot1->yield('shutdown');
$bot2->yield('shutdown');
}
sub irc_001 {
my ($heap, $server) = @_[HEAP, ARG0];
my $irc = $_[SENDER]->get_heap();
pass($irc->nick_name() . ' logged in');
$heap->{logged_in}++;
return if $heap->{logged_in} != 2;
$bot2->yield(dcc => $bot1->nick_name() => CHAT => undef, undef, 5);
}
sub irc_dcc_request {
my ($sender, $cookie) = @_[SENDER, ARG3];
my $irc = $sender->get_heap();
pass($irc->nick_name() . ' got dcc request');
$irc->yield(dcc_accept => $cookie);
}
sub irc_dcc_start {
my ($sender, $heap, $id) = @_[SENDER, HEAP, ARG0];
my $irc = $sender->get_heap();
pass($irc->nick_name() . ' got irc_dcc_started');
$heap->{started}++;
if ($heap->{started} == 2) {
$irc->yield(dcc_chat => $id, 'Oh hi');
$irc->yield(dcc_chat => $id, "\001ACTION does something\001");
}
}
sub irc_dcc_chat {
my ($heap, $sender, $id, $msg) = @_[HEAP, SENDER, ARG0, ARG3];
my $irc = $sender->get_heap();
$heap->{msgs}++;
if ($heap->{msgs} == 2) {
$irc->yield(dcc_chat => $id, 'Hi yourself');
$irc->yield(dcc_chat => $id, "\001ACTION does something as well\001");
}
elsif ($heap->{msgs} == 4) {
$irc->yield(dcc_close => $id);
$bot1->yield('quit');
$bot2->yield('quit');
}
}
sub irc_disconnected {
my ($kernel, $heap) = @_[KERNEL, HEAP];
pass('irc_disconnected');
$heap->{count}++;
if ($heap->{count} == 2) {
verify_log();
$kernel->yield('_shutdown');
}
}
sub verify_log {
open my $log, '<', $file or die "Can't open log file '$file': $!";
my @lines = <$log>;
close $log;
my $check = 0;
for my $line (@lines) {
next if $line =~ /^\*{3}/;
chomp $line;
$line = substr($line, 20);
last if !defined $correct[$check];
if (ref $correct[$check] eq 'Regexp') {
like($line, $correct[$check], 'Line ' . ($check+1));
}
else {
is($line, $correct[$check], 'Line ' . ($check+1));
}
$check++;
}
fail('Log too short') if $check > @correct;
}