#! /usr/bin/perl
use lib 't/lib';
use Test::More tests => 11;
use strict;
# This test verifies the solution to a bug reported by Manuel Krenzke.
# The server was not recognizing when clients disconnect.
# Manuel proposed a solution, which is implemented between
# Device::Modbus::TCP and Device::Modbus::TCP::Server.
BEGIN {
use_ok 'Device::Modbus::TCP::Client';
use_ok 'Device::Modbus::TCP::Server';
use_ok 'Test::Unit';
}
# note $Device::Modbus::TCP::VERSION;
# Fork. Child builds a server who dies after a couple of seconds
$|++;
my $pid = fork;
die "Could not fork: $!" unless defined $pid;
unless ($pid) {
# We are the child. Start a server.
# Send an alarm signal in two seconds.
# Then, send a SIGINT to stop the server.
$SIG{ALRM} = sub { kill 2, $$ };
alarm(2);
my $unit = Test::Unit->new( id => 3 );
my $server = Device::Modbus::TCP::Server->new(
port => 6545,
log_level => 4,
log_file => "/tmp/log$$",
server_type => ['PreFork'],
min_servers => 1,
max_servers => 10,
min_spare_servers => 0,
max_spare_servers => 5,
max_requests => 1000,
);
$server->add_server_unit($unit);
$server->start;
}
# The parent is the client. Send requests and evaluate responses.
ok $pid, "Test forked, and server started on PID $pid";
my $client = Device::Modbus::TCP::Client->new( port => 6545 );
isa_ok $client, 'Device::Modbus::Client';
my $req = $client->read_holding_registers(
unit => 3,
address => 2,
quantity => 1
);
isa_ok $req, 'Device::Modbus::Request';
# Get the actual binary string to send to the server
my $adu = $client->new_adu($req);
my $bin = $adu->binary_message;
my $len = length $bin;
my $part1 = substr $bin, 0, $len-2;
my $part2 = substr $bin, -2;
sleep 1;
eval {
# Send request in two parts
$client->socket->send($part1);
$client->socket->shutdown(2);
};
ok !$@, 'Survived sending broken request to forked server';
note $@ if $@;
is wait(), $pid, "Waited for child whose pid was $pid" ;
# Now check the log of the server. Pull everything into a variable
my $log;
{
local $/ = undef;
open my $server_log, '<', "/tmp/log$pid"
or die "Unable to open log file: $!";
$log = <$server_log>;
close $server_log;
}
like $log, qr/Device::Modbus::TCP::Server .*? starting!/,
'Server logged its starting line';
like $log, qr/^Binding to TCP port 6545/m,
'Logged port binding step';
like $log, qr/Server closing!$/m,
'Server closed';
done_testing();