Hypersonic

 view release on metacpan or  search on metacpan

t/lib/HypersonicTest.pm  view on Meta::CPAN

    close $log;

    my $pid = fork();
    die "Fork failed: $!" unless defined $pid;
    if ($pid == 0) {
        # Child: redirect both streams to the log so anything the
        # server prints (or croaks with) survives the fork boundary.
        # We open + dup2 at the fd level (via POSIX::dup2) so that C
        # code in the JIT-loaded .so that writes directly with
        # fprintf(stderr,...) or write(2,...) also lands in the log.
        # Pure `open STDERR, ...` only redirects Perl's PerlIO layer
        # and can leave C stdio still pointing at the original fd 2,
        # which is why earlier CPAN tester reports showed
        # "(child wrote no output)" - the croak DID happen but its
        # bytes went to a closed fd.
        require POSIX;
        open(my $log_fh, '>', $log_path) or die "open log: $!";
        $log_fh->autoflush(1);
        POSIX::dup2(fileno($log_fh), 1) or die "dup2 stdout: $!";
        POSIX::dup2(fileno($log_fh), 2) or die "dup2 stderr: $!";
        # Re-open Perl's STDOUT/STDERR onto the now-redirected fds so
        # `print` / `warn` from Perl also reach the log.
        open STDOUT, '>&=', 1 or die "reopen stdout: $!";
        open STDERR, '>&=', 2 or die "reopen stderr: $!";
        select STDERR; $| = 1;
        select STDOUT; $| = 1;
        # Force Hypersonic to print a breadcrumb before JIT compile so
        # the captured log is never empty if wait_for_port gives up.
        local $ENV{HYPERSONIC_COMPILE_DIAG} = 1;
        # Emit an EARLY breadcrumb (before `require Hypersonic`)
        # because on slow smoker hosts (e.g. ARMv6 Pi w/ DEBUGGING
        # perl) just loading Hypersonic.pm can take many seconds, and
        # if the parent's wait_for_port budget elapses during that
        # load we want the log to at least say "child reached fork"
        # rather than be empty.
        print STDERR "# HypersonicTest: child pid $$ alive, loading...\n";
        eval { $child_code->(); };
        my $err = $@;
        STDOUT->flush;
        STDERR->flush;
        if ($err) {
            print STDERR "child died: $err\n";
            STDERR->flush;
            POSIX::_exit(70);  # EX_SOFTWARE
        }
        POSIX::_exit(0);
    }
    return ($pid, $log_path);
}

# wait_for_port($port [, $opts]) -> 1 / 0
#
# Probes 127.0.0.1:$port until something accepts or we give up. On
# timeout, if $opts->{log} is given, diag()s the child's captured
# output; if $opts->{pid} is given, diag()s the child's exit status.
sub wait_for_port {
    my ($port, $opts) = @_;
    $opts //= {};
    # Default: 60 seconds (300 tries x 0.2s). The JIT compile of a
    # full Hypersonic server (all event backends + TLS + HTTP/2 +
    # WebSocket + SSE + streaming) can take >30s on a debugging-perl
    # smoke host (gcc -O0 -g), and on older perls/hosts even longer.
    # The previous 5s default caused t/0035-e2e-streaming.t bailouts
    # in CPAN tester reports on the k93msid host for perl 5.12..5.42.
    my $max_tries = $opts->{tries} // 300;
    my $sleep     = $opts->{sleep} // 0.2;

    # Enforce a MINIMUM total wait of 60 wallclock seconds regardless
    # of what the caller asked for. Most test files in this dist pass
    # `tries => 50` which (at 0.2s/try) is only 10s; that's nowhere
    # near enough for the JIT compile on a slow/DEBUGGING smoker.
    # Also scale by PERL_TEST_TIME_OUT_FACTOR (some smokers set this
    # to 3 to indicate "I'm a slow box, give tests 3x the budget").
    # CPAN tester reports for Hypersonic 0.16 from k93msid (perl
    # 5.34.1 DEBUGGING) and a Raspberry Pi (armv6, perl 5.42.2) both
    # bailed because individual tests had `tries => 50` / `tries => 100`
    # hard-coded. Rather than rewrite every test, raise the floor here.
    my $factor = $ENV{PERL_TEST_TIME_OUT_FACTOR};
    $factor = 1 unless defined $factor && $factor =~ /^\d+(?:\.\d+)?$/ && $factor > 0;
    my $min_seconds = 60 * $factor;
    my $asked_seconds = $max_tries * $sleep;
    if ($asked_seconds < $min_seconds) {
        $max_tries = int($min_seconds / $sleep) + 1;
    }

    for my $try (1 .. $max_tries) {
        my $sock = IO::Socket::INET->new(
            PeerAddr => '127.0.0.1',
            PeerPort => $port,
            Proto    => 'tcp',
            Timeout  => 0.1,
        );
        if ($sock) { close $sock; return 1; }
        select undef, undef, undef, $sleep;
    }

    # Server didn't come up - surface as much detail as we can.
    if ($opts->{pid}) {
        my $waited = waitpid($opts->{pid}, 1);  # WNOHANG
        if ($waited == $opts->{pid}) {
            Test::More::diag("Child server exited prematurely "
                . "(wstat=$?, exit=" . ($? >> 8)
                . ", signal=" . ($? & 0x7f) . ")");
        } else {
            Test::More::diag("Child server still alive but not "
                . "listening on port $port; killing");
            kill 'TERM', $opts->{pid};
        }
    }
    if ($opts->{log}) {
        diag_child_log($opts->{log});
    }
    return 0;
}

# diag_child_log($path) - dump captured child output via diag,
# truncating absurdly long output so a runaway child can't drown the
# test summary.
sub diag_child_log {
    my $path = shift;
    return unless -e $path;
    open my $fh, '<', $path or do {
        Test::More::diag("(could not read child log $path: $!)");



( run in 1.025 second using v1.01-cache-2.11-cpan-140bd7fdf52 )