* mu-cmd-find, mu-cmd-cfind, mu-cmd-index: use GError for error reporting
This commit is contained in:
@ -167,7 +167,7 @@ each_contact (const char *email, const char *name, time_t tstamp,
|
|||||||
|
|
||||||
static MuError
|
static MuError
|
||||||
run_cmd_cfind (const char* pattern, MuConfigFormat format,
|
run_cmd_cfind (const char* pattern, MuConfigFormat format,
|
||||||
gboolean color)
|
gboolean color, GError **err)
|
||||||
{
|
{
|
||||||
gboolean rv;
|
gboolean rv;
|
||||||
MuContacts *contacts;
|
MuContacts *contacts;
|
||||||
@ -176,7 +176,8 @@ run_cmd_cfind (const char* pattern, MuConfigFormat format,
|
|||||||
|
|
||||||
contacts = mu_contacts_new (mu_runtime_path(MU_RUNTIME_PATH_CONTACTS));
|
contacts = mu_contacts_new (mu_runtime_path(MU_RUNTIME_PATH_CONTACTS));
|
||||||
if (!contacts) {
|
if (!contacts) {
|
||||||
g_warning ("could not retrieve contacts");
|
g_set_error (err, 0, MU_ERROR_CONTACTS_CANNOT_RETRIEVE,
|
||||||
|
"could not retrieve contacts");
|
||||||
return MU_ERROR_CONTACTS_CANNOT_RETRIEVE;
|
return MU_ERROR_CONTACTS_CANNOT_RETRIEVE;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -223,15 +224,19 @@ cfind_params_valid (MuConfig *opts)
|
|||||||
|
|
||||||
|
|
||||||
MuError
|
MuError
|
||||||
mu_cmd_cfind (MuConfig *opts)
|
mu_cmd_cfind (MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
g_return_val_if_fail (opts, MU_ERROR_INTERNAL);
|
g_return_val_if_fail (opts, MU_ERROR_INTERNAL);
|
||||||
g_return_val_if_fail (opts->cmd == MU_CONFIG_CMD_CFIND,
|
g_return_val_if_fail (opts->cmd == MU_CONFIG_CMD_CFIND,
|
||||||
MU_ERROR_INTERNAL);
|
MU_ERROR_INTERNAL);
|
||||||
|
|
||||||
if (!cfind_params_valid (opts))
|
if (!cfind_params_valid (opts)) {
|
||||||
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
|
"invalid parameters");
|
||||||
return MU_ERROR_IN_PARAMETERS;
|
return MU_ERROR_IN_PARAMETERS;
|
||||||
|
}
|
||||||
|
|
||||||
return run_cmd_cfind (opts->params[1], opts->format, opts->color);
|
return run_cmd_cfind (opts->params[1], opts->format, opts->color,
|
||||||
|
err);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@ -47,50 +47,34 @@
|
|||||||
|
|
||||||
|
|
||||||
static gboolean output_links (MuMsgIter *iter, const char* linksdir,
|
static gboolean output_links (MuMsgIter *iter, const char* linksdir,
|
||||||
gboolean clearlinks,size_t *count);
|
gboolean clearlinks, GError **err);
|
||||||
static gboolean output_sexp (MuMsgIter *iter, gboolean threads,
|
static gboolean output_sexp (MuMsgIter *iter, gboolean threads,
|
||||||
gboolean include_unreadable, size_t *count);
|
gboolean include_unreadable, GError **err);
|
||||||
static gboolean output_xml (MuMsgIter *iter,gboolean include_unreadable,
|
static gboolean output_xml (MuMsgIter *iter,gboolean include_unreadable,
|
||||||
size_t *count);
|
GError **err);
|
||||||
static gboolean output_plain (MuMsgIter *iter, const char *fields,
|
static gboolean output_plain (MuMsgIter *iter, const char *fields,
|
||||||
gboolean summary,gboolean threads,
|
gboolean summary,gboolean threads,
|
||||||
gboolean color, gboolean include_unreadable,
|
gboolean color, gboolean include_unreadable,
|
||||||
size_t *count);
|
GError **err);
|
||||||
|
|
||||||
static void
|
|
||||||
upgrade_warning (void)
|
|
||||||
{
|
|
||||||
g_warning ("the database needs to be updated to version %s\n",
|
|
||||||
MU_STORE_SCHEMA_VERSION);
|
|
||||||
g_message ("please run 'mu index --rebuild' (see the man page)");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
print_xapian_query (MuQuery *xapian, const gchar *query, size_t *count)
|
print_xapian_query (MuQuery *xapian, const gchar *query, GError **err)
|
||||||
{
|
{
|
||||||
char *querystr;
|
char *querystr;
|
||||||
GError *err;
|
|
||||||
|
|
||||||
err = NULL;
|
querystr = mu_query_as_string (xapian, query, err);
|
||||||
|
if (!querystr)
|
||||||
querystr = mu_query_as_string (xapian, query, &err);
|
|
||||||
if (!querystr) {
|
|
||||||
g_warning ("error: %s", err->message);
|
|
||||||
g_error_free (err);
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
|
||||||
|
|
||||||
g_print ("%s\n", querystr);
|
g_print ("%s\n", querystr);
|
||||||
g_free (querystr);
|
g_free (querystr);
|
||||||
|
|
||||||
*count = 1;
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* returns NULL if there is an error */
|
/* returns MU_MSG_FIELD_ID_NONE if there is an error */
|
||||||
static MuMsgFieldId
|
static MuMsgFieldId
|
||||||
sort_field_from_string (const char* fieldstr)
|
sort_field_from_string (const char* fieldstr, GError **err)
|
||||||
{
|
{
|
||||||
MuMsgFieldId mfid;
|
MuMsgFieldId mfid;
|
||||||
|
|
||||||
@ -102,29 +86,27 @@ sort_field_from_string (const char* fieldstr)
|
|||||||
mfid = mu_msg_field_id_from_shortcut(fieldstr[0],
|
mfid = mu_msg_field_id_from_shortcut(fieldstr[0],
|
||||||
FALSE);
|
FALSE);
|
||||||
if (mfid == MU_MSG_FIELD_ID_NONE)
|
if (mfid == MU_MSG_FIELD_ID_NONE)
|
||||||
g_warning ("not a valid sort field: '%s'\n",
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
fieldstr);
|
"not a valid sort field: '%s'\n", fieldstr);
|
||||||
return mfid;
|
return mfid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
output_query_results (MuMsgIter *iter, MuConfig *opts, size_t *count)
|
output_query_results (MuMsgIter *iter, MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
switch (opts->format) {
|
switch (opts->format) {
|
||||||
case MU_CONFIG_FORMAT_LINKS:
|
case MU_CONFIG_FORMAT_LINKS:
|
||||||
return output_links (iter, opts->linksdir, opts->clearlinks,
|
return output_links (iter, opts->linksdir, opts->clearlinks, err);
|
||||||
count);
|
|
||||||
case MU_CONFIG_FORMAT_PLAIN:
|
case MU_CONFIG_FORMAT_PLAIN:
|
||||||
return output_plain (iter, opts->fields, opts->summary,
|
return output_plain (iter, opts->fields, opts->summary,
|
||||||
opts->threads, opts->color,
|
opts->threads, opts->color,
|
||||||
opts->include_unreadable,
|
opts->include_unreadable, err);
|
||||||
count);
|
|
||||||
case MU_CONFIG_FORMAT_XML:
|
case MU_CONFIG_FORMAT_XML:
|
||||||
return output_xml (iter, opts->include_unreadable, count);
|
return output_xml (iter, opts->include_unreadable, err);
|
||||||
case MU_CONFIG_FORMAT_SEXP:
|
case MU_CONFIG_FORMAT_SEXP:
|
||||||
return output_sexp (iter, opts->threads,
|
return output_sexp (iter, opts->threads,
|
||||||
opts->include_unreadable, count);
|
opts->include_unreadable, err);
|
||||||
default:
|
default:
|
||||||
g_assert_not_reached ();
|
g_assert_not_reached ();
|
||||||
return FALSE;
|
return FALSE;
|
||||||
@ -133,49 +115,36 @@ output_query_results (MuMsgIter *iter, MuConfig *opts, size_t *count)
|
|||||||
|
|
||||||
|
|
||||||
static MuMsgIter*
|
static MuMsgIter*
|
||||||
run_query (MuQuery *xapian, const gchar *query, MuConfig *opts, size_t *count)
|
run_query (MuQuery *xapian, const gchar *query, MuConfig *opts,
|
||||||
|
GError **err)
|
||||||
{
|
{
|
||||||
GError *err;
|
|
||||||
MuMsgIter *iter;
|
MuMsgIter *iter;
|
||||||
MuMsgFieldId sortid;
|
MuMsgFieldId sortid;
|
||||||
|
|
||||||
sortid = MU_MSG_FIELD_ID_NONE;
|
sortid = MU_MSG_FIELD_ID_NONE;
|
||||||
if (opts->sortfield) {
|
if (opts->sortfield) {
|
||||||
sortid = sort_field_from_string (opts->sortfield);
|
sortid = sort_field_from_string (opts->sortfield, err);
|
||||||
if (sortid == MU_MSG_FIELD_ID_NONE) /* error occured? */
|
if (sortid == MU_MSG_FIELD_ID_NONE) /* error occured? */
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = NULL;
|
|
||||||
iter = mu_query_run (xapian, query, opts->threads, sortid,
|
iter = mu_query_run (xapian, query, opts->threads, sortid,
|
||||||
opts->descending ? FALSE : TRUE,
|
opts->descending ? FALSE : TRUE, err);
|
||||||
&err);
|
|
||||||
if (!iter) {
|
|
||||||
g_warning ("error: %s", err->message);
|
|
||||||
g_error_free (err);
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
return iter;
|
return iter;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
process_query (MuQuery *xapian, const gchar *query, MuConfig *opts,
|
process_query (MuQuery *xapian, const gchar *query, MuConfig *opts, GError **err)
|
||||||
size_t *count)
|
|
||||||
{
|
{
|
||||||
MuMsgIter *iter;
|
MuMsgIter *iter;
|
||||||
gboolean rv;
|
gboolean rv;
|
||||||
|
|
||||||
iter = run_query (xapian, query, opts, count);
|
iter = run_query (xapian, query, opts, err);
|
||||||
if (!iter)
|
if (!iter)
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
rv = output_query_results (iter, opts, count);
|
rv = output_query_results (iter, opts, err);
|
||||||
|
|
||||||
if (rv && count && *count == 0)
|
|
||||||
g_warning ("no matching messages found");
|
|
||||||
|
|
||||||
mu_msg_iter_destroy (iter);
|
mu_msg_iter_destroy (iter);
|
||||||
|
|
||||||
return rv;
|
return rv;
|
||||||
@ -183,15 +152,15 @@ process_query (MuQuery *xapian, const gchar *query, MuConfig *opts,
|
|||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
exec_cmd (const char *path, const char *cmd)
|
exec_cmd (const char *path, const char *cmd, GError **err)
|
||||||
{
|
{
|
||||||
gint status;
|
gint status;
|
||||||
GError *err;
|
|
||||||
char *cmdline, *escpath;
|
char *cmdline, *escpath;
|
||||||
gboolean rv;
|
gboolean rv;
|
||||||
|
|
||||||
if (access (path, R_OK) != 0) {
|
if (access (path, R_OK) != 0) {
|
||||||
g_warning ("cannot read %s: %s", path, strerror(errno));
|
g_set_error (err, 0, MU_ERROR_FILE_CANNOT_READ,
|
||||||
|
"cannot read %s: %s", path, strerror(errno));
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -200,32 +169,26 @@ exec_cmd (const char *path, const char *cmd)
|
|||||||
cmdline = g_strdup_printf ("%s %s", cmd, escpath);
|
cmdline = g_strdup_printf ("%s %s", cmd, escpath);
|
||||||
err = NULL;
|
err = NULL;
|
||||||
rv = g_spawn_command_line_sync (cmdline, NULL, NULL,
|
rv = g_spawn_command_line_sync (cmdline, NULL, NULL,
|
||||||
&status, &err);
|
&status, err);
|
||||||
g_free (cmdline);
|
g_free (cmdline);
|
||||||
g_free (escpath);
|
g_free (escpath);
|
||||||
|
|
||||||
if (!rv) {
|
return rv;
|
||||||
g_warning ("command returned %d on %s: %s\n",
|
|
||||||
status, path, err->message);
|
|
||||||
g_error_free (err);
|
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
return TRUE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
exec_cmd_on_query (MuQuery *xapian, const gchar *query, MuConfig *opts,
|
exec_cmd_on_query (MuQuery *xapian, const gchar *query, MuConfig *opts,
|
||||||
size_t *count)
|
GError **err)
|
||||||
{
|
{
|
||||||
MuMsgIter *iter;
|
MuMsgIter *iter;
|
||||||
gboolean rv;
|
gboolean rv;
|
||||||
|
size_t count;
|
||||||
|
|
||||||
if (!(iter = run_query (xapian, query, opts, count)))
|
if (!(iter = run_query (xapian, query, opts, err)))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
for (rv = TRUE, *count = 0; !mu_msg_iter_is_done (iter);
|
for (rv = TRUE, count = 0; !mu_msg_iter_is_done (iter);
|
||||||
mu_msg_iter_next(iter)) {
|
mu_msg_iter_next(iter)) {
|
||||||
const char *path;
|
const char *path;
|
||||||
MuMsg *msg;
|
MuMsg *msg;
|
||||||
@ -240,23 +203,25 @@ exec_cmd_on_query (MuQuery *xapian, const gchar *query, MuConfig *opts,
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
rv = exec_cmd (path, opts->exec);
|
rv = exec_cmd (path, opts->exec, err);
|
||||||
if (rv)
|
if (rv)
|
||||||
++*count;
|
++count;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (rv && count && *count == 0)
|
if (count == 0) {
|
||||||
g_warning ("no matching messages found");
|
g_set_error (err, 0, MU_ERROR_NO_MATCHES,
|
||||||
|
"no matches for search expression");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
mu_msg_iter_destroy (iter);
|
mu_msg_iter_destroy (iter);
|
||||||
|
|
||||||
return rv;
|
return rv;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
format_params_valid (MuConfig *opts)
|
format_params_valid (MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
switch (opts->format) {
|
switch (opts->format) {
|
||||||
case MU_CONFIG_FORMAT_PLAIN:
|
case MU_CONFIG_FORMAT_PLAIN:
|
||||||
@ -266,18 +231,21 @@ format_params_valid (MuConfig *opts)
|
|||||||
case MU_CONFIG_FORMAT_XQUERY:
|
case MU_CONFIG_FORMAT_XQUERY:
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
g_warning ("invalid output format %s",
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
|
"invalid output format %s",
|
||||||
opts->formatstr ? opts->formatstr : "<none>");
|
opts->formatstr ? opts->formatstr : "<none>");
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (opts->format == MU_CONFIG_FORMAT_LINKS && !opts->linksdir) {
|
if (opts->format == MU_CONFIG_FORMAT_LINKS && !opts->linksdir) {
|
||||||
g_warning ("missing --linksdir argument");
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
|
"missing --linksdir argument");
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (opts->linksdir && opts->format != MU_CONFIG_FORMAT_LINKS) {
|
if (opts->linksdir && opts->format != MU_CONFIG_FORMAT_LINKS) {
|
||||||
g_warning ("--linksdir is only valid with --format=links");
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
|
"--linksdir is only valid with --format=links");
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -285,7 +253,7 @@ format_params_valid (MuConfig *opts)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
query_params_valid (MuConfig *opts)
|
query_params_valid (MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
const gchar *xpath;
|
const gchar *xpath;
|
||||||
|
|
||||||
@ -294,14 +262,13 @@ query_params_valid (MuConfig *opts)
|
|||||||
if (mu_util_check_dir (xpath, TRUE, FALSE))
|
if (mu_util_check_dir (xpath, TRUE, FALSE))
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
||||||
g_warning ("'%s' is not a readable Xapian directory\n", xpath);
|
g_set_error (err, 0, MU_ERROR_FILE_CANNOT_READ,
|
||||||
g_message ("did you run 'mu index'?");
|
"'%s' is not a readable Xapian directory", xpath);
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
static gchar*
|
static gchar*
|
||||||
resolve_bookmark (MuConfig *opts)
|
resolve_bookmark (MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
MuBookmarks *bm;
|
MuBookmarks *bm;
|
||||||
char* val;
|
char* val;
|
||||||
@ -310,36 +277,38 @@ resolve_bookmark (MuConfig *opts)
|
|||||||
bmfile = mu_runtime_path (MU_RUNTIME_PATH_BOOKMARKS);
|
bmfile = mu_runtime_path (MU_RUNTIME_PATH_BOOKMARKS);
|
||||||
bm = mu_bookmarks_new (bmfile);
|
bm = mu_bookmarks_new (bmfile);
|
||||||
if (!bm) {
|
if (!bm) {
|
||||||
g_warning ("failed to open bookmarks file '%s'", bmfile);
|
g_set_error (err, 0, MU_ERROR_FILE_CANNOT_OPEN,
|
||||||
|
"failed to open bookmarks file '%s'", bmfile);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
val = (gchar*)mu_bookmarks_lookup (bm, opts->bookmark);
|
val = (gchar*)mu_bookmarks_lookup (bm, opts->bookmark);
|
||||||
if (!val)
|
if (!val)
|
||||||
g_warning ("bookmark '%s' not found", opts->bookmark);
|
g_set_error (err, 0, MU_ERROR_NO_MATCHES,
|
||||||
|
"bookmark '%s' not found", opts->bookmark);
|
||||||
else
|
else
|
||||||
val = g_strdup (val);
|
val = g_strdup (val);
|
||||||
|
|
||||||
mu_bookmarks_destroy (bm);
|
mu_bookmarks_destroy (bm);
|
||||||
|
|
||||||
return val;
|
return val;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static gchar*
|
static gchar*
|
||||||
get_query (MuConfig *opts)
|
get_query (MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
gchar *query, *bookmarkval;
|
gchar *query, *bookmarkval;
|
||||||
|
|
||||||
/* params[0] is 'find', actual search params start with [1] */
|
/* params[0] is 'find', actual search params start with [1] */
|
||||||
if (!opts->bookmark && !opts->params[1]) {
|
if (!opts->bookmark && !opts->params[1]) {
|
||||||
g_warning ("usage: mu find [options] search-expression");
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
return FALSE;
|
"error in parameters");
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
bookmarkval = NULL;
|
bookmarkval = NULL;
|
||||||
if (opts->bookmark) {
|
if (opts->bookmark) {
|
||||||
bookmarkval = resolve_bookmark (opts);
|
bookmarkval = resolve_bookmark (opts, err);
|
||||||
if (!bookmarkval)
|
if (!bookmarkval)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
@ -357,41 +326,33 @@ get_query (MuConfig *opts)
|
|||||||
return query;
|
return query;
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
|
||||||
db_is_ready (MuStore *store)
|
static MuQuery*
|
||||||
|
get_query_obj (MuStore *store, GError **err)
|
||||||
{
|
{
|
||||||
if (mu_store_count (store) == 0) {
|
MuQuery *mquery;
|
||||||
g_warning ("database is empty; use 'mu index' to "
|
unsigned count;
|
||||||
"add messages");
|
|
||||||
return FALSE;
|
count = mu_store_count (store, err);
|
||||||
|
|
||||||
|
if (count == (unsigned)-1)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
if (count == 0) {
|
||||||
|
g_set_error (err, 0, MU_ERROR_XAPIAN_IS_EMPTY,
|
||||||
|
"the database is empty");
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mu_store_needs_upgrade (store)) {
|
if (mu_store_needs_upgrade (store)) {
|
||||||
upgrade_warning ();
|
g_set_error (err, 0, MU_ERROR_XAPIAN_NOT_UP_TO_DATE,
|
||||||
return FALSE;
|
"the database is not up-to-date");
|
||||||
}
|
|
||||||
|
|
||||||
return TRUE;
|
|
||||||
}
|
|
||||||
|
|
||||||
static MuQuery*
|
|
||||||
get_query_obj (MuStore *store)
|
|
||||||
{
|
|
||||||
GError *err;
|
|
||||||
MuQuery *mquery;
|
|
||||||
|
|
||||||
if (!db_is_ready(store)) {
|
|
||||||
g_warning ("database is not ready");
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = NULL;
|
mquery = mu_query_new (store, err);
|
||||||
mquery = mu_query_new (store, &err);
|
if (!mquery)
|
||||||
if (!mquery) {
|
|
||||||
g_warning ("error: %s", err->message);
|
|
||||||
g_error_free (err);
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
|
||||||
|
|
||||||
return mquery;
|
return mquery;
|
||||||
}
|
}
|
||||||
@ -452,10 +413,10 @@ link_message (const char *src, const char *destdir)
|
|||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
output_links (MuMsgIter *iter, const char* linksdir,
|
output_links (MuMsgIter *iter, const char* linksdir, gboolean clearlinks,
|
||||||
gboolean clearlinks, size_t *count)
|
GError **err)
|
||||||
{
|
{
|
||||||
size_t okcount, errcount;
|
size_t count, errcount;
|
||||||
MuMsgIter *myiter;
|
MuMsgIter *myiter;
|
||||||
|
|
||||||
g_return_val_if_fail (iter, FALSE);
|
g_return_val_if_fail (iter, FALSE);
|
||||||
@ -465,7 +426,7 @@ output_links (MuMsgIter *iter, const char* linksdir,
|
|||||||
if (!create_linksdir_maybe (linksdir, clearlinks))
|
if (!create_linksdir_maybe (linksdir, clearlinks))
|
||||||
return FALSE;
|
return FALSE;
|
||||||
|
|
||||||
for (myiter = iter, errcount = okcount = 0; !mu_msg_iter_is_done (myiter);
|
for (myiter = iter, count = errcount = 0; !mu_msg_iter_is_done (myiter);
|
||||||
mu_msg_iter_next (myiter)) {
|
mu_msg_iter_next (myiter)) {
|
||||||
|
|
||||||
const char* path;
|
const char* path;
|
||||||
@ -477,14 +438,22 @@ output_links (MuMsgIter *iter, const char* linksdir,
|
|||||||
|
|
||||||
path = mu_msg_get_path (msg);
|
path = mu_msg_get_path (msg);
|
||||||
if (access (path, R_OK) == 0) /* only link to readable */
|
if (access (path, R_OK) == 0) /* only link to readable */
|
||||||
link_message (path, linksdir) ? ++okcount : ++errcount;
|
link_message (path, linksdir) ? ++count : ++errcount;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errcount > 0) {
|
||||||
|
g_set_error (err, 0, MU_ERROR_FILE_CANNOT_LINK,
|
||||||
|
"error linking %u message(s)", errcount);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count) {
|
||||||
|
g_set_error (err, 0, MU_ERROR_NO_MATCHES,
|
||||||
|
"no existing matches for search expression");
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (errcount > 0)
|
|
||||||
g_warning ("error linking some of the messages");
|
|
||||||
|
|
||||||
if (count)
|
|
||||||
*count = okcount;
|
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
@ -628,14 +597,14 @@ thread_indent (MuMsgIter *iter)
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
static size_t
|
static void
|
||||||
output_plain_fields (MuMsg *msg, const char *fields,
|
output_plain_fields (MuMsg *msg, const char *fields,
|
||||||
gboolean color, gboolean threads)
|
gboolean color, gboolean threads)
|
||||||
{
|
{
|
||||||
const char* myfields;
|
const char* myfields;
|
||||||
size_t len;
|
int nonempty;
|
||||||
|
|
||||||
for (myfields = fields, len = 0; *myfields; ++myfields) {
|
for (myfields = fields, nonempty = 0; *myfields; ++myfields) {
|
||||||
|
|
||||||
MuMsgFieldId mfid;
|
MuMsgFieldId mfid;
|
||||||
mfid = mu_msg_field_id_from_shortcut (*myfields, FALSE);
|
mfid = mu_msg_field_id_from_shortcut (*myfields, FALSE);
|
||||||
@ -643,35 +612,35 @@ output_plain_fields (MuMsg *msg, const char *fields,
|
|||||||
if (mfid == MU_MSG_FIELD_ID_NONE ||
|
if (mfid == MU_MSG_FIELD_ID_NONE ||
|
||||||
(!mu_msg_field_xapian_value (mfid) &&
|
(!mu_msg_field_xapian_value (mfid) &&
|
||||||
!mu_msg_field_xapian_contact (mfid)))
|
!mu_msg_field_xapian_contact (mfid)))
|
||||||
len += printf ("%c", *myfields);
|
nonempty += printf ("%c", *myfields);
|
||||||
|
|
||||||
else {
|
else {
|
||||||
ansi_color_maybe (mfid, color);
|
ansi_color_maybe (mfid, color);
|
||||||
len += mu_util_fputs_encoded
|
nonempty += mu_util_fputs_encoded
|
||||||
(display_field (msg, mfid), stdout);
|
(display_field (msg, mfid), stdout);
|
||||||
ansi_reset_maybe (mfid, color);
|
ansi_reset_maybe (mfid, color);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return len;
|
if (nonempty)
|
||||||
|
fputs ("\n", stdout);
|
||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
output_plain (MuMsgIter *iter, const char *fields, gboolean summary,
|
output_plain (MuMsgIter *iter, const char *fields, gboolean summary,
|
||||||
gboolean threads, gboolean color, gboolean include_unreadable,
|
gboolean threads, gboolean color, gboolean include_unreadable,
|
||||||
size_t *count)
|
GError **err)
|
||||||
{
|
{
|
||||||
MuMsgIter *myiter;
|
MuMsgIter *myiter;
|
||||||
size_t mycount;
|
size_t count;
|
||||||
|
|
||||||
g_return_val_if_fail (iter, FALSE);
|
g_return_val_if_fail (iter, FALSE);
|
||||||
g_return_val_if_fail (fields, FALSE);
|
g_return_val_if_fail (fields, FALSE);
|
||||||
|
|
||||||
for (myiter = iter, mycount = 0; !mu_msg_iter_is_done (myiter);
|
for (myiter = iter, count = 0; !mu_msg_iter_is_done (myiter);
|
||||||
mu_msg_iter_next (myiter)) {
|
mu_msg_iter_next (myiter)) {
|
||||||
size_t len;
|
|
||||||
MuMsg *msg;
|
|
||||||
|
|
||||||
|
MuMsg *msg;
|
||||||
msg = mu_msg_iter_get_msg_floating (iter); /* don't unref */
|
msg = mu_msg_iter_get_msg_floating (iter); /* don't unref */
|
||||||
if (!msg)
|
if (!msg)
|
||||||
continue;
|
continue;
|
||||||
@ -687,17 +656,19 @@ output_plain (MuMsgIter *iter, const char *fields, gboolean summary,
|
|||||||
if (threads)
|
if (threads)
|
||||||
thread_indent (iter);
|
thread_indent (iter);
|
||||||
|
|
||||||
len = output_plain_fields (msg, fields, color, threads);
|
output_plain_fields (msg, fields, color, threads);
|
||||||
|
|
||||||
g_print (len > 0 ? "\n" : "");
|
|
||||||
if (summary)
|
if (summary)
|
||||||
print_summary (msg);
|
print_summary (msg);
|
||||||
|
|
||||||
++mycount;
|
++count;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (count)
|
if (count == 0) {
|
||||||
*count = mycount;
|
g_set_error (err, 0, MU_ERROR_NO_MATCHES,
|
||||||
|
"no existing matches for search expression");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
@ -719,14 +690,14 @@ print_attr_xml (const char* elm, const char *str)
|
|||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
output_sexp (MuMsgIter *iter, gboolean threads,
|
output_sexp (MuMsgIter *iter, gboolean threads,
|
||||||
gboolean include_unreadable, size_t *count)
|
gboolean include_unreadable, GError **err)
|
||||||
{
|
{
|
||||||
MuMsgIter *myiter;
|
MuMsgIter *myiter;
|
||||||
size_t mycount;
|
size_t count;
|
||||||
|
|
||||||
g_return_val_if_fail (iter, FALSE);
|
g_return_val_if_fail (iter, FALSE);
|
||||||
|
|
||||||
for (myiter = iter, mycount = 0; !mu_msg_iter_is_done (myiter);
|
for (myiter = iter, count = 0; !mu_msg_iter_is_done (myiter);
|
||||||
mu_msg_iter_next (myiter)) {
|
mu_msg_iter_next (myiter)) {
|
||||||
|
|
||||||
MuMsg *msg;
|
MuMsg *msg;
|
||||||
@ -748,12 +719,14 @@ output_sexp (MuMsgIter *iter, gboolean threads,
|
|||||||
fputs (sexp, stdout);
|
fputs (sexp, stdout);
|
||||||
g_free (sexp);
|
g_free (sexp);
|
||||||
|
|
||||||
++mycount;
|
++count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (count == 0) {
|
||||||
if (count)
|
g_set_error (err, 0, MU_ERROR_NO_MATCHES,
|
||||||
*count = mycount;
|
"no existing matches for search expression");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
@ -778,17 +751,17 @@ output_xml_msg (MuMsg *msg)
|
|||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
output_xml (MuMsgIter *iter, gboolean include_unreadable, size_t *count)
|
output_xml (MuMsgIter *iter, gboolean include_unreadable, GError **err)
|
||||||
{
|
{
|
||||||
MuMsgIter *myiter;
|
MuMsgIter *myiter;
|
||||||
size_t mycount;
|
size_t count;
|
||||||
|
|
||||||
g_return_val_if_fail (iter, FALSE);
|
g_return_val_if_fail (iter, FALSE);
|
||||||
|
|
||||||
g_print ("<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n");
|
g_print ("<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n");
|
||||||
g_print ("<messages>\n");
|
g_print ("<messages>\n");
|
||||||
|
|
||||||
for (myiter = iter, mycount = 0; !mu_msg_iter_is_done (myiter);
|
for (myiter = iter, count = 0; !mu_msg_iter_is_done (myiter);
|
||||||
mu_msg_iter_next (myiter)) {
|
mu_msg_iter_next (myiter)) {
|
||||||
|
|
||||||
GError *err;
|
GError *err;
|
||||||
@ -805,50 +778,78 @@ output_xml (MuMsgIter *iter, gboolean include_unreadable, size_t *count)
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
output_xml_msg (msg);
|
output_xml_msg (msg);
|
||||||
++mycount;
|
++count;
|
||||||
}
|
}
|
||||||
g_print ("</messages>\n");
|
g_print ("</messages>\n");
|
||||||
|
|
||||||
if (count)
|
if (count == 0) {
|
||||||
*count = mycount;
|
g_set_error (err, 0, MU_ERROR_NO_MATCHES,
|
||||||
|
"no existing matches for search expression");
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
return TRUE;
|
return TRUE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gboolean
|
||||||
|
execute_find (MuStore *store, MuConfig *opts, GError **err)
|
||||||
|
{
|
||||||
|
char *query_str;
|
||||||
|
MuQuery *oracle;
|
||||||
|
gboolean rv;
|
||||||
|
|
||||||
|
oracle = get_query_obj(store, err);
|
||||||
|
if (!oracle)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
query_str = get_query (opts, err);
|
||||||
|
if (!query_str) {
|
||||||
|
mu_query_destroy (oracle);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (opts->format == MU_CONFIG_FORMAT_XQUERY)
|
||||||
|
rv = print_xapian_query (oracle, query_str, err);
|
||||||
|
else if (opts->exec)
|
||||||
|
rv = exec_cmd_on_query (oracle, query_str, opts, err);
|
||||||
|
else
|
||||||
|
rv = process_query (oracle, query_str, opts, err);
|
||||||
|
|
||||||
|
mu_query_destroy (oracle);
|
||||||
|
g_free (query_str);
|
||||||
|
|
||||||
|
return rv;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
show_usage (void)
|
||||||
|
{
|
||||||
|
const char *usage_str =
|
||||||
|
"usage: mu find [options] <search expression>\n";
|
||||||
|
g_message ("%s", usage_str);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
MuError
|
MuError
|
||||||
mu_cmd_find (MuStore *store, MuConfig *opts)
|
mu_cmd_find (MuStore *store, MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
MuQuery *xapian;
|
|
||||||
gboolean rv;
|
|
||||||
gchar *query;
|
|
||||||
size_t count = 0;
|
|
||||||
|
|
||||||
g_return_val_if_fail (opts, MU_ERROR_INTERNAL);
|
g_return_val_if_fail (opts, MU_ERROR_INTERNAL);
|
||||||
g_return_val_if_fail (opts->cmd == MU_CONFIG_CMD_FIND,
|
g_return_val_if_fail (opts->cmd == MU_CONFIG_CMD_FIND,
|
||||||
MU_ERROR_INTERNAL);
|
MU_ERROR_INTERNAL);
|
||||||
|
|
||||||
if (!query_params_valid (opts) || !format_params_valid(opts))
|
if (!query_params_valid (opts, err) || !format_params_valid(opts, err)) {
|
||||||
return MU_ERROR_IN_PARAMETERS;
|
|
||||||
|
|
||||||
xapian = get_query_obj(store);
|
if (MU_G_ERROR_CODE(err) == MU_ERROR_IN_PARAMETERS)
|
||||||
query = get_query (opts);
|
show_usage ();
|
||||||
|
|
||||||
if (!xapian ||!query)
|
return MU_G_ERROR_CODE (err);
|
||||||
return MU_ERROR_INTERNAL;
|
}
|
||||||
|
|
||||||
if (opts->format == MU_CONFIG_FORMAT_XQUERY)
|
if (!execute_find (store, opts, err))
|
||||||
rv = print_xapian_query (xapian, query, &count);
|
return MU_G_ERROR_CODE(err);
|
||||||
else if (opts->exec)
|
|
||||||
rv = exec_cmd_on_query (xapian, query, opts, &count);
|
|
||||||
else
|
else
|
||||||
rv = process_query (xapian, query, opts, &count);
|
return MU_OK;
|
||||||
|
|
||||||
mu_query_destroy (xapian);
|
|
||||||
g_free (query);
|
|
||||||
|
|
||||||
if (!rv)
|
|
||||||
return MU_ERROR;
|
|
||||||
|
|
||||||
return count == 0 ? MU_ERROR_NO_MATCHES : MU_OK;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -39,13 +39,6 @@
|
|||||||
|
|
||||||
static gboolean MU_CAUGHT_SIGNAL;
|
static gboolean MU_CAUGHT_SIGNAL;
|
||||||
|
|
||||||
static void
|
|
||||||
update_warning (void)
|
|
||||||
{
|
|
||||||
g_warning ("note: the database needs to be upgraded to version %s",
|
|
||||||
MU_STORE_SCHEMA_VERSION);
|
|
||||||
g_warning ("please run 'mu index --rebuild' (see the manpage)");
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
sig_handler (int sig)
|
sig_handler (int sig)
|
||||||
@ -79,27 +72,24 @@ install_sig_handler (void)
|
|||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
check_index_or_cleanup_params (MuConfig *opts)
|
check_params (MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
/* param[0] == 'index' or 'cleanup', there should be no
|
/* param[0] == 'index' there should be no param[1] */
|
||||||
* param[1] */
|
|
||||||
if (opts->params[1]) {
|
if (opts->params[1]) {
|
||||||
g_warning ("usage: mu %s [options]", opts->params[0]);
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
return FALSE;
|
"unexpected parameter");
|
||||||
}
|
|
||||||
|
|
||||||
if (opts->linksdir) {
|
|
||||||
g_warning ("invalid option(s) for command");
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (opts->xbatchsize < 0) {
|
if (opts->xbatchsize < 0) {
|
||||||
g_warning ("the Xapian batch size must be non-negative");
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
|
"the batch size must be non-negative");
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (opts->max_msg_size < 0) {
|
if (opts->max_msg_size < 0) {
|
||||||
g_warning ("the maximum message size must be non-negative");
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
|
"the maximum message size must be non-negative");
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -107,21 +97,24 @@ check_index_or_cleanup_params (MuConfig *opts)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
check_maildir (const char *maildir)
|
check_maildir (const char *maildir, GError **err)
|
||||||
{
|
{
|
||||||
if (!maildir) {
|
if (!maildir) {
|
||||||
g_warning ("no maildir to work on; "
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
"use --maildir= to set it explicitly");
|
"no maildir to work on; use --maildir=");
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!g_path_is_absolute (maildir)) {
|
if (!g_path_is_absolute (maildir)) {
|
||||||
g_warning ("maildir path '%s' is not absolute", maildir);
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
|
"maildir path '%s' is not absolute",
|
||||||
|
maildir);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!mu_util_check_dir (maildir, TRUE, FALSE)) {
|
if (!mu_util_check_dir (maildir, TRUE, FALSE)) {
|
||||||
g_warning ("not a valid Maildir: %s", maildir);
|
g_set_error (err, 0, MU_ERROR_IN_PARAMETERS,
|
||||||
|
"not a valid Maildir: %s", maildir);
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -195,18 +188,19 @@ index_msg_cb (MuIndexStats* stats, void *user_data)
|
|||||||
|
|
||||||
|
|
||||||
static gboolean
|
static gboolean
|
||||||
database_version_check_and_update (MuStore *store, MuConfig *opts)
|
database_version_check_and_update (MuStore *store, MuConfig *opts,
|
||||||
|
GError **err)
|
||||||
{
|
{
|
||||||
if (mu_store_count (store) == 0)
|
if (mu_store_count (store, err) == 0)
|
||||||
return TRUE;
|
return TRUE;
|
||||||
|
|
||||||
/* when rebuilding, we empty the database before doing
|
/* when rebuilding, we empty the database before doing
|
||||||
* anything */
|
* anything */
|
||||||
if (opts->rebuild) {
|
if (opts->rebuild) {
|
||||||
opts->reindex = TRUE;
|
opts->reindex = TRUE;
|
||||||
g_message ("clearing database");
|
g_debug ("clearing database");
|
||||||
g_message ("clearing contacts-cache");
|
g_debug ("clearing contacts-cache");
|
||||||
return mu_store_clear (store);
|
return mu_store_clear (store, err);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!mu_store_needs_upgrade (store))
|
if (!mu_store_needs_upgrade (store))
|
||||||
@ -215,11 +209,10 @@ database_version_check_and_update (MuStore *store, MuConfig *opts)
|
|||||||
/* ok, database is not up to date */
|
/* ok, database is not up to date */
|
||||||
if (opts->autoupgrade) {
|
if (opts->autoupgrade) {
|
||||||
opts->reindex = TRUE;
|
opts->reindex = TRUE;
|
||||||
g_message ("auto-upgrade: clearing old database and cache");
|
g_debug ("auto-upgrade: clearing old database and cache");
|
||||||
return mu_store_clear (store);
|
return mu_store_clear (store, err);
|
||||||
}
|
}
|
||||||
|
|
||||||
update_warning ();
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -235,41 +228,10 @@ show_time (unsigned t, unsigned processed)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static gboolean
|
|
||||||
update_maildir_path_maybe (MuIndex *idx, MuConfig *opts)
|
|
||||||
{
|
|
||||||
gchar *exp;
|
|
||||||
|
|
||||||
/* if 'maildir_guessed' is TRUE, we can override it later
|
|
||||||
* with mu_index_last_used_maildir (in mu-cmd-index.c)
|
|
||||||
*/
|
|
||||||
if (!opts->maildir) {
|
|
||||||
|
|
||||||
const char *tmp;
|
|
||||||
|
|
||||||
/* try the last used one */
|
|
||||||
tmp = mu_index_last_used_maildir (idx);
|
|
||||||
if (tmp)
|
|
||||||
opts->maildir = g_strdup (tmp);
|
|
||||||
else
|
|
||||||
opts->maildir = mu_util_guess_maildir ();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (opts->maildir) {
|
|
||||||
exp = mu_util_dir_expand(opts->maildir);
|
|
||||||
if (exp) {
|
|
||||||
g_free(opts->maildir);
|
|
||||||
opts->maildir = exp;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return check_maildir (opts->maildir);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
static MuError
|
static MuError
|
||||||
cmd_cleanup (MuIndex *midx, MuConfig *opts, MuIndexStats *stats,
|
cleanup_missing (MuIndex *midx, MuConfig *opts, MuIndexStats *stats,
|
||||||
gboolean show_progress)
|
gboolean show_progress, GError **err)
|
||||||
{
|
{
|
||||||
MuError rv;
|
MuError rv;
|
||||||
time_t t;
|
time_t t;
|
||||||
@ -277,25 +239,26 @@ cmd_cleanup (MuIndex *midx, MuConfig *opts, MuIndexStats *stats,
|
|||||||
g_message ("cleaning up messages [%s]",
|
g_message ("cleaning up messages [%s]",
|
||||||
mu_runtime_path (MU_RUNTIME_PATH_XAPIANDB));
|
mu_runtime_path (MU_RUNTIME_PATH_XAPIANDB));
|
||||||
|
|
||||||
|
mu_index_stats_clear (stats);
|
||||||
|
|
||||||
t = time (NULL);
|
t = time (NULL);
|
||||||
rv = mu_index_cleanup (midx, stats,
|
rv = mu_index_cleanup (midx, stats,
|
||||||
show_progress ? index_msg_cb : index_msg_silent_cb,
|
show_progress ? index_msg_cb : index_msg_silent_cb,
|
||||||
NULL);
|
NULL, err);
|
||||||
|
|
||||||
if (!opts->quiet) {
|
if (!opts->quiet) {
|
||||||
print_stats (stats, TRUE);
|
print_stats (stats, TRUE);
|
||||||
g_print ("\n");
|
g_print ("\n");
|
||||||
show_time ((unsigned)(time(NULL)-t),stats->_processed);
|
show_time ((unsigned)(time(NULL)-t),stats->_processed);
|
||||||
}
|
}
|
||||||
|
|
||||||
return (rv == MU_OK || rv == MU_STOP) ? MU_OK: MU_ERROR;
|
return (rv == MU_OK || rv == MU_STOP) ? MU_OK: MU_G_ERROR_CODE(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static MuError
|
static MuError
|
||||||
cmd_index (MuIndex *midx, MuConfig *opts, MuIndexStats *stats,
|
cmd_index (MuIndex *midx, MuConfig *opts, MuIndexStats *stats,
|
||||||
gboolean show_progress)
|
gboolean show_progress, GError **err)
|
||||||
{
|
{
|
||||||
MuError rv;
|
MuError rv;
|
||||||
time_t t;
|
time_t t;
|
||||||
@ -318,72 +281,37 @@ cmd_index (MuIndex *midx, MuConfig *opts, MuIndexStats *stats,
|
|||||||
MU_WRITE_LOG ("index: processed: %u; updated/new: %u",
|
MU_WRITE_LOG ("index: processed: %u; updated/new: %u",
|
||||||
stats->_processed, stats->_updated);
|
stats->_processed, stats->_updated);
|
||||||
|
|
||||||
if (rv == MU_OK && !opts->nocleanup) {
|
if (rv == MU_OK && !opts->nocleanup)
|
||||||
mu_index_stats_clear (stats);
|
rv = cleanup_missing (midx, opts, stats, show_progress, err);
|
||||||
rv = cmd_cleanup (midx, opts, stats, show_progress);
|
|
||||||
if (rv == MU_OK) {
|
|
||||||
MU_WRITE_LOG ("cleanup: processed: %u; cleaned-up: %u",
|
|
||||||
stats->_processed, stats->_cleaned_up);
|
|
||||||
return MU_OK;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return (rv==MU_OK||rv==MU_STOP) ? MU_OK : MU_ERROR;
|
if (rv == MU_STOP)
|
||||||
|
rv = MU_OK;
|
||||||
|
|
||||||
|
if (rv != MU_OK && !err)
|
||||||
|
g_set_error (err, 0, rv, "error while indexing");
|
||||||
|
|
||||||
|
return rv;
|
||||||
}
|
}
|
||||||
|
|
||||||
static MuError
|
|
||||||
handle_index_error_and_free (GError *err)
|
|
||||||
{
|
|
||||||
MuError code;
|
|
||||||
|
|
||||||
if (!err)
|
|
||||||
return MU_ERROR_INTERNAL;
|
|
||||||
|
|
||||||
switch (err->code) {
|
|
||||||
|
|
||||||
case MU_ERROR_XAPIAN_CANNOT_GET_WRITELOCK:
|
|
||||||
g_warning ("cannot get Xapian writelock");
|
|
||||||
g_warning ("maybe mu index is already running?");
|
|
||||||
code = MU_ERROR_XAPIAN_CANNOT_GET_WRITELOCK;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case MU_ERROR_XAPIAN_CORRUPTION:
|
|
||||||
g_warning ("xapian database seems to be corrupted");
|
|
||||||
g_warning ("try 'mu index --rebuild");
|
|
||||||
code = MU_ERROR_XAPIAN_CORRUPTION;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
g_warning ("indexing error: %s",
|
|
||||||
err->message ? err->message : "");
|
|
||||||
code = MU_ERROR;
|
|
||||||
}
|
|
||||||
|
|
||||||
g_error_free (err);
|
|
||||||
return code;
|
|
||||||
}
|
|
||||||
|
|
||||||
static MuIndex*
|
static MuIndex*
|
||||||
init_mu_index (MuStore *store, MuConfig *opts, MuError *code)
|
init_mu_index (MuStore *store, MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
MuIndex *midx;
|
MuIndex *midx;
|
||||||
GError *err;
|
|
||||||
|
|
||||||
if (!check_index_or_cleanup_params (opts)) {
|
if (!check_params (opts, err))
|
||||||
*code = MU_ERROR_IN_PARAMETERS;
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
|
||||||
|
|
||||||
if (!database_version_check_and_update(store, opts)) {
|
if (!database_version_check_and_update(store, opts, err))
|
||||||
*code = MU_ERROR_XAPIAN_NOT_UP_TO_DATE;
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
|
||||||
|
|
||||||
err = NULL;
|
|
||||||
midx = mu_index_new (store, &err);
|
if (!check_maildir (opts->maildir, err))
|
||||||
if (!midx) {
|
return NULL;
|
||||||
*code = handle_index_error_and_free (err);
|
|
||||||
|
midx = mu_index_new (store, err);
|
||||||
|
if (!midx)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
|
||||||
|
|
||||||
mu_index_set_max_msg_size (midx, opts->max_msg_size);
|
mu_index_set_max_msg_size (midx, opts->max_msg_size);
|
||||||
mu_index_set_xbatch_size (midx, opts->xbatchsize);
|
mu_index_set_xbatch_size (midx, opts->xbatchsize);
|
||||||
@ -392,24 +320,21 @@ init_mu_index (MuStore *store, MuConfig *opts, MuError *code)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static MuError
|
MuError
|
||||||
cmd_index_or_cleanup (MuStore *store, MuConfig *opts)
|
mu_cmd_index (MuStore *store, MuConfig *opts, GError **err)
|
||||||
{
|
{
|
||||||
MuIndex *midx;
|
MuIndex *midx;
|
||||||
MuIndexStats stats;
|
MuIndexStats stats;
|
||||||
gboolean rv, show_progress;
|
gboolean rv, show_progress;
|
||||||
MuError code;
|
|
||||||
|
g_return_val_if_fail (opts, FALSE);
|
||||||
|
g_return_val_if_fail (opts->cmd == MU_CONFIG_CMD_INDEX,
|
||||||
|
FALSE);
|
||||||
|
|
||||||
/* create, and do error handling if needed */
|
/* create, and do error handling if needed */
|
||||||
midx = init_mu_index (store, opts, &code);
|
midx = init_mu_index (store, opts, err);
|
||||||
if (!midx)
|
if (!midx)
|
||||||
return code;
|
return MU_G_ERROR_CODE(err);
|
||||||
|
|
||||||
/* we determine the maildir path only here, as it may depend on
|
|
||||||
* mu_index_last_used_maildir
|
|
||||||
*/
|
|
||||||
if (!update_maildir_path_maybe (midx, opts))
|
|
||||||
return MU_ERROR_FILE;
|
|
||||||
|
|
||||||
/* note, 'opts->quiet' already cause g_message output not to
|
/* note, 'opts->quiet' already cause g_message output not to
|
||||||
* be shown; here, we make sure we only print progress info if
|
* be shown; here, we make sure we only print progress info if
|
||||||
@ -419,41 +344,9 @@ cmd_index_or_cleanup (MuStore *store, MuConfig *opts)
|
|||||||
mu_index_stats_clear (&stats);
|
mu_index_stats_clear (&stats);
|
||||||
install_sig_handler ();
|
install_sig_handler ();
|
||||||
|
|
||||||
switch (opts->cmd) {
|
rv = cmd_index (midx, opts, &stats, show_progress, err);
|
||||||
case MU_CONFIG_CMD_INDEX:
|
|
||||||
rv = cmd_index (midx, opts, &stats, show_progress); break;
|
|
||||||
case MU_CONFIG_CMD_CLEANUP:
|
|
||||||
rv = cmd_cleanup (midx, opts, &stats, show_progress); break;
|
|
||||||
default:
|
|
||||||
rv = MU_ERROR_INTERNAL;
|
|
||||||
g_assert_not_reached ();
|
|
||||||
}
|
|
||||||
|
|
||||||
mu_index_destroy (midx);
|
mu_index_destroy (midx);
|
||||||
|
|
||||||
return rv;
|
return rv;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
MuError
|
|
||||||
mu_cmd_index (MuStore *store, MuConfig *opts)
|
|
||||||
{
|
|
||||||
g_return_val_if_fail (opts, FALSE);
|
|
||||||
g_return_val_if_fail (opts->cmd == MU_CONFIG_CMD_INDEX,
|
|
||||||
FALSE);
|
|
||||||
|
|
||||||
return cmd_index_or_cleanup (store, opts);
|
|
||||||
}
|
|
||||||
|
|
||||||
MuError
|
|
||||||
mu_cmd_cleanup (MuStore *store, MuConfig *opts)
|
|
||||||
{
|
|
||||||
g_return_val_if_fail (opts, MU_ERROR_INTERNAL);
|
|
||||||
g_return_val_if_fail (opts->cmd != MU_CONFIG_CMD_CLEANUP,
|
|
||||||
MU_ERROR_INTERNAL);
|
|
||||||
|
|
||||||
return cmd_index_or_cleanup (store, opts);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user