diff options
Diffstat (limited to 'camel/providers/imap/camel-imap-utils.c')
-rw-r--r-- | camel/providers/imap/camel-imap-utils.c | 425 |
1 files changed, 235 insertions, 190 deletions
diff --git a/camel/providers/imap/camel-imap-utils.c b/camel/providers/imap/camel-imap-utils.c index 954defabc5..af4cec0cb2 100644 --- a/camel/providers/imap/camel-imap-utils.c +++ b/camel/providers/imap/camel-imap-utils.c @@ -23,24 +23,13 @@ #include <stdio.h> #include <string.h> +#include <gtk/gtk.h> #include "camel-imap-utils.h" #include "string-utils.h" +#include <e-sexp.h> #define d(x) x -struct sexp_node { - struct sexp_node *l_node, *r_node; - char *function; - char *data; -}; - -static char *get_quoted_token (char *string, int *len); -static char *get_token (char *string, int *len); -struct sexp_node *get_sexp_node (const char *exp); -static void print_node (struct sexp_node *node, int depth); -static char *str_sexp_node (struct sexp_node *node); -static void free_sexp_node (struct sexp_node *node); - char * imap_next_word (char *buf) { @@ -109,233 +98,289 @@ imap_parse_list_response (char *buf, char *namespace, char **flags, char **sep, return TRUE; } -static char * -get_quoted_token (char *string, int *len) +struct prop_info { + char *query_prop; + char *imap_attr; +} prop_info_table[] = { + /* query prop, imap attr */ + { "body-contains", "BODY" }, + { "header-contains", "HEADER" } +}; + +static int num_prop_infos = sizeof (prop_info_table) / sizeof (prop_info_table[0]); + +static gchar * +query_prop_to_imap (gchar *query_prop) { - char *ep; - - for (ep = string + 1; *ep; ep++) - if (*ep == '"' && *(ep - 1) != '\\') - break; - if (*ep) - ep++; - - *len = ep - string; - - return g_strndup (string, *len); + int i; + + for (i = 0; i < num_prop_infos; i ++) + if (!strcmp (query_prop, prop_info_table[i].query_prop)) + return prop_info_table[i].imap_attr; + + return NULL; } -static char * -get_token (char *string, int *len) +static ESExpResult * +func_and (struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { - char *p, *ep; + GList **list = data; + ESExpResult *r; + char **strings; - for (p = string; *p && *p == ' '; p++); - - if (*p == '"') { - char *token; + if (argc > 0) { int i; - token = get_quoted_token (p, &i); + strings = g_malloc0 (argc + 3); + strings[0] = g_strdup ("(AND"); + strings[argc+3 - 2] = g_strdup (")"); + strings[argc+3 - 1] = NULL; - *len = i + (p - string); + for (i = 0; i < argc; i++) { + GList *list_head = *list; + strings[argc - i] = (*list)->data; + *list = g_list_remove_link (*list, *list); + g_list_free_1 (list_head); + } - return token; + *list = g_list_prepend (*list, g_strjoinv (" ", strings)); + + for (i = 0 ; i < argc + 2; i++) + g_free (strings[i]); + + g_free (strings); } - for (ep = p; *ep && *ep != ' '; ep++); + r = e_sexp_result_new (ESEXP_RES_BOOL); + r->value.bool = FALSE; - *len = ep - string; - - return g_strndup (p, *len); + return r; } -struct sexp_node * -get_sexp_node (const char *exp) +static ESExpResult * +func_or (struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { - struct sexp_node *node = NULL; - char *p, *ep; - int len; - - switch (*exp) { - case '(': - node = g_malloc0 (sizeof (struct sexp_node)); - node->l_node = NULL; - node->r_node = NULL; - node->data = NULL; - - p = (char *) exp + 1; - - node->function = get_token (p, &len); - - p += len; - for (ep = p; *ep && *ep != '(' && *ep != ')'; ep++); - node->data = g_strndup (p, (gint)(ep - p)); - g_strstrip (node->data); - - p = ep; - - if (*p == '(') - node->r_node = get_sexp_node (p); - else - node->l_node = get_sexp_node (p); - - return node; - break; - case '\0': - return NULL; - break; - case ')': - for (p = (char *) exp + 1; *p && *p == ' '; p++); - return get_sexp_node (p); - break; - default: - node = g_malloc0 (sizeof (struct sexp_node)); - node->l_node = NULL; - node->r_node = NULL; - node->data = NULL; - - p = (char *) exp; + GList **list = data; + ESExpResult *r; + char **strings; + + if (argc > 0) { + int i; - node->function = get_token (p, &len); - - p += len; - for (ep = p; *ep && *ep != '(' && *ep != ')'; ep++); - node->data = g_strndup (p, (gint)(ep - p)); - g_strstrip (node->data); + strings = g_malloc0 (argc+3); + strings[0] = g_strdup ("(OR"); + strings[argc+3 - 2] = g_strdup (")"); + strings[argc+3 - 1] = NULL; + for (i = 0; i < argc; i++) { + GList *list_head = *list; + strings[argc - i] = (*list)->data; + *list = g_list_remove_link (*list, *list); + g_list_free_1 (list_head); + } - p = ep; + *list = g_list_prepend (*list, g_strjoinv (" ", strings)); - if (*p == '(') - node->r_node = get_sexp_node (p); - else - node->l_node = get_sexp_node (p); + for (i = 0 ; i < argc + 2; i++) + g_free (strings[i]); - return node; + g_free (strings); } + + r = e_sexp_result_new (ESEXP_RES_BOOL); + r->value.bool = FALSE; + + return r; } -static void -print_node (struct sexp_node *node, int depth) +static ESExpResult * +func_not (struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { - int i; - - for (i = 0; i < depth; i++) - d(fprintf (stderr, " ")); + GList **list = data; + ESExpResult *r; + + /* just replace the head of the list with the NOT of it. */ + if (argc > 0) { + char *term = (*list)->data; + (*list)->data = g_strdup_printf ("(NOT %s)", term); + g_free (term); + } - d(fprintf (stderr, "%s\n", node->function)); + r = e_sexp_result_new (ESEXP_RES_BOOL); + r->value.bool = FALSE; - if (*node->data) { - for (i = 0; i < depth + 1; i++) - d(fprintf (stderr, " ")); + return r; +} + +static ESExpResult * +func_contains (struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) +{ + GList **list = data; + ESExpResult *r; + + if (argc == 2 + && argv[0]->type == ESEXP_RES_STRING + && argv[1]->type == ESEXP_RES_STRING) { + char *propname = argv[0]->value.string; + char *str = argv[1]->value.string; + char *imap_attr = query_prop_to_imap (propname); + gboolean one_star = FALSE; - d(fprintf (stderr, "%s\n", node->data)); + if (strlen (str) == 0) + one_star = TRUE; + + if (imap_attr) + *list = g_list_prepend (*list, + g_strdup_printf ("(%s=*%s%s)", + imap_attr, + str, + one_star ? "" : "*")); } - if (node->r_node) - print_node (node->r_node, depth + 1); + r = e_sexp_result_new (ESEXP_RES_BOOL); + r->value.bool = FALSE; - if (node->l_node) - print_node (node->l_node, depth); + return r; } -static char *esexp_keys[] = { "or", "body-contains", "header-contains", "match-all", NULL }; -static char *imap_keys[] = { "OR", "BODY", "HEADER", NULL }; - -static char * -str_sexp_node (struct sexp_node *node) +static ESExpResult * +func_is (struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { - char *node_str, *func, *str, *l_str, *r_str; - int i; - - for (i = 0; esexp_keys[i]; i++) - if (!strncmp (esexp_keys[i], node->function, strlen (node->function))) - break; - - if (esexp_keys[i]) - func = imap_keys[i]; - else - func = node->function; - - if (func) { - if (*node->data) - str = g_strdup_printf ("%s %s", func, node->data); - else - str = g_strdup (func); - } else { - str = NULL; + GList **list = data; + ESExpResult *r; + + if (argc == 2 + && argv[0]->type == ESEXP_RES_STRING + && argv[1]->type == ESEXP_RES_STRING) { + char *propname = argv[0]->value.string; + char *str = argv[1]->value.string; + char *imap_attr = query_prop_to_imap (propname); + + if (imap_attr) + *list = g_list_prepend (*list, + g_strdup_printf ("(%s=%s)", + imap_attr, str)); } - r_str = NULL; - if (node->r_node) - r_str = str_sexp_node (node->r_node); - - l_str = NULL; - if (node->l_node) - l_str = str_sexp_node (node->l_node); + r = e_sexp_result_new (ESEXP_RES_BOOL); + r->value.bool = FALSE; - if (str) { - if (r_str) { - if (l_str) - node_str = g_strdup_printf ("(%s (%s)) %s", str, r_str, l_str); - else - node_str = g_strdup_printf ("(%s %s)", str, r_str); - } else { - if (l_str) - node_str = g_strdup_printf ("(%s) %s", str, l_str); - else - node_str = g_strdup_printf ("(%s)", str); - } - } else { - if (r_str) { - if (l_str) - node_str = g_strdup_printf ("(%s) %s", r_str, l_str); - else - node_str = g_strdup_printf ("%s", r_str); - } else { - if (l_str) - node_str = g_strdup_printf ("%s", l_str); - else - node_str = g_strdup (""); - } + return r; +} + +static ESExpResult * +func_beginswith (struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) +{ + GList **list = data; + ESExpResult *r; + + if (argc == 2 + && argv[0]->type == ESEXP_RES_STRING + && argv[1]->type == ESEXP_RES_STRING) { + char *propname = argv[0]->value.string; + char *str = argv[1]->value.string; + char *imap_attr = query_prop_to_imap (propname); + gboolean one_star = FALSE; + + if (strlen(str) == 0) + one_star = TRUE; + + if (imap_attr) + *list = g_list_prepend (*list, + g_strdup_printf ("(%s=%s*)", + imap_attr, + str)); } - g_free (str); - g_free (l_str); - g_free (r_str); + r = e_sexp_result_new (ESEXP_RES_BOOL); + r->value.bool = FALSE; - return node_str; + return r; } -static void -free_sexp_node (struct sexp_node *node) +static ESExpResult * +func_endswith (struct _ESExp *f, int argc, struct _ESExpResult **argv, void *data) { - if (node->r_node) - free_sexp_node (node->r_node); + GList **list = data; + ESExpResult *r; + + if (argc == 2 + && argv[0]->type == ESEXP_RES_STRING + && argv[1]->type == ESEXP_RES_STRING) { + char *propname = argv[0]->value.string; + char *str = argv[1]->value.string; + char *imap_attr = query_prop_to_imap (propname); + gboolean one_star = FALSE; + + if (strlen (str) == 0) + one_star = TRUE; + + if (imap_attr) + *list = g_list_prepend (*list, + g_strdup_printf ("(%s=*%s)", + imap_attr, + str)); + } - if (node->l_node) - free_sexp_node (node->l_node); - - g_free (node->function); - g_free (node->data); - g_free (node); + r = e_sexp_result_new(ESEXP_RES_BOOL); + r->value.bool = FALSE; + + return r; } +/* 'builtin' functions */ +static struct { + char *name; + ESExpFunc *func; + int type; /* set to 1 if a function can perform shortcut evaluation, or + doesn't execute everything, 0 otherwise */ +} symbols[] = { + { "and", func_and, 0 }, + { "or", func_or, 0 }, + { "not", func_not, 0 }, + { "contains", func_contains, 0 }, + { "is", func_is, 0 }, + { "beginswith", func_beginswith, 0 }, + { "endswith", func_endswith, 0 }, +}; + char * imap_translate_sexp (const char *expression) { - struct sexp_node *root; - char *sexp; + ESExp *sexp; + ESExpResult *r; + gchar *retval; + GList *list = NULL; + int i; - root = get_sexp_node (expression); + sexp = e_sexp_new (); - d(print_node (root, 0)); + for (i = 0; i < sizeof (symbols) / sizeof (symbols[0]); i++) { + if (symbols[i].type == 1) { + e_sexp_add_ifunction (sexp, 0, symbols[i].name, + (ESExpIFunc *)symbols[i].func, &list); + } else { + e_sexp_add_function (sexp, 0, symbols[i].name, + symbols[i].func, &list); + } + } + + e_sexp_input_text (sexp, expression, strlen (expression)); + e_sexp_parse (sexp); - sexp = str_sexp_node (root); - sexp[strlen (sexp) - 1] = '\0'; - strcpy (sexp, sexp + 1); + r = e_sexp_eval (sexp); + + gtk_object_unref (GTK_OBJECT (sexp)); + e_sexp_result_free (r); + + if (list->next) { + g_warning ("conversion to imap expression string failed"); + retval = NULL; + g_list_foreach (list, (GFunc)g_free, NULL); + } else { + retval = list->data; + } - free_sexp_node (root); + g_list_free (list); - return sexp; + return retval; } |