aboutsummaryrefslogtreecommitdiffhomepage
path: root/lib
diff options
context:
space:
mode:
authorGravatar Keith Packard <keithp@keithp.com>2009-11-20 23:15:07 -0800
committerGravatar Carl Worth <cworth@cworth.org>2009-11-23 06:33:54 +0100
commit53f8cc565126db4a003dbfc02850d2bf3b260636 (patch)
tree9797fb6cd1acfc5070bc7ede019772f356fee522 /lib
parent43daa6f070f962959bf26fa49a860f528b2bbfa1 (diff)
Add 'notmuch count' command to show the count of matching messages
Getting the count of matching threads or messages is a fairly expensive operation. Xapian provides a very efficient mechanism that returns an approximate value, so use that for this new command. This returns the number of matching messages, not threads, as that is cheap to compute. Signed-off-by: Keith Packard <keithp@keithp.com>
Diffstat (limited to 'lib')
-rw-r--r--lib/notmuch.h8
-rw-r--r--lib/query.cc52
2 files changed, 60 insertions, 0 deletions
diff --git a/lib/notmuch.h b/lib/notmuch.h
index a61cd020..260cc22d 100644
--- a/lib/notmuch.h
+++ b/lib/notmuch.h
@@ -468,6 +468,14 @@ notmuch_threads_advance (notmuch_threads_t *threads);
void
notmuch_threads_destroy (notmuch_threads_t *threads);
+/* Return an estimate of the number of messages matching a search
+ *
+ * This function performs a search and returns Xapian's best
+ * guess as to number of matching messages.
+ */
+unsigned
+notmuch_query_count_messages (notmuch_query_t *query);
+
/* Get the thread ID of 'thread'.
*
* The returned string belongs to 'thread' and as such, should not be
diff --git a/lib/query.cc b/lib/query.cc
index 86167352..686d75f9 100644
--- a/lib/query.cc
+++ b/lib/query.cc
@@ -279,3 +279,55 @@ notmuch_threads_destroy (notmuch_threads_t *threads)
{
talloc_free (threads);
}
+
+unsigned
+notmuch_query_count_messages (notmuch_query_t *query)
+{
+ notmuch_database_t *notmuch = query->notmuch;
+ const char *query_string = query->query_string;
+ Xapian::doccount count;
+
+ try {
+ Xapian::Enquire enquire (*notmuch->xapian_db);
+ Xapian::Query mail_query (talloc_asprintf (query, "%s%s",
+ _find_prefix ("type"),
+ "mail"));
+ Xapian::Query string_query, final_query;
+ Xapian::MSet mset;
+ unsigned int flags = (Xapian::QueryParser::FLAG_BOOLEAN |
+ Xapian::QueryParser::FLAG_PHRASE |
+ Xapian::QueryParser::FLAG_LOVEHATE |
+ Xapian::QueryParser::FLAG_BOOLEAN_ANY_CASE |
+ Xapian::QueryParser::FLAG_WILDCARD |
+ Xapian::QueryParser::FLAG_PURE_NOT);
+
+ if (strcmp (query_string, "") == 0) {
+ final_query = mail_query;
+ } else {
+ string_query = notmuch->query_parser->
+ parse_query (query_string, flags);
+ final_query = Xapian::Query (Xapian::Query::OP_AND,
+ mail_query, string_query);
+ }
+
+ enquire.set_weighting_scheme(Xapian::BoolWeight());
+ enquire.set_docid_order(Xapian::Enquire::ASCENDING);
+
+#if DEBUG_QUERY
+ fprintf (stderr, "Final query is:\n%s\n", final_query.get_description().c_str());
+#endif
+
+ enquire.set_query (final_query);
+
+ mset = enquire.get_mset (0, notmuch->xapian_db->get_doccount ());
+
+ count = mset.get_matches_estimated();
+
+ } catch (const Xapian::Error &error) {
+ fprintf (stderr, "A Xapian exception occurred: %s\n",
+ error.get_msg().c_str());
+ fprintf (stderr, "Query string was: %s\n", query->query_string);
+ }
+
+ return count;
+}