The Perl Toolchain Summit needs more sponsors. If your company depends on Perl, please support this very important event.
use Forks::Super qw(:test_CA overload);
use Forks::Super::Util qw(is_socket is_pipe);
use Test::More tests => 21;
use strict;
use warnings;

$Forks::Super::SOCKET_READ_TIMEOUT = 0.25;
$ENV{TEST_LENIENT} = 1 if $^O =~ /freebsd|midnightbsd/;

# test blocked and unblocked reading for file handles.

my $pid = fork {
    child_fh => "out,err",
    sub => sub {
	print STDERR "foo\n";
	sleep 5;
	print STDOUT "bar\n";
	sleep 5;
	print STDOUT "baz\n";
	exit 0; # unless Forks::Super::Util::is_socket(*STDOUT);
    }
};

ok(isValidPid($pid), "$pid is valid pid");
SKIP: {
    if (Forks::Super::Config::CONFIG('filehandles') == 0) {
	skip "filehandles are unconfigured, ignore handle type test", 1;
    } else {
	ok(!is_socket($pid->{child_stdout}) && !is_pipe($pid->{child_stdout}),
	   "ipc with filehandles");
    }
}
sleep 1;
my $t0 = Time::HiRes::time();
my $err = Forks::Super::read_stderr($pid, "block" => 1);
my $t1 = Time::HiRes::time() - $t0;
ok($err =~ /^foo/, "read stderr");
okl($t1 <= 1.0, "read blocked stderr fast ${t1}s, expected <1s");

my $out = Forks::Super::read_stdout($pid, "block" => 1);
my $t2 = Time::HiRes::time() - $t0;
ok($out =~ /^bar/, "read stdout");
okl($t2 > 2.95, "read blocked stdout ${t2}s, expected ~4s");

$out = Forks::Super::read_stdout($pid, "block" => 0);
my $t3 = Time::HiRes::time() - $t0;
my $t32 = $t3 - $t2;
ok(!defined($out) || (defined($out) && $out eq ''), 
   "non-blocking read on stdout returned empty");
okl($t32 <= 1.0, "non-blocking read took ${t32}s, "
	      . "expected ~${Forks::Super::SOCKET_READ_TIMEOUT}s");

$out = Forks::Super::read_stdout($pid, "block" => 1);
my $t4 = Time::HiRes::time() - $t0;
my $t43 = $t4 - $t3;
ok($out =~ /^baz/, "successful blocking read on stdout");
okl($t43 > 3.25, "read blocked stdout ${t43}s, expected ~5s"); ### 10 ###

#### no more input on STDOUT or STDERR

$err = Forks::Super::read_stderr($pid, "block" => 1);
my $t5 = Time::HiRes::time() - $t0;
my $t54 = $t5 - $t4;
ok(!defined($err), "blocking read on empty stderr returns empty");
okl($t54 <= 5.0,                                       ### 12 ###
   "blocking read on empty stderr can take a moment ${t54}s, expected ~3s");

# print "\$err = $err, time = $t5, $t54\n";

$out = Forks::Super::read_stdout($pid, "block" => 0);
my $t6 = Time::HiRes::time() - $t0;
my $t65 = $t6 - $t5;
ok(!defined($out), "non-blocking read on empty stdout returns empty")
   or diag("read \"$out\" from stdout, expected undef");
okl($t65 <= 1.75, 			               ### 14 ### obs 1.74
   "non-blocking read on empty stdout fast ${t65}s, expected <1.0s");


# read_stdXXX with timeout

$pid = fork {
    child_fh => "out,err",
    sub => sub {
	print STDERR "foo\n";
	sleep 5;
	print STDOUT "bar\n";
	sleep 8;
	print STDOUT "baz\n";
	exit 5;
    }
};
my $x = $pid->read_stderr(timeout => 1);
ok($x, "read avail stderr with timeout");           ### 15 ###
$x = $pid->read_stdout(timeout => 1);
ok(!$x, "read unavail stdout with timeout");        ### 16 ###
my $t = Time::HiRes::time();
$x = $pid->read_stdout(timeout => 9);
ok($x, "read avail stdout with timeout");           ### 17 ###
$t = Time::HiRes::time() - $t;
okl($t <= 7.0, "read took ${t}s, expected ~2-3s");  ### 18 ###
$x = $pid->read_stdout(timeout => 1);
ok(!$x, "read unavail stdout with timeout")         ### 19 ###
    or diag "unexpectedly read: $x\n";
$x = $pid->read_stdout(block => 1);
ok($x, "read stdout with block")                    ### 20 ###
    or diag "expected read, but did not get one";
$x = $pid->read_stderr();
ok(!$x, "read unavail stderr");                     ### 21 ###