]> git.ipfire.org Git - thirdparty/public-inbox.git/commitdiff
lei import|tag|rm: support --commit-delay=SECONDS
authorEric Wong <e@80x24.org>
Wed, 11 Oct 2023 07:20:57 +0000 (07:20 +0000)
committerEric Wong <e@80x24.org>
Wed, 11 Oct 2023 22:10:51 +0000 (22:10 +0000)
Delayed commits  allows users to trade off immediate safety for
throughput and reduced storage wear when running multiple
discreet commands.

This feature is currently useful for providing a way to make
t/lei-store-fail.t reliable and for ensuring `lei blob' can
retrieve messages which have not yet been committed.

In the future, it'll also be useful for the FUSE layer to batch
git activity.

lib/PublicInbox/LEI.pm
lib/PublicInbox/LeiStore.pm
t/lei-import.t
t/lei-store-fail.t
t/lei-tag.t

index e2b3c0d92156690ff595f9d7a0bbff5eecf55b43..af39f8aff2198551e0e1189fd5cb4bf5be3e60e9 100644 (file)
@@ -231,13 +231,13 @@ our %CMD = ( # sorted in order of importance/use:
 'rm' => [ '--stdin|LOCATION...',
        'remove a message from the index and prevent reindexing',
        'stdin|', # /|\z/ must be first for lone dash
-       qw(in-format|F=s lock=s@), @net_opt, @c_opt ],
+       qw(in-format|F=s lock=s@ commit-delay=i), @net_opt, @c_opt ],
 'plonk' => [ '--threads|--from=IDENT',
        'exclude mail matching From: or threads from non-Message-ID searches',
        qw(stdin| threads|t from|f=s mid=s oid=s), @c_opt ],
-'tag' => [ 'KEYWORDS... LOCATION...|--stdin',
+tag => [ 'KEYWORDS... LOCATION...|--stdin',
        'set/unset keywords and/or labels on message(s)',
-       qw(stdin| in-format|F=s input|i=s@ oid=s@ mid=s@),
+       qw(stdin| in-format|F=s input|i=s@ oid=s@ mid=s@ commit-delay=i),
        @net_opt, @c_opt, pass_through('-kw:foo for delete') ],
 
 'purge-mailsource' => [ 'LOCATION|--all',
@@ -262,10 +262,11 @@ our %CMD = ( # sorted in order of importance/use:
        qw(in-format|F=s kw! offset=i recursive|r exclude=s include|I=s
        verbose|v+ incremental!), @net_opt, # mainly for --proxy=
         @c_opt ],
-'import' => [ 'LOCATION...|--stdin [LABELS...]',
+import => [ 'LOCATION...|--stdin [LABELS...]',
        'one-time import/update from URL or filesystem',
        qw(stdin| offset=i recursive|r exclude=s include|I=s new-only
-       lock=s@ in-format|F=s kw! verbose|v+ incremental! mail-sync!),
+       lock=s@ in-format|F=s kw! verbose|v+ incremental! mail-sync!
+       commit-delay=i),
        @net_opt, @c_opt ],
 'forget-mail-sync' => [ 'LOCATION...',
        'forget sync information for a mail folder', @c_opt ],
@@ -1539,10 +1540,14 @@ sub sto_done_request {
        my ($lei, $wq) = @_;
        return unless $lei->{sto} && $lei->{sto}->{-wq_s1};
        local $current_lei = $lei;
-       my $s = ($wq ? $wq->{lei_sock} : undef) // $lei->{sock};
-       my $errfh = $lei->{2} // *STDERR{GLOB};
-       my @io = $s ? ($errfh, $s) : ($errfh);
-       eval { $lei->{sto}->wq_io_do('done', \@io) };
+       if (my $n = $lei->{opt}->{'commit-delay'}) {
+               eval { $lei->{sto}->wq_do('schedule_commit', $n) };
+       } else {
+               my $s = ($wq ? $wq->{lei_sock} : undef) // $lei->{sock};
+               my $errfh = $lei->{2} // *STDERR{GLOB};
+               my @io = $s ? ($errfh, $s) : ($errfh);
+               eval { $lei->{sto}->wq_io_do('done', \@io) };
+       }
        warn($@) if $@;
 }
 
index 9c07af14eaebe95b45c9593f9a080fc1e8a39ad7..aebb85a9f75267e4e9ebcc0f65096bf383d13f60 100644 (file)
@@ -34,6 +34,7 @@ use Sys::Syslog qw(syslog openlog);
 use Errno qw(EEXIST ENOENT);
 use PublicInbox::Syscall qw(rename_noreplace);
 use PublicInbox::LeiStoreErr;
+use PublicInbox::DS qw(add_uniq_timer);
 
 sub new {
        my (undef, $dir, $opt) = @_;
@@ -113,6 +114,11 @@ sub cat_blob {
        $self->{im} ? $self->{im}->cat_blob($oid) : undef;
 }
 
+sub schedule_commit {
+       my ($self, $sec) = @_;
+       add_uniq_timer($self->{priv_eidx}->{topdir}, $sec, \&done, $self);
+}
+
 # follows the stderr file
 sub _tail_err {
        my ($self) = @_;
index 8b09d3aa3ead8f1cd9ad808e50c7fea8ec5a615a..b2c1de9b9f9033f4bf366cfd021bd5866fdcc45e 100644 (file)
@@ -2,6 +2,7 @@
 # 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 PublicInbox::DS qw(now);
 use autodie qw(open close);
 test_lei(sub {
 ok(!lei(qw(import -F bogus), 't/plack-qp.eml'), 'fails with bogus format');
@@ -141,6 +142,18 @@ $res = json_utf8->decode($lei_out);
 is_deeply($res->[0]->{kw}, [qw(answered flagged seen)], 'keyword added');
 is_deeply($res->[0]->{L}, [qw(boombox inbox)], 'labels preserved');
 
+lei_ok qw(import --commit-delay=1 +L:bin -F eml t/data/binary.patch);
+lei_ok 'ls-label';
+unlike($lei_out, qr/\bbin\b/, 'commit-delay delays label');
+my $end = now + 10;
+my $n = 1;
+diag 'waiting for lei/store commit...';
+do {
+       tick $n;
+       $n = 0.1;
+} until (!lei('ls-label') || $lei_out =~ /\bbin\b/ || now > $end);
+like($lei_out, qr/\bbin\b/, 'commit-delay eventually commits');
+
 # see t/lei_to_mail.t for "import -F mbox*"
 });
 done_testing;
index fb0f2b75f89c82753c5e381b58e4e2e62b0862b7..c2f03148878056669f5ca3aed29e9b087127f6a9 100644 (file)
@@ -9,8 +9,11 @@ use Fcntl qw(SEEK_SET);
 use File::Path qw(remove_tree);
 
 my $start_home = $ENV{HOME}; # bug guard
+my $utf8_oid = '9bf1002c49eb075df47247b74d69bcd555e23422';
 test_lei(sub {
        lei_ok qw(import -q t/plack-qp.eml); # start the store
+       ok(!lei(qw(blob --mail), $utf8_oid), 't/utf8.eml not imported, yet');
+
        my $opt;
        pipe($opt->{0}, my $in_w);
        open $opt->{1}, '+>', undef;
@@ -20,27 +23,30 @@ test_lei(sub {
        my $tp = start_script($cmd, undef, $opt);
        close $opt->{0};
        $in_w->autoflush(1);
-       for (1..500) { # need to fill up 64k read buffer
-               print $in_w <<EOM or xbail "print $!";
+       print $in_w <<EOM or xbail "print: $!";
 From k\@y Fri Oct  2 00:00:00 1993
 From: <k\@example.com>
 Date: Sat, 02 Oct 2010 00:00:00 +0000
 Subject: hi
-Message-ID: <$_\@t>
+Message-ID: <0\@t>
 
 will this save?
 EOM
-       }
-       tick 0.2; # XXX ugh, this is so hacky
+       # import another message w/ delay while mboxrd import is still running
+       lei_ok qw(import -q --commit-delay=300 t/utf8.eml);
+       lei_ok qw(blob --mail), $utf8_oid,
+               \'blob immediately available despite --commit-delay';
+       lei_ok qw(q m:testmessage@example.com);
+       is($lei_out, "[null]\n", 'delayed commit is unindexed');
 
-       # make sto_done_request fail:
+       # make immediate ->sto_done_request fail from mboxrd import:
        remove_tree("$ENV{HOME}/.local/share/lei/store");
        # subsequent lei commands are undefined behavior,
        # but we need to make sure the current lei command fails:
 
        close $in_w; # should trigger ->done
        $tp->join;
-       isnt($?, 0, 'lei import error code set on failure');
+       isnt($?, 0, 'lei import -F mboxrd error code set on failure');
        is(-s $opt->{1}, 0, 'nothing in stdout');
        isnt(-s $opt->{2}, 0, 'stderr not empty');
        seek($opt->{2}, 0, SEEK_SET);
index cccf0af6b245a308c5720e462f93ba08efeaa92e..7278dfcdc22b4fc3e46ce7f65abac65be828d918 100644 (file)
@@ -1,9 +1,10 @@
 #!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 v5.12; use PublicInbox::TestCommon;
 require_git 2.6;
 require_mods(qw(json DBD::SQLite Xapian));
+use PublicInbox::DS qw(now);
 my ($ro_home, $cfg_path) = setup_public_inboxes;
 my $check_kw = sub {
        my ($exp, %opt) = @_;
@@ -104,5 +105,17 @@ test_lei(sub {
        lei_ok qw(tag +L:nope -F eml t/data/binary.patch);
        like $lei_err, qr/\b1 unimported messages/, 'noted unimported'
                or diag $lei_err;
+
+       lei_ok qw(tag -F eml --commit-delay=1 t/utf8.eml +L:utf8);
+       lei_ok 'ls-label';
+       unlike($lei_out, qr/\butf8\b/, 'commit-delay delays label');
+       my $end = now + 10;
+       my $n = 1;
+       diag 'waiting for lei/store commit...';
+       do {
+               tick $n;
+               $n = 0.1;
+       } until (!lei('ls-label') || $lei_out =~ /\butf8\b/ || now > $end);
+       like($lei_out, qr/\butf8\b/, 'commit-delay eventually commits');
 });
 done_testing;