| #!perl -w | 
 | # Copyright (C) all contributors <meta@public-inbox.org> | 
 | # License: AGPL-3.0+ <https://www.gnu.org/licenses/agpl-3.0.txt> | 
 | use v5.12; | 
 | use PublicInbox::TestCommon; | 
 | use autodie; | 
 | use Config; | 
 | use PublicInbox::Spawn qw(popen_rd); | 
 | require_mods('DBD::SQLite'); | 
 | use_ok 'PublicInbox::Msgmap'; | 
 | my ($tmpdir, $for_destroy) = tmpdir(); | 
 | my $f = "$tmpdir/msgmap.sqlite3"; | 
 | my $d = PublicInbox::Msgmap->new_file($f, 1); | 
 |  | 
 | my %mid2num; | 
 | my %num2mid; | 
 | my @mids = qw(a@b c@d e@f g@h aa@bb aa@cc); | 
 | is_deeply([$d->minmax], [0,0], "zero min max on new DB"); | 
 |  | 
 | foreach my $mid (@mids) { | 
 | 	my $n = $d->mid_insert($mid); | 
 | 	ok($n, "mid $mid inserted"); | 
 | 	$mid2num{$mid} = $n; | 
 | 	$num2mid{$n} = $mid; | 
 | } | 
 |  | 
 | $@ = undef; | 
 | my $ret = $d->mid_insert('a@b'); | 
 | is($ret, undef, 'duplicate mid_insert in undef result'); | 
 | is($d->num_for('a@b'), $mid2num{'a@b'}, 'existing number not clobbered'); | 
 | my $next = (sort(keys %num2mid))[-1]; | 
 | is($d->mid_insert('ok@unique'), $next + 1, | 
 | 	'got expected num after failing mid_insert'); | 
 |  | 
 | foreach my $n (keys %num2mid) { | 
 | 	is($d->mid_for($n), $num2mid{$n}, "num:$n maps correctly"); | 
 | } | 
 | foreach my $mid (@mids) { | 
 | 	is($d->num_for($mid), $mid2num{$mid}, "mid:$mid maps correctly"); | 
 | } | 
 |  | 
 | is(undef, $d->last_commit, "last commit not set"); | 
 | my $lc = 'deadbeef' x 5; | 
 | is(undef, $d->last_commit($lc), 'previous last commit (undef) returned'); | 
 | is($lc, $d->last_commit, 'last commit was set correctly'); | 
 |  | 
 | my $nc = 'deaddead' x 5; | 
 | is($lc, $d->last_commit($nc), 'returned previously set commit'); | 
 | is($nc, $d->last_commit, 'new commit was set correctly'); | 
 |  | 
 | is($d->mid_delete('a@b'), 1, 'deleted a@b'); | 
 | is($d->mid_delete('a@b') + 0, 0, 'delete again returns zero'); | 
 | is(undef, $d->num_for('a@b'), 'num_for fails on deleted msg'); | 
 | $d = undef; | 
 |  | 
 | ok($d = PublicInbox::Msgmap->new_file($f, 1), 'idempotent DB creation'); | 
 | my ($min, $max) = $d->minmax; | 
 | ok($min > 0, "article min OK"); | 
 | ok($max > 0 && $max < 10, "article max OK"); | 
 | ok($min < $max, "article counts OK"); | 
 |  | 
 | my $orig = $d->mid_insert('spam@1'); | 
 | $d->mid_delete('spam@1'); | 
 | is($d->mid_insert('spam@2'), 1 + $orig, "last number not recycled"); | 
 |  | 
 | my $tmp = $d->tmp_clone($tmpdir); | 
 | is_deeply([$d->minmax], [$tmp->minmax], 'Cloned temporary DB matches'); | 
 | ok($tmp->mid_delete('spam@2'), 'temporary DB is writable'); | 
 |  | 
 | is(eval { | 
 | 	$tmp->atfork_prepare; | 
 | 	$tmp->atfork_parent; | 
 | 	'ok' | 
 | }, 'ok', 'atfork_* work on tmp_clone'); | 
 |  | 
 | SKIP: { | 
 | 	my $strace = strace_inject(1); | 
 | 	open my $fh, '>', my $trace = "$tmpdir/trace.out"; | 
 | 	my $rd = popen_rd([ $strace, '-p', $$, '-o', $trace, | 
 | 		'-e', 'inject=pwrite64:error=ENOSPC'], undef, { 2 => 1 }); | 
 | 	$rd->poll_in(10_000) or die 'strace not ready'; | 
 | 	is eval { $d->mid_insert('this-better-trigger-ENOSPC@error') }, | 
 | 		undef, 'insert fails w/ ENOSPC'; | 
 | 	like $@, qr/ disk is full/, '$@ reports ENOSPC'; | 
 | 	kill 'TERM', $rd->attached_pid; | 
 | 	$rd->close; | 
 | } | 
 |  | 
 | done_testing; |