# Copyright (C) 2016 all contributors
# License: AGPL-3.0+
use strict;
use warnings;
use Test::More;
foreach my $mod (qw(Plack::Util Plack::Builder Danga::Socket
HTTP::Date HTTP::Status)) {
eval "require $mod";
plan skip_all => "$mod missing for httpd.t" if $@;
}
use File::Temp qw/tempdir/;
use Cwd qw/getcwd/;
use IO::Socket;
use Fcntl qw(FD_CLOEXEC F_SETFD F_GETFD);
use Socket qw(SO_KEEPALIVE IPPROTO_TCP TCP_NODELAY);
# FIXME: too much setup
my $tmpdir = tempdir('pi-httpd-XXXXXX', TMPDIR => 1, CLEANUP => 1);
my $home = "$tmpdir/pi-home";
my $err = "$tmpdir/stderr.log";
my $out = "$tmpdir/stdout.log";
my $maindir = "$tmpdir/main.git";
my $group = 'test-httpd';
my $addr = $group . '@example.com';
my $cfgpfx = "publicinbox.$group";
my $httpd = 'blib/script/public-inbox-httpd';
my $init = 'blib/script/public-inbox-init';
my %opts = (
LocalAddr => '127.0.0.1',
ReuseAddr => 1,
Proto => 'tcp',
Type => SOCK_STREAM,
Listen => 1024,
);
my $sock = IO::Socket::INET->new(%opts);
my $pid;
use_ok 'PublicInbox::Git';
use_ok 'PublicInbox::Import';
use_ok 'Email::MIME';
END { kill 'TERM', $pid if defined $pid };
{
local $ENV{HOME} = $home;
ok(!system($init, $group, $maindir, 'http://example.com/', $addr),
'init ran properly');
# ensure successful message delivery
{
my $mime = Email::MIME->new(<
To: You
Cc: $addr
Message-Id:
Subject: hihi
Date: Thu, 01 Jan 1970 06:06:06 +0000
nntp
EOF
my $git = PublicInbox::Git->new($maindir);
my $im = PublicInbox::Import->new($git, 'test', $addr);
$im->add($mime);
$im->done($mime);
}
ok($sock, 'sock created');
$! = 0;
my $fl = fcntl($sock, F_GETFD, 0);
ok(! $!, 'no error from fcntl(F_GETFD)');
is($fl, FD_CLOEXEC, 'cloexec set by default (Perl behavior)');
$pid = fork;
if ($pid == 0) {
use POSIX qw(dup2);
# pretend to be systemd
fcntl($sock, F_SETFD, $fl &= ~FD_CLOEXEC);
dup2(fileno($sock), 3) or die "dup2 failed: $!\n";
$ENV{LISTEN_PID} = $$;
$ENV{LISTEN_FDS} = 1;
exec $httpd, "--stdout=$out", "--stderr=$err";
die "FAIL: $!\n";
}
ok(defined $pid, 'forked httpd process successfully');
$! = 0;
fcntl($sock, F_SETFD, $fl |= FD_CLOEXEC);
ok(! $!, 'no error from fcntl(F_SETFD)');
my $host = $sock->sockhost;
my $port = $sock->sockport;
my $conn = IO::Socket::INET->new(PeerAddr => $host,
PeerPort => $port,
Proto => 'tcp',
Type => SOCK_STREAM);
ok($conn, 'connected');
ok($conn->write("GET / HTTP/1.0\r\n\r\n"), 'wrote data to socket');
{
my $buf;
ok($conn->read($buf, 4096), 'read some bytes');
like($buf, qr!\AHTTP/1\.[01] 404\b!, 'got 404 response');
is($conn->read($buf, 1), 0, "EOF");
}
is(system(qw(git clone -q --mirror),
"http://$host:$port/$group", "$tmpdir/clone.git"),
0, 'smart clone successful');
# ensure dumb cloning works, too:
is(system('git', "--git-dir=$maindir",
qw(config http.uploadpack false)),
0, 'disable http.uploadpack');
is(system(qw(git clone -q --mirror),
"http://$host:$port/$group", "$tmpdir/dumb.git"),
0, 'clone successful');
ok(kill('TERM', $pid), 'killed httpd');
$pid = undef;
waitpid(-1, 0);
is(system('git', "--git-dir=$tmpdir/clone.git",
qw(fsck --no-verbose)), 0,
'fsck on cloned directory successful');
}
done_testing();
1;