mu/src/tests/test-mu-query.c

634 lines
14 KiB
C
Raw Normal View History

/* -*-mode: c; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*-*/
2011-08-29 22:39:11 +02:00
/*
** Copyright (C) 2008-2011 Dirk-Jan C. Binnema <djcb@djcbsoftware.nl>
**
** This program is free software; you can redistribute it and/or modify it
** under the terms of the GNU General Public License as published by the
** Free Software Foundation; either version 3, or (at your option) any
** later version.
**
** This program is distributed in the hope that it will be useful,
** but WITHOUT ANY WARRANTY; without even the implied warranty of
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
** GNU General Public License for more details.
**
** You should have received a copy of the GNU General Public License
** along with this program; if not, write to the Free Software Foundation,
2011-08-29 22:39:11 +02:00
** Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
**
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif /*HAVE_CONFIG_H*/
#include <glib.h>
#include <glib/gstdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <locale.h>
#include "test-mu-common.h"
2010-08-25 20:46:16 +02:00
#include "src/mu-query.h"
2011-05-15 09:40:29 +02:00
#include "src/mu-str.h"
2011-08-30 20:59:05 +02:00
#include "src/mu-store.h"
static gchar*
fill_database (const char *testdir)
{
gchar *cmdline, *tmpdir, *xpath;
2011-08-29 22:39:11 +02:00
tmpdir = test_mu_common_get_random_tmpdir();
cmdline = g_strdup_printf ("%s index --muhome=%s --maildir=%s"
" --quiet",
MU_PROGRAM, tmpdir, testdir);
2010-11-20 13:43:35 +01:00
if (g_test_verbose())
g_printerr ("\n%s\n", cmdline);
2011-08-29 22:39:11 +02:00
2010-11-20 13:43:35 +01:00
g_assert (g_spawn_command_line_sync (cmdline, NULL, NULL,
NULL, NULL));
g_free (cmdline);
xpath= g_strdup_printf ("%s%c%s", tmpdir,
G_DIR_SEPARATOR, "xapian");
g_free (tmpdir);
2011-08-29 22:39:11 +02:00
return xpath;
}
2011-06-18 17:49:46 +02:00
static void
assert_no_dups (MuMsgIter *iter)
{
GHashTable *hash;
hash = g_hash_table_new_full (g_str_hash, g_str_equal,
(GDestroyNotify)g_free, NULL);
mu_msg_iter_reset (iter);
while (!mu_msg_iter_is_done(iter)) {
2011-08-30 20:59:05 +02:00
MuMsg *msg;
msg = mu_msg_iter_get_msg_floating (iter);
2011-06-18 17:49:46 +02:00
/* make sure there are no duplicates */
g_assert (!g_hash_table_lookup (hash, mu_msg_get_path (msg)));
g_hash_table_insert (hash, g_strdup (mu_msg_get_path(msg)),
GUINT_TO_POINTER(TRUE));
mu_msg_iter_next (iter);
}
mu_msg_iter_reset (iter);
g_hash_table_destroy (hash);
}
/* note: this also *moves the iter* */
static guint
run_and_count_matches (const char *xpath, const char *query)
{
MuQuery *mquery;
MuMsgIter *iter;
2011-08-30 20:59:05 +02:00
MuStore *store;
guint count1, count2;
2011-08-29 22:39:11 +02:00
2011-08-30 20:59:05 +02:00
store = mu_store_new_read_only (xpath, NULL);
g_assert (store);
mquery = mu_query_new (store, NULL);
g_assert (query);
2011-08-30 20:59:05 +02:00
mu_store_unref (store);
if (g_test_verbose()) {
char *xs;
g_print ("\n==> query: %s\n", query);
xs = mu_query_preprocess (query, NULL);
g_print ("==> preproc: '%s'\n", xs);
g_free (xs);
xs = mu_query_as_string (mquery, query, NULL);
g_print ("==> xquery: '%s'\n", xs);
g_free (xs);
}
2011-08-29 22:39:11 +02:00
iter = mu_query_run (mquery, query, FALSE, MU_MSG_FIELD_ID_NONE,
FALSE, -1, NULL);
2010-11-11 21:05:10 +01:00
mu_query_destroy (mquery);
g_assert (iter);
2010-11-20 13:43:35 +01:00
2011-06-18 17:49:46 +02:00
assert_no_dups (iter);
2011-08-29 22:39:11 +02:00
/* run query twice, to test mu_msg_iter_reset */
2011-08-29 22:39:11 +02:00
for (count1 = 0; !mu_msg_iter_is_done(iter);
mu_msg_iter_next(iter), ++count1);
2011-06-18 17:49:46 +02:00
mu_msg_iter_reset (iter);
2011-08-29 22:39:11 +02:00
2011-06-18 17:49:46 +02:00
assert_no_dups (iter);
2011-08-29 22:39:11 +02:00
for (count2 = 0; !mu_msg_iter_is_done(iter);
mu_msg_iter_next(iter), ++count2);
2011-08-29 22:39:11 +02:00
mu_msg_iter_destroy (iter);
g_assert_cmpuint (count1, ==, count2);
2011-08-29 22:39:11 +02:00
return count1;
}
typedef struct {
const char *query;
size_t count; /* expected number of matches */
} QResults;
static void
test_mu_query_01 (void)
{
gchar *xpath;
int i;
2011-08-29 22:39:11 +02:00
QResults queries[] = {
{ "basic", 3 },
{ "question", 5 },
{ "thanks", 2 },
{ "html", 4 },
{ "subject:elisp", 1 },
{ "html AND contains", 1 },
{ "html and contains", 1 },
2010-09-04 13:37:19 +02:00
{ "from:pepernoot", 0 },
{ "foo:pepernoot", 0 },
2010-11-11 21:05:10 +01:00
{ "funky", 1 },
2010-09-26 14:23:17 +02:00
{ "fünkÿ", 1 },
{ "", 13 }
};
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
g_free (xpath);
}
2010-02-04 21:48:13 +01:00
static void
test_mu_query_02 (void)
{
const char* q;
gchar *xpath;
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath);
2011-08-29 22:39:11 +02:00
2010-02-09 21:11:15 +01:00
q = "i:f7ccd24b0808061357t453f5962w8b61f9a453b684d0@mail.gmail.com";
2010-02-04 21:48:13 +01:00
g_assert_cmpuint (run_and_count_matches(xpath, q), ==, 1);
2010-02-04 21:48:13 +01:00
g_free (xpath);
}
static void
test_mu_query_03 (void)
{
gchar *xpath;
int i;
2011-08-29 22:39:11 +02:00
QResults queries[] = {
{ "ploughed", 1},
2010-11-20 13:43:35 +01:00
{ "i:3BE9E6535E3029448670913581E7A1A20D852173@"
"emss35m06.us.lmco.com", 1},
2010-11-11 21:05:10 +01:00
/* subsets of the words in the subject should match */
{ "s:gcc include search order" , 1},
{ "s:gcc include search" , 1},
{ "s:search order" , 1},
{ "s:include" , 1},
2011-08-29 22:39:11 +02:00
2010-11-11 21:05:10 +01:00
{ "s:lisp", 1},
{ "s:LISP", 1},
2011-08-29 22:39:11 +02:00
{ "s:\"Re: Learning LISP; Scheme vs elisp.\"", 1},
{ "subject:Re: Learning LISP; Scheme vs elisp.", 0},
{ "subject:\"Re: Learning LISP; Scheme vs elisp.\"", 1},
2010-11-11 21:05:10 +01:00
{ "to:help-gnu-emacs@gnu.org", 4},
{ "t:help-gnu-emacs", 0},
};
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
g_free (xpath);
}
2010-02-04 21:48:13 +01:00
static void
test_mu_query_04 (void)
{
gchar *xpath;
int i;
2011-08-29 22:39:11 +02:00
QResults queries[] = {
{ "frodo@example.com", 1}, /* does not match: see mu-find (1) */
{ "f:frodo@example.com", 1},
{ "f:Frodo Baggins", 1},
{ "bilbo@anotherexample.com", 1}, /* same things */
{ "t:bilbo@anotherexample.com", 1},
{ "t:bilbo", 1},
{ "f:bilbo", 0},
{ "baggins", 1},
{ "prio:h", 1},
{ "prio:high", 1},
2011-06-30 22:43:08 +02:00
{ "prio:normal", 5},
{ "prio:l", 7},
2011-06-30 22:43:08 +02:00
{ "not prio:l", 6},
};
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
g_free (xpath);
}
static void
test_mu_query_accented_chars_01 (void)
{
MuQuery *query;
MuMsgIter *iter;
MuMsg *msg;
2011-08-30 20:59:05 +02:00
MuStore *store;
gchar *xpath;
2011-03-21 00:56:32 +01:00
GError *err;
2011-05-15 09:40:29 +02:00
gchar *summ;
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-30 20:59:05 +02:00
store = mu_store_new_read_only (xpath, NULL);
g_assert (store);
query = mu_query_new (store, NULL);
mu_store_unref (store);
iter = mu_query_run (query, "fünkÿ", FALSE, MU_MSG_FIELD_ID_NONE,
FALSE, -1, NULL);
2011-03-21 00:56:32 +01:00
err = NULL;
2011-08-30 20:59:05 +02:00
msg = mu_msg_iter_get_msg_floating (iter); /* don't unref */
2011-03-21 00:56:32 +01:00
if (!msg) {
g_warning ("error getting message: %s", err->message);
g_error_free (err);
g_assert_not_reached ();
}
2011-08-29 22:39:11 +02:00
g_assert_cmpstr (mu_msg_get_subject(msg),==,
"Greetings from Lothlórien");
2011-05-14 17:07:07 +02:00
/* TODO: fix this again */
2011-05-15 09:40:29 +02:00
summ = mu_str_summarize (mu_msg_get_body_text(msg), 5);
g_assert_cmpstr (summ,==, "Let's write some fünkÿ text using umlauts. Foo.");
g_free (summ);
2011-08-29 22:39:11 +02:00
mu_msg_iter_destroy (iter);
mu_query_destroy (query);
g_free (xpath);
}
static void
test_mu_query_accented_chars_02 (void)
{
gchar *xpath;
int i;
2011-08-29 22:39:11 +02:00
QResults queries[] = {
{ "f:mü", 1},
{ "s:motörhead", 1},
{ "t:Helmut", 1},
2011-08-29 22:39:11 +02:00
{ "t:Kröger", 1},
{ "s:MotorHeäD", 1},
{ "queensryche", 1},
{ "Queensrÿche", 1},
};
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
g_free (xpath);
}
static void
test_mu_query_wildcards (void)
{
gchar *xpath;
int i;
2011-08-29 22:39:11 +02:00
QResults queries[] = {
{ "f:mü", 1},
{ "s:mo*", 1},
{ "t:Helm*", 1},
{ "queensryche", 1},
{ "Queen*", 1},
};
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
g_free (xpath);
}
2011-07-20 23:39:00 +02:00
static void
test_mu_query_dates_helsinki (void)
{
gchar *xpath;
int i;
const char *old_tz;
QResults queries[] = {
{ "date:20080731..20080804", 5},
/* { "date:20080804..20080731", 5}, */
{ "date:20080731..20080804", 5},
{ "date:20080731..20080804 s:gcc", 1},
{ "date:200808110803..now", 1},
{ "date:200808110803..today", 1},
/* { "date:now..2008-08-11-08-03", 1}, */
/* { "date:today..2008-08-11-08-03", 1}, */
{ "date:200808110801..now", 1},
};
old_tz = set_tz ("Europe/Helsinki");
2011-08-29 22:39:11 +02:00
2011-07-20 23:39:00 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
2011-07-20 23:39:00 +02:00
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
g_free (xpath);
set_tz (old_tz);
2011-08-29 22:39:11 +02:00
2011-07-20 23:39:00 +02:00
}
static void
2011-07-20 23:39:00 +02:00
test_mu_query_dates_sydney (void)
{
gchar *xpath;
int i;
2011-07-20 23:39:00 +02:00
const char *old_tz;
QResults queries[] = {
{ "date:20080731..20080804", 5},
/* { "date:20080804..20080731", 5}, */
2011-07-20 23:39:00 +02:00
{ "date:20080731..20080804", 5},
{ "date:20080731..20080804 s:gcc", 1},
{ "date:200808110803..now", 1},
{ "date:200808110803..today", 1},
/* { "date:now..2008-08-11-08-03", 1}, */
/* { "date:today..2008-08-11-08-03", 1}, */
2011-07-20 23:39:00 +02:00
{ "date:200808110801..now", 1},
};
2011-07-20 23:39:00 +02:00
old_tz = set_tz ("Australia/Sydney");
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
g_free (xpath);
2011-07-20 23:39:00 +02:00
set_tz (old_tz);
2011-08-29 22:39:11 +02:00
}
2011-07-20 23:39:00 +02:00
static void
test_mu_query_dates_la (void)
{
gchar *xpath;
int i;
const char *old_tz;
2011-08-29 22:39:11 +02:00
2011-07-20 23:39:00 +02:00
QResults queries[] = {
{ "date:20080731..20080804", 5},
/* { "date:20080804..20080731", 5}, */
{ "date:20080731..20080804", 5},
{ "date:20080731..20080804 s:gcc", 1},
{ "date:200808110803..now", 0},
{ "date:200808110803..today", 0},
/* { "date:now..2008-08-11-08-03", 1}, */
/* { "date:today..2008-08-11-08-03", 1}, */
{ "date:200808110801..now", 0}, /* does not match in LA */
};
2011-08-29 22:39:11 +02:00
2011-07-20 23:39:00 +02:00
old_tz = set_tz ("America/Los_Angeles");
2011-08-29 22:39:11 +02:00
2011-07-20 23:39:00 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
2011-07-20 23:39:00 +02:00
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
g_free (xpath);
set_tz (old_tz);
}
static void
test_mu_query_sizes (void)
{
gchar *xpath;
int i;
2011-08-29 22:39:11 +02:00
QResults queries[] = {
2011-06-30 22:43:08 +02:00
{ "size:0b..2m", 13},
{ "size:2k..4k", 2},
2011-06-30 22:43:08 +02:00
{ "size:2m..0b", 13}
};
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR);
g_assert (xpath != NULL);
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
g_free (xpath);
2011-08-29 22:39:11 +02:00
}
static void
test_mu_query_attach (void)
{
gchar *xpath;
int i;
2011-08-29 22:39:11 +02:00
QResults queries[] = {
2011-11-23 23:15:21 +01:00
{ "j:sittingbull.jpg", 1},
{ "file:custer", 0},
{ "file:custer.jpg", 1}
};
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR2);
g_assert (xpath != NULL);
/* g_print ("(%s)\n", xpath); */
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
2011-08-29 22:39:11 +02:00
g_free (xpath);
}
static void
test_mu_query_tags (void)
{
gchar *xpath;
int i;
2011-08-29 22:39:11 +02:00
QResults queries[] = {
{ "x:paradise", 1},
{ "tag:lost", 1},
{ "tag:lost tag:paradise", 1},
{ "tag:lost tag:horizon", 0},
{ "tag:lost OR tag:horizon", 1},
{ "x:paradise,lost", 0},
};
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR2);
g_assert (xpath != NULL);
/* g_print ("(%s)\n", xpath); */
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i)
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
2011-08-29 22:39:11 +02:00
g_free (xpath);
}
static void
test_mu_query_tags_02 (void)
{
gchar *xpath;
int i;
2011-08-29 22:39:11 +02:00
QResults queries[] = {
{ "x:paradise", 1},
{ "tag:@NextActions", 1},
{ "x:queensrÿche", 1},
{ "tag:lost OR tag:operation*", 2},
};
2011-08-29 22:39:11 +02:00
xpath = fill_database (MU_TESTMAILDIR2);
g_assert (xpath != NULL);
/* g_print ("(%s)\n", xpath); */
2011-08-29 22:39:11 +02:00
for (i = 0; i != G_N_ELEMENTS(queries); ++i) {
/* g_print ("%s\n", queries[i].query); */
g_assert_cmpuint (run_and_count_matches (xpath, queries[i].query),
==, queries[i].count);
}
2011-08-29 22:39:11 +02:00
g_free (xpath);
}
static void
test_mu_query_preprocess (void)
{
unsigned u;
struct {
const gchar *expr, *expected;
} testcases [] = {
{ "hello", "hello" },
{ "/[Gmail].Sent Mail", "__gmail__sent mail" }
/* add more */
};
for (u = 0; u != G_N_ELEMENTS(testcases); ++u) {
gchar *prep;
prep = mu_query_preprocess (testcases[u].expr, NULL);
g_assert_cmpstr (prep, ==, testcases[u].expected);
g_free (prep);
}
}
int
main (int argc, char *argv[])
{
int rv;
2011-08-29 22:39:11 +02:00
g_test_init (&argc, &argv, NULL);
g_test_add_func ("/mu-query/test-mu-query-preprocess",
test_mu_query_preprocess);
g_test_add_func ("/mu-query/test-mu-query-01", test_mu_query_01);
2011-08-29 22:39:11 +02:00
g_test_add_func ("/mu-query/test-mu-query-02", test_mu_query_02);
g_test_add_func ("/mu-query/test-mu-query-03", test_mu_query_03);
g_test_add_func ("/mu-query/test-mu-query-04", test_mu_query_04);
g_test_add_func ("/mu-query/test-mu-query-accented-chars-1",
test_mu_query_accented_chars_01);
g_test_add_func ("/mu-query/test-mu-query-accented-chars-2",
test_mu_query_accented_chars_02);
g_test_add_func ("/mu-query/test-mu-query-wildcards",
test_mu_query_wildcards);
g_test_add_func ("/mu-query/test-mu-query-sizes",
test_mu_query_sizes);
2011-07-20 23:39:00 +02:00
g_test_add_func ("/mu-query/test-mu-query-dates-helsinki",
test_mu_query_dates_helsinki); /* finland */
g_test_add_func ("/mu-query/test-mu-query-dates-sydney",
test_mu_query_dates_sydney);
g_test_add_func ("/mu-query/test-mu-query-dates-la",
test_mu_query_dates_la);
g_test_add_func ("/mu-query/test-mu-query-attach",
test_mu_query_attach);
g_test_add_func ("/mu-query/test-mu-query-tags",
test_mu_query_tags);
g_test_add_func ("/mu-query/test-mu-query-tags_02",
test_mu_query_tags_02);
2011-08-29 22:39:11 +02:00
if (!g_test_verbose())
g_log_set_handler (NULL,
G_LOG_LEVEL_MASK | G_LOG_FLAG_FATAL| G_LOG_FLAG_RECURSION,
(GLogFunc)black_hole, NULL);
rv = g_test_run ();
2011-08-29 22:39:11 +02:00
return rv;
}