1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
| | #!perl -w
# Copyright (C) 2021 all contributors <meta@public-inbox.org>
# License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt>
use strict; use v5.10.1; use PublicInbox::TestCommon;
use autodie qw(chdir);
use POSIX qw(_exit);
use PublicInbox::DS qw(now);
use Errno qw(EAGAIN);
use PublicInbox::OnDestroy;
use_ok 'PublicInbox::MboxLock';
my ($tmpdir, $for_destroy) = tmpdir();
my $f = "$tmpdir/f";
my $mbl = PublicInbox::MboxLock->acq($f, 1, ['dotlock']);
ok(-f "$f.lock", 'dotlock created');
undef $mbl;
ok(!-f "$f.lock", 'dotlock gone');
$mbl = PublicInbox::MboxLock->acq($f, 1, ['none']);
ok(!-f "$f.lock", 'no dotlock with none');
undef $mbl;
{
opendir my $cur, '.' or BAIL_OUT $!;
my $od = on_destroy \&chdir, $cur;
chdir $tmpdir;
my $abs = "$tmpdir/rel.lock";
my $rel = PublicInbox::MboxLock->acq('rel', 1, ['dotlock']);
chdir '/';
ok(-f $abs, 'lock with abs path created');
undef $rel;
ok(!-f $abs, 'lock gone despite being in the wrong dir');
}
eval {
PublicInbox::MboxLock->acq($f, 1, ['bogus']);
fail "should not succeed with `bogus'";
};
ok($@, "fails on `bogus' lock method");
eval {
PublicInbox::MboxLock->acq($f, 1, ['timeout=1']);
fail "should not succeed with only timeout";
};
ok($@, "fails with only `timeout=' and no lock method");
my $defaults = PublicInbox::MboxLock->defaults;
is(ref($defaults), 'ARRAY', 'default lock methods');
my $test_rw_lock = sub {
my ($func) = @_;
my $m = ["$func,timeout=0.000001"];
for my $i (1..2) {
pipe(my ($r, $w)) or BAIL_OUT "pipe: $!";
my $t0 = now;
my $pid = fork // BAIL_OUT "fork $!";
if ($pid == 0) {
eval { PublicInbox::MboxLock->acq($f, 1, $m) };
my $err = $@;
syswrite $w, "E: $err";
_exit($err ? 0 : 1);
}
undef $w;
waitpid($pid, 0);
is($?, 0, "$func r/w lock behaved as expected #$i");
my $d = now - $t0;
ok($d < 1, "$func r/w timeout #$i") or diag "elapsed=$d";
my $err = do { local $/; <$r> };
$! = EAGAIN;
my $msg = "$!";
like($err, qr/\Q$msg\E/, "got EAGAIN in child #$i");
}
};
my $test_ro_lock = sub {
my ($func) = @_;
for my $i (1..2) {
my $t0 = now;
my $pid = fork // BAIL_OUT "fork $!";
if ($pid == 0) {
eval { PublicInbox::MboxLock->acq($f, 0, [ $func ]) };
_exit($@ ? 1 : 0);
}
waitpid($pid, 0);
is($?, 0, "$func ro lock behaved as expected #$i");
my $d = now - $t0;
ok($d < 1, "$func timeout respected #$i") or diag "elapsed=$d";
}
};
SKIP: {
grep(/fcntl/, @$defaults) or skip 'File::FcntlLock not available', 1;
my $top = PublicInbox::MboxLock->acq($f, 1, $defaults);
ok($top, 'fcntl lock acquired');
$test_rw_lock->('fcntl');
undef $top;
$top = PublicInbox::MboxLock->acq($f, 0, $defaults);
ok($top, 'fcntl read lock acquired');
$test_ro_lock->('fcntl');
}
$mbl = PublicInbox::MboxLock->acq($f, 1, ['flock']);
ok($mbl, 'flock acquired');
$test_rw_lock->('flock');
undef $mbl;
$mbl = PublicInbox::MboxLock->acq($f, 0, ['flock']);
$test_ro_lock->('flock');
done_testing;
|