* mu-query.*: fix querying after recent changes

This commit is contained in:
Dirk-Jan C. Binnema 2009-12-05 20:10:58 +02:00
parent a35c27784e
commit c16cea11c9
3 changed files with 64 additions and 82 deletions

View File

@ -28,14 +28,49 @@
#include "mu-msg-xapian.h" #include "mu-msg-xapian.h"
#include "mu-msg-xapian-priv.hh" #include "mu-msg-xapian-priv.hh"
struct _MuQueryXapian { static void add_prefix (const MuMsgField* field, Xapian::QueryParser* qparser);
class _MuQueryXapian {
public:
_MuQueryXapian (const char *path) : _db(Xapian::Database(path)) {
_qparser.set_database(_db);
_qparser.set_default_op(Xapian::Query::OP_OR);
_qparser.set_stemming_strategy
(Xapian::QueryParser::STEM_SOME);
memset (_sorters, 0, sizeof(_sorters));
mu_msg_field_foreach ((MuMsgFieldForEachFunc)add_prefix,
(gconstpointer)&_qparser);
}
~_MuQueryXapian () {
try {
for (int i = 0; i != MU_MSG_FIELD_TYPE_NUM; ++i)
delete _sorters[i];
} catch (...) {
g_warning ("%s: caught exception", __FUNCTION__);
}
}
_MuQueryXapian (const char *path) : Xapian::Query get_query (const char* searchexpr) {
_db(Xapian::Database(path)) {} return _qparser.parse_query
(searchexpr,
const Xapian::Database& _db; Xapian::QueryParser::FLAG_BOOLEAN |
Xapian::QueryParser _qparser; Xapian::QueryParser::FLAG_PHRASE |
Xapian::Sorter* _sorters[MU_MSG_FIELD_TYPE_NUM]; Xapian::QueryParser::FLAG_LOVEHATE |
Xapian::QueryParser::FLAG_BOOLEAN_ANY_CASE |
Xapian::QueryParser::FLAG_WILDCARD |
Xapian::QueryParser::FLAG_PURE_NOT |
Xapian::QueryParser::FLAG_PARTIAL);
}
const Xapian::Database& db() { return _db; }
private:
const Xapian::Database _db;
Xapian::QueryParser _qparser;
Xapian::Sorter* _sorters[MU_MSG_FIELD_TYPE_NUM];
}; };
static void static void
@ -56,69 +91,39 @@ add_prefix (const MuMsgField* field, Xapian::QueryParser* qparser)
} }
MuQueryXapian* MuQueryXapian*
mu_query_xapian_new (const char* path, GError **err) mu_query_xapian_new (const char* path)
{ {
MuQueryXapian *self; char *xpath;
g_return_val_if_fail (path, NULL); g_return_val_if_fail (path, NULL);
if (!g_file_test (path, G_FILE_TEST_IS_DIR) || /* move this to common place? */
g_access(path, R_OK) != 0) { xpath = g_strdup_printf ("%s%c%s", path, G_DIR_SEPARATOR, "xapian");
g_set_error (err, 0, 0,"'%s' is not a readable xapian dir", path); if (!g_file_test (xpath, G_FILE_TEST_IS_DIR) ||
g_access(xpath, R_OK) != 0) {
g_warning ("'%s' is not a readable xapian dir", xpath);
g_free (xpath);
return NULL; return NULL;
} }
self = 0;
try { try {
self = new MuQueryXapian (path); MuQueryXapian *qx (new MuQueryXapian (xpath));
g_free (xpath);
self->_qparser.set_database(self->_db); return qx;
self->_qparser.set_default_op(Xapian::Query::OP_OR);
self->_qparser.set_stemming_strategy
(Xapian::QueryParser::STEM_SOME);
} catch (const Xapian::Error &ex) {
g_set_error (err, 0, 0,"%s: caught xapian exception '%s' (%s)",
__FUNCTION__, ex.get_msg().c_str(),
ex.get_error_string());
delete self;
self = 0;
} catch (...) {
g_set_error (err, 0, 0,"%s: caught exception", __FUNCTION__);
delete self;
self = 0;
}
if (self) {
memset (self->_sorters, 0, sizeof(self->_sorters));
mu_msg_field_foreach ((MuMsgFieldForEachFunc)add_prefix, } catch (...) {
(gconstpointer)&self->_qparser); g_free (xpath);
g_warning ("%s: caught exception", __FUNCTION__);
return NULL;
} }
return self ? self: NULL;
} }
void void
mu_query_xapian_destroy (MuQueryXapian *self) mu_query_xapian_destroy (MuQueryXapian *self)
{ {
if (!self)
return;
try { try {
for (int i = 0; i != MU_MSG_FIELD_TYPE_NUM; ++i) {
delete self->_sorters[i];
self->_sorters[i] = 0;
}
delete self; delete self;
} catch (const Xapian::Error &err) {
g_warning ("%s: caught xapian exception '%s' (%s)",
__FUNCTION__, err.get_msg().c_str(),
err.get_error_string());
} catch (...) { } catch (...) {
g_warning ("%s: caught exception", __FUNCTION__); g_warning ("%s: caught exception", __FUNCTION__);
} }
@ -126,19 +131,6 @@ mu_query_xapian_destroy (MuQueryXapian *self)
static Xapian::Query
get_query (MuQueryXapian *self, const char* searchexpr)
{
return (self->_qparser.parse_query
(searchexpr,
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 |
Xapian::QueryParser::FLAG_PARTIAL));
}
MuMsgXapian* MuMsgXapian*
mu_query_xapian_run (MuQueryXapian *self, const char* searchexpr, mu_query_xapian_run (MuQueryXapian *self, const char* searchexpr,
@ -148,8 +140,8 @@ mu_query_xapian_run (MuQueryXapian *self, const char* searchexpr,
g_return_val_if_fail (searchexpr, NULL); g_return_val_if_fail (searchexpr, NULL);
try { try {
Xapian::Query q(get_query(self, searchexpr)); Xapian::Query q(self->get_query(searchexpr));
Xapian::Enquire enq (self->_db); Xapian::Enquire enq (self->db());
if (sortfield) if (sortfield)
enq.set_sort_by_value ( enq.set_sort_by_value (
@ -180,7 +172,7 @@ mu_query_xapian_as_string (MuQueryXapian *self, const char* searchexpr)
g_return_val_if_fail (searchexpr, NULL); g_return_val_if_fail (searchexpr, NULL);
try { try {
Xapian::Query q(get_query(self, searchexpr)); Xapian::Query q(self->get_query(searchexpr));
return g_strdup(q.get_description().c_str()); return g_strdup(q.get_description().c_str());
} catch (const Xapian::Error &err) { } catch (const Xapian::Error &err) {
@ -212,8 +204,4 @@ mu_query_xapian_combine (GSList *lst, gboolean connect_or)
} }
return g_string_free (str, FALSE); return g_string_free (str, FALSE);
} }

View File

@ -41,7 +41,7 @@ typedef struct _MuQueryXapian MuQueryXapian;
* when the instance is no longer needed, use mu_query_xapian_destroy * when the instance is no longer needed, use mu_query_xapian_destroy
* to free it * to free it
*/ */
MuQueryXapian *mu_query_xapian_new (const char* path, GError **err); MuQueryXapian *mu_query_xapian_new (const char* path);
/** /**
* destroy the MuQueryXapian instance * destroy the MuQueryXapian instance

View File

@ -180,22 +180,16 @@ do_output (MuQueryXapian *xapian, GSList *args, MuConfigOptions* opts)
MuResult MuResult
mu_query_run (MuConfigOptions *opts, GSList *args) mu_query_run (MuConfigOptions *opts, GSList *args)
{ {
GError *err = 0;
MuQueryXapian *xapian; MuQueryXapian *xapian;
MuResult rv; MuResult rv;
rv = MU_OK; rv = MU_OK;
handle_options (opts); handle_options (opts);
xapian = mu_query_xapian_new ("/home/djcb/.mu", &err); xapian = mu_query_xapian_new (opts->muhome);
if (!xapian) { if (!xapian)
if (err) {
g_printerr ("error: %s\n", err->message);
g_error_free (err);
}
return MU_ERROR; return MU_ERROR;
}
rv = do_output (xapian, args, opts) ? 0 : 1; rv = do_output (xapian, args, opts) ? 0 : 1;
mu_query_xapian_destroy (xapian); mu_query_xapian_destroy (xapian);