]> Sergey Matveev's repositories - public-inbox.git/commitdiff
lei q: -tt marks direct hits as "flagged"
authorEric Wong <e@80x24.org>
Thu, 25 Feb 2021 10:11:06 +0000 (10:11 +0000)
committerEric Wong <e@80x24.org>
Fri, 26 Feb 2021 02:35:50 +0000 (02:35 +0000)
This can be used to quickly distinguish messages which were
direct hits when doing thread expansion vs messages that
were merely part of the same thread.

This is NOT mairix-derived behavior, but I occasionally found
it useful when looking at results in an MUA to know whether
a message was a direct hit or not.

This makes "-t" consistent with non-"-t" cases as far as keyword
reading goes.

Documentation/lei-q.pod
MANIFEST
lib/PublicInbox/LEI.pm
lib/PublicInbox/LeiXSearch.pm
t/lei-q-thread.t [new file with mode: 0644]

index 75fdc613579cdc18d28f5589b4038eda65fb850b..0959beac385048414887135664bf16e616fd56fb 100644 (file)
@@ -79,6 +79,14 @@ Augment output destination instead of clobbering it.
 
 Return all messages in the same thread as the actual match(es).
 
+Using this twice (C<-tt>) sets the C<flagged> (AKA "important")
+on messages which were actual messages.  This is useful to distinguish
+messages which were direct hits from messages which were merely part
+of the same thread.
+
+TODO: Warning: this flag may become persistent and saved in
+lei/store unless an MUA unflags it!  (Behavior undecided)
+
 =item -d STRATEGY, --dedupe=STRATEGY
 
 Strategy for deduplicating messages: C<content>, C<oid>, C<mid>, or
index adbd108fbd21d8d00690e8698e732dcb49ac9af3..9cf33d487fb8d30e9d3721d7033dfd67f0336e5e 100644 (file)
--- a/MANIFEST
+++ b/MANIFEST
@@ -373,6 +373,7 @@ t/lei-import-nntp.t
 t/lei-import.t
 t/lei-mirror.t
 t/lei-q-remote-import.t
+t/lei-q-thread.t
 t/lei.t
 t/lei_dedupe.t
 t/lei_external.t
index 8eb96e78bc1a12593b15eb6de9175cdb91fc65eb..8825fa4379d87006880a715b64abe9621060cca1 100644 (file)
@@ -109,7 +109,7 @@ sub index_opt {
 # command => [ positional_args, 1-line description, Getopt::Long option spec ]
 our %CMD = ( # sorted in order of importance/use:
 'q' => [ '--stdin|SEARCH_TERMS...', 'search for messages matching terms', qw(
-       save-as=s output|mfolder|o=s format|f=s dedupe|d=s threads|t augment|a
+       save-as=s output|mfolder|o=s format|f=s dedupe|d=s threads|t+ augment|a
        sort|s=s reverse|r offset=i remote! local! external! pretty
        include|I=s@ exclude=s@ only=s@ jobs|j=s globoff|g stdin|
        import-remote!
@@ -233,7 +233,7 @@ my %OPTDESC = (
 'dedupe|d=s' => ['STRATEGY|content|oid|mid|none',
                'deduplication strategy'],
 'show  threads|t' => 'display entire thread a message belongs to',
-'q     threads|t' =>
+'q     threads|t+' =>
        'return all messages in the same threads as the actual match(es)',
 'alert=s@' => ['CMD,:WINCH,:bell,<any command>',
        'run command(s) or perform ops when done writing to output ' .
index 2d399653591d1323749ff615929903ffcaa2bd6c..eb015978708024cfa1884ba9a98cf67727a6e20f 100644 (file)
@@ -66,6 +66,13 @@ sub remotes { @{$_[0]->{remotes} // []} }
 # called by PublicInbox::Search::xdb
 sub xdb_shards_flat { @{$_[0]->{shards_flat} // []} }
 
+sub mitem_kw ($$;$) {
+       my ($smsg, $mitem, $flagged) = @_;
+       my $kw = xap_terms('K', $mitem->get_document);
+       $kw->{flagged} = 1 if $flagged;
+       $smsg->{kw} = [ sort keys %$kw ];
+}
+
 # like over->get_art
 sub smsg_for {
        my ($self, $mitem) = @_;
@@ -76,10 +83,7 @@ sub smsg_for {
        my $num = int(($docid - 1) / $nshard) + 1;
        my $ibx = $self->{shard2ibx}->[$shard];
        my $smsg = $ibx->over->get_art($num);
-       if (ref($ibx->can('msg_keywords'))) {
-               my $kw = xap_terms('K', $mitem->get_document);
-               $smsg->{kw} = [ sort keys %$kw ];
-       }
+       mitem_kw($smsg, $mitem) if $ibx->can('msg_keywords');
        $smsg->{docid} = $docid;
        $smsg;
 }
@@ -143,6 +147,8 @@ sub query_thread_mset { # for --threads
        my $mo = { %{$lei->{mset_opt}} };
        my $mset;
        my $each_smsg = $lei->{ovv}->ovv_each_smsg_cb($lei, $ibxish);
+       my $can_kw = !!$ibxish->can('msg_keywords');
+       my $fl = $lei->{opt}->{threads} > 1;
        do {
                $mset = $srch->mset($mo->{qstr}, $mo);
                mset_progress($lei, $desc, $mset->size,
@@ -156,6 +162,13 @@ sub query_thread_mset { # for --threads
                                my $smsg = $over->get_art($n) or next;
                                wait_startq($lei);
                                my $mitem = delete $n2item{$smsg->{num}};
+                               if ($mitem) {
+                                       if ($can_kw) {
+                                               mitem_kw($smsg, $mitem, $fl);
+                                       } else {
+                                               $smsg->{kw} = [ 'flagged' ];
+                                       }
+                               }
                                $each_smsg->($smsg, $mitem);
                        }
                        @{$ctx->{xids}} = ();
diff --git a/t/lei-q-thread.t b/t/lei-q-thread.t
new file mode 100644 (file)
index 0000000..66db28a
--- /dev/null
@@ -0,0 +1,47 @@
+#!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;
+require_git 2.6;
+require_mods(qw(json DBD::SQLite Search::Xapian));
+use PublicInbox::LeiToMail;
+my ($ro_home, $cfg_path) = setup_public_inboxes;
+test_lei(sub {
+       my $eml = eml_load('t/utf8.eml');
+       my $buf = PublicInbox::LeiToMail::eml2mboxrd($eml, { kw => ['seen'] });
+       lei_ok([qw(import -F mboxrd -)], undef, { 0 => $buf, %$lei_opt });
+
+       lei_ok qw(q -t m:testmessage@example.com);
+       my $res = json_utf8->decode($lei_out);
+       is_deeply($res->[0]->{kw}, [ 'seen' ], 'q -t sets keywords');
+
+       $eml = eml_load('t/utf8.eml');
+       $eml->header_set('References', $eml->header('Message-ID'));
+       $eml->header_set('Message-ID', '<a-reply@miss>');
+       $buf = PublicInbox::LeiToMail::eml2mboxrd($eml, { kw => ['draft'] });
+       lei_ok([qw(import -F mboxrd -)], undef, { 0 => $buf, %$lei_opt });
+
+       lei_ok qw(q -t m:testmessage@example.com);
+       $res = json_utf8->decode($lei_out);
+       is(scalar(@$res), 3, 'got 2 results');
+       pop @$res;
+       my %m = map { $_->{'m'} => $_ } @$res;
+       is_deeply($m{'<testmessage@example.com>'}->{kw}, ['seen'],
+               'flag set in direct hit');
+       'TODO' or is_deeply($m{'<a-reply@miss>'}->{kw}, ['draft'],
+               'flag set in thread hit');
+
+       lei_ok qw(q -t -t m:testmessage@example.com);
+       $res = json_utf8->decode($lei_out);
+       is(scalar(@$res), 3, 'got 2 results with -t -t');
+       pop @$res;
+       %m = map { $_->{'m'} => $_ } @$res;
+       is_deeply($m{'<testmessage@example.com>'}->{kw}, ['flagged', 'seen'],
+               'flagged set in direct hit');
+       'TODO' or is_deeply($m{'<testmessage@example.com>'}->{kw}, ['draft'],
+               'flagged set in direct hit');
+       lei_ok qw(q -t -t m:testmessage@example.com --only), "$ro_home/t2";
+       $res = json_utf8->decode($lei_out);
+       is_deeply($res->[0]->{kw}, [ 'flagged' ], 'flagged set on external');
+});
+done_testing;