[evolution-patches] 65178, maildir issues
- From: Not Zed <notzed ximian com>
- To: asdf <evolution-patches lists ximian com>
- Subject: [evolution-patches] 65178, maildir issues
- Date: Mon, 17 Jan 2005 14:14:11 +0800
It changes a few things with maildir stores:
- fixes the bug
- doesn't put folders under "inbox" in the display anymore, they all go in the top level
- fixes a long-existing vfolder bug with folder uri comparisons, needed for the next bit
- puts code in to handle old uri's with folders under inbox transparently, so that old vfolders and filters still work without needing an upgrade
- adds a new test case for maildir stores (should probably loop it and test all local stores similarly).
Since this is quite big, i'm not sure if this should go into 2.0.x, this patch is against head but it should also apply to 2.0.x.
? camel/camel-mime-tables.c
Index: camel/ChangeLog
===================================================================
RCS file: /cvs/gnome/evolution-data-server/camel/ChangeLog,v
retrieving revision 1.2371
diff -u -p -r1.2371 ChangeLog
--- camel/ChangeLog 13 Jan 2005 21:22:48 -0000 1.2371
+++ camel/ChangeLog 17 Jan 2005 05:57:14 -0000
@@ -1,3 +1,28 @@
+2005-01-14 Not Zed <NotZed Ximian com>
+
+ ** Better fix for #65178.
+
+ * tests/folder/test11.c: new maildir test.
+
+ * camel-store.c (camel_store_folder_uri_equal): dont pass NULL
+ values to compare_folder_name.
+
+ * providers/local/camel-maildir-store.c (get_folder_info):
+ reimplemented, now everything isn't a subfolder of INBOX ('.')
+ anymore.
+ (get_inbox): call main entry point so we don't create multiple
+ inboxes each call.
+ (scan_dirs): re-implemented using non-recursive implementation.
+ (scan_fi, scan_free, scan_equal, scan_hash): helpers for above.
+ (md_canon_name): helper to canonicalise old-format names to new
+ ones.
+ (maildir_hash_folder_name, maildir_compare_folder_name): implement
+ custom folder hash functions so canonicalisation works.
+ (get_folder): canoncalise name before using it. special case '.'
+ aka 'inbox', so that it is implicitly created if accessed.
+ (delete_folder): don't let the caller delete ".".
+ (camel_folder_info_new): removed.
+
2005-01-12 Jeffrey Stedfast <fejj novell com>
* providers/imap4/camel-imap4-provider.c: Use the same key
Index: camel/camel-store.c
===================================================================
RCS file: /cvs/gnome/evolution-data-server/camel/camel-store.c,v
retrieving revision 1.155
diff -u -p -r1.155 camel-store.c
--- camel/camel-store.c 4 Nov 2004 05:34:05 -0000 1.155
+++ camel/camel-store.c 17 Jan 2005 05:57:14 -0000
@@ -1196,7 +1196,7 @@ camel_store_folder_uri_equal (CamelStore
name1 = url1->path[0] == '/' ? url1->path + 1 : url1->path;
}
- equal = CS_CLASS (store)->compare_folder_name (name0, name1);
+ equal = name0 && name1 && CS_CLASS (store)->compare_folder_name (name0, name1);
}
camel_url_free (url0);
Index: camel/providers/local/camel-maildir-store.c
===================================================================
RCS file: /cvs/gnome/evolution-data-server/camel/providers/local/camel-maildir-store.c,v
retrieving revision 1.41
diff -u -p -r1.41 camel-maildir-store.c
--- camel/providers/local/camel-maildir-store.c 13 Jan 2005 04:00:11 -0000 1.41
+++ camel/providers/local/camel-maildir-store.c 17 Jan 2005 05:57:14 -0000
@@ -55,6 +55,9 @@ static void maildir_rename_folder(CamelS
static CamelFolderInfo * get_folder_info (CamelStore *store, const char *top, guint32 flags, CamelException *ex);
+static gboolean maildir_compare_folder_name(const void *a, const void *b);
+static guint maildir_hash_folder_name(const void *a);
+
static void camel_maildir_store_class_init(CamelObjectClass * camel_maildir_store_class)
{
CamelStoreClass *camel_store_class = CAMEL_STORE_CLASS(camel_maildir_store_class);
@@ -63,6 +66,8 @@ static void camel_maildir_store_class_in
parent_class = (CamelLocalStoreClass *)camel_type_get_global_classfuncs(camel_local_store_get_type());
/* virtual method overload, use defaults for most */
+ camel_store_class->hash_folder_name = maildir_hash_folder_name;
+ camel_store_class->compare_folder_name = maildir_compare_folder_name;
camel_store_class->get_folder = get_folder;
camel_store_class->get_inbox = get_inbox;
camel_store_class->delete_folder = delete_folder;
@@ -89,31 +94,68 @@ CamelType camel_maildir_store_get_type(v
return camel_maildir_store_type;
}
+/* This fixes up some historical cruft of names starting with "./" */
+static const char *
+md_canon_name(const char *a)
+{
+ if (a != NULL) {
+ if (a[0] == '/')
+ a++;
+ if (a[0] == '.' && a[1] == '/')
+ a+=2;
+ }
+ return a;
+}
+
+static guint maildir_hash_folder_name(const void *a)
+{
+ return g_str_hash(md_canon_name(a));
+}
+
+static gboolean maildir_compare_folder_name(const void *a, const void *b)
+{
+ return g_str_equal(md_canon_name(a), md_canon_name(b));
+}
+
static CamelFolder *
-get_folder(CamelStore * store, const char *infolder_name, guint32 flags, CamelException * ex)
+get_folder(CamelStore * store, const char *folder_name, guint32 flags, CamelException * ex)
{
- char *name, *tmp, *cur, *new, *folder_namea = NULL;
- const char *folder_name;
+ char *name, *tmp, *cur, *new;
struct stat st;
CamelFolder *folder = NULL;
- /* HACK: if we try to create a folder "/xxx", convert it to implicit "./xxx" as getfolderinfo will */
- if (infolder_name[0] == '/')
- folder_name = folder_namea = g_strdup_printf(".%s", infolder_name);
- else
- folder_name = infolder_name;
+ printf("maildir: get_folder('%s' -> '%s') = ", folder_name, md_canon_name(folder_name));
+
+ folder_name = md_canon_name(folder_name);
- if (!((CamelStoreClass *)parent_class)->get_folder(store, folder_name, flags, ex)) {
- g_free(folder_namea);
+ if (!((CamelStoreClass *)parent_class)->get_folder(store, folder_name, flags, ex))
return NULL;
- }
name = g_strdup_printf("%s%s", CAMEL_LOCAL_STORE(store)->toplevel_dir, folder_name);
tmp = g_strdup_printf("%s/tmp", name);
cur = g_strdup_printf("%s/cur", name);
new = g_strdup_printf("%s/new", name);
- if (stat(name, &st) == -1) {
+ if (!strcmp(folder_name, ".")) {
+ /* special case "." (aka inbox), may need to be created */
+ if (stat(tmp, &st) != 0 || !S_ISDIR(st.st_mode)
+ || stat(cur, &st) != 0 || !S_ISDIR(st.st_mode)
+ || stat(new, &st) != 0 || !S_ISDIR(st.st_mode)) {
+ if (mkdir(tmp, 0700) != 0
+ || mkdir(cur, 0700) != 0
+ || mkdir(new, 0700) != 0) {
+ camel_exception_setv(ex, CAMEL_EXCEPTION_SYSTEM,
+ _("Cannot create folder `%s': %s"),
+ folder_name, g_strerror(errno));
+ rmdir(tmp);
+ rmdir(cur);
+ rmdir(new);
+ goto fail;
+ }
+ }
+ folder = camel_maildir_folder_new(store, folder_name, flags, ex);
+ } else if (stat(name, &st) == -1) {
+ /* folder doesn't exist, see if we should create it */
if (errno != ENOENT) {
camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
_("Cannot get folder `%s': %s"),
@@ -142,6 +184,7 @@ get_folder(CamelStore * store, const cha
|| stat(tmp, &st) != 0 || !S_ISDIR(st.st_mode)
|| stat(cur, &st) != 0 || !S_ISDIR(st.st_mode)
|| stat(new, &st) != 0 || !S_ISDIR(st.st_mode)) {
+ /* folder exists, but not maildir */
camel_exception_setv(ex, CAMEL_EXCEPTION_SYSTEM,
_("Cannot get folder `%s': not a maildir directory."), name);
} else if (flags & CAMEL_STORE_FOLDER_EXCL) {
@@ -151,12 +194,13 @@ get_folder(CamelStore * store, const cha
} else {
folder = camel_maildir_folder_new(store, folder_name, flags, ex);
}
-
+fail:
g_free(name);
g_free(tmp);
g_free(cur);
g_free(new);
- g_free(folder_namea);
+
+ printf("%p\n", folder);
return folder;
}
@@ -164,7 +208,7 @@ get_folder(CamelStore * store, const cha
static CamelFolder *
get_inbox (CamelStore *store, CamelException *ex)
{
- return get_folder (store, ".", 0, ex);
+ return camel_store_get_folder(store, ".", CAMEL_STORE_FOLDER_CREATE, ex);
}
static void delete_folder(CamelStore * store, const char *folder_name, CamelException * ex)
@@ -172,6 +216,12 @@ static void delete_folder(CamelStore * s
char *name, *tmp, *cur, *new;
struct stat st;
+ if (strcmp(folder_name, ".") == 0) {
+ camel_exception_setv(ex, CAMEL_EXCEPTION_STORE_NO_FOLDER,
+ _("Cannot delete folder: %s: Invalid operation"), _("Inbox"));
+ return;
+ }
+
name = g_strdup_printf("%s%s", CAMEL_LOCAL_STORE(store)->toplevel_dir, folder_name);
tmp = g_strdup_printf("%s/tmp", name);
@@ -247,26 +297,6 @@ maildir_rename_folder(CamelStore *store,
((CamelStoreClass *)parent_class)->rename_folder(store, old, new, ex);
}
-static CamelFolderInfo *camel_folder_info_new(CamelURL *url, const char *full, const char *name)
-{
- CamelFolderInfo *fi;
-
- fi = g_malloc0(sizeof(*fi));
- fi->uri = camel_url_to_string(url, 0);
- fi->full_name = g_strdup(full);
- if (!strcmp(full, ".")) {
- fi->flags |= CAMEL_FOLDER_SYSTEM;
- fi->name = g_strdup(_("Inbox"));
- } else
- fi->name = g_strdup(name);
- fi->unread = -1;
- fi->total = -1;
-
- d(printf("Adding maildir info: '%s' '%s' '%s'\n", fi->name, fi->full_name, fi->uri));
-
- return fi;
-}
-
static void
fill_fi(CamelStore *store, CamelFolderInfo *fi, guint32 flags)
{
@@ -304,69 +334,112 @@ fill_fi(CamelStore *store, CamelFolderIn
}
}
-/* used to find out where we've visited already */
-struct _inode {
+struct _scan_node {
+ struct _scan_node *next;
+ struct _scan_node *prev;
+
+ CamelFolderInfo *fi;
+
dev_t dnode;
ino_t inode;
};
-/* returns number of records found at or below this level */
-static int scan_dir(CamelStore *store, GHashTable *visited, CamelURL *url, const char *path, guint32 flags, CamelFolderInfo *parent, CamelFolderInfo **fip, CamelException *ex)
+static guint scan_hash(const void *d)
{
- DIR *dir;
- struct dirent *d;
- char *name, *tmp, *cur, *new;
- const char *base, *root = ((CamelService *)store)->url->path;
- CamelFolderInfo *fi = NULL;
- struct stat st;
+ const struct _scan_node *v = d;
- /* look for folders matching the right structure, recursively */
- name = g_strdup_printf("%s/%s", root, path);
+ return v->inode ^ v->dnode;
+}
- d(printf("checking dir '%s' part '%s' for maildir content\n", root, path));
+static gboolean scan_equal(const void *a, const void *b)
+{
+ const struct _scan_node *v1 = a, *v2 = b;
+
+ return v1->inode == v2->inode && v1->dnode == v2->dnode;
+}
- tmp = g_strdup_printf("%s/tmp", name);
- cur = g_strdup_printf("%s/cur", name);
- new = g_strdup_printf("%s/new", name);
+static void scan_free(void *k, void *v, void *d)
+{
+ g_free(k);
+}
- base = strrchr(path, '/');
- if (base)
- base++;
- else
- base = path;
+static CamelFolderInfo *scan_fi(CamelStore *store, guint32 flags, CamelURL *url, const char *full, const char *name)
+{
+ CamelFolderInfo *fi;
+ char *tmp, *cur, *new;
+ struct stat st;
- camel_url_set_fragment(url, path);
+ fi = g_malloc0(sizeof(*fi));
+ fi->full_name = g_strdup(full);
+ fi->name = g_strdup(name);
+ camel_url_set_fragment(url, fi->full_name);
+ fi->uri = camel_url_to_string(url, 0);
+ fi->unread = -1;
+ fi->total = -1;
+ /* we only calculate nochildren properly if we're recursive */
+ if (((flags & CAMEL_STORE_FOLDER_INFO_RECURSIVE) != 0))
+ fi->flags = CAMEL_FOLDER_NOCHILDREN;
- fi = camel_folder_info_new(url, path, base);
- fill_fi(store, fi, flags);
+ d(printf("Adding maildir info: '%s' '%s' '%s'\n", fi->name, fi->full_name, fi->uri));
+
+ tmp = g_build_filename(url->path, fi->full_name, "tmp", NULL);
+ cur = g_build_filename(url->path, fi->full_name, "cur", NULL);
+ new = g_build_filename(url->path, fi->full_name, "new", NULL);
if (!(stat(tmp, &st) == 0 && S_ISDIR(st.st_mode)
&& stat(cur, &st) == 0 && S_ISDIR(st.st_mode)
&& stat(new, &st) == 0 && S_ISDIR(st.st_mode)))
fi->flags |= CAMEL_FOLDER_NOSELECT;
- d(printf("found! uri = %s\n", fi->uri));
- d(printf(" full_name = %s\n name = '%s'\n", fi->full_name, fi->name));
-
- fi->parent = parent;
- fi->next = *fip;
- *fip = fi;
-
- g_free(tmp);
- g_free(cur);
g_free(new);
+ g_free(cur);
+ g_free(tmp);
+
+ fill_fi(store, fi, flags);
+
+ return fi;
+}
+
+static int
+scan_dirs(CamelStore *store, guint32 flags, CamelFolderInfo *topfi, CamelURL *url, CamelException *ex)
+{
+ EDList queue = E_DLIST_INITIALISER(queue);
+ struct _scan_node *sn;
+ const char *root = ((CamelService *)store)->url->path;
+ char *tmp;
+ GHashTable *visited;
+ struct stat st;
+ int res = -1;
+
+ visited = g_hash_table_new(scan_hash, scan_equal);
+
+ sn = g_malloc0(sizeof(*sn));
+ sn->fi = topfi;
+ e_dlist_addtail(&queue, (EDListNode *)sn);
+ g_hash_table_insert(visited, sn, sn);
+
+ while (!e_dlist_empty(&queue)) {
+ char *name;
+ DIR *dir;
+ struct dirent *d;
+ CamelFolderInfo *last;
- /* always look further if asked */
- if (((flags & CAMEL_STORE_FOLDER_INFO_RECURSIVE) || parent == NULL)) {
- int children = 0;
+ sn = (struct _scan_node *)e_dlist_remhead(&queue);
+
+ last = (CamelFolderInfo *)&sn->fi->child;
+
+ if (!strcmp(sn->fi->full_name, "."))
+ name = g_strdup(root);
+ else
+ name = g_build_filename(root, sn->fi->full_name, NULL);
dir = opendir(name);
if (dir == NULL) {
- camel_exception_setv (ex, CAMEL_EXCEPTION_SYSTEM,
- _("Could not scan folder `%s': %s"),
- root, g_strerror (errno));
g_free(name);
- return -1;
+ camel_exception_setv(ex, CAMEL_EXCEPTION_SYSTEM,
+ _("Could not scan folder `%s': %s"),
+ root, g_strerror(errno));
+ goto fail;
}
while ( (d = readdir(dir)) ) {
@@ -377,95 +450,100 @@ static int scan_dir(CamelStore *store, G
|| strcmp(d->d_name, "..") == 0)
continue;
- tmp = g_strdup_printf("%s/%s", name, d->d_name);
+ tmp = g_build_filename(name, d->d_name, NULL);
if (stat(tmp, &st) == 0 && S_ISDIR(st.st_mode)) {
- struct _inode in = { st.st_dev, st.st_ino };
+ struct _scan_node in;
+
+ in.dnode = st.st_dev;
+ in.inode = st.st_ino;
/* see if we've visited already */
if (g_hash_table_lookup(visited, &in) == NULL) {
- struct _inode *inew = g_malloc(sizeof(*inew));
+ struct _scan_node *snew = g_malloc(sizeof(*snew));
+ char *full;
+
+ snew->dnode = in.dnode;
+ snew->inode = in.inode;
+
+ if (!strcmp(sn->fi->full_name, "."))
+ full = g_strdup(d->d_name);
+ else
+ full = g_strdup_printf("%s/%s", sn->fi->full_name, d->d_name);
+ snew->fi = scan_fi(store, flags, url, full, d->d_name);
+ g_free(full);
+
+ last->next = snew->fi;
+ last = snew->fi;
+ snew->fi->parent = sn->fi;
+
+ sn->fi->flags &= ~CAMEL_FOLDER_NOCHILDREN;
+ sn->fi->flags |= CAMEL_FOLDER_CHILDREN;
- children++;
+ g_hash_table_insert(visited, snew, snew);
- *inew = in;
- g_hash_table_insert(visited, inew, inew);
- new = g_strdup_printf("%s/%s", path, d->d_name);
- if (scan_dir(store, visited, url, new, flags, fi, &fi->child, ex) == -1) {
- g_free(tmp);
- g_free(new);
- closedir(dir);
- return -1;
- }
- g_free(new);
+ if (((flags & CAMEL_STORE_FOLDER_INFO_RECURSIVE) != 0))
+ e_dlist_addtail(&queue, (EDListNode *)snew);
}
}
g_free(tmp);
}
closedir(dir);
-
- if (children)
- fi->flags |= CAMEL_FOLDER_CHILDREN;
- else
- fi->flags |= CAMEL_FOLDER_NOCHILDREN;
}
- g_free(name);
-
- return 0;
-}
-
-static guint inode_hash(const void *d)
-{
- const struct _inode *v = d;
-
- return v->inode ^ v->dnode;
-}
+ res = 0;
+fail:
+ g_hash_table_foreach(visited, scan_free, NULL);
+ g_hash_table_destroy(visited);
-static gboolean inode_equal(const void *a, const void *b)
-{
- const struct _inode *v1 = a, *v2 = b;
-
- return v1->inode == v2->inode && v1->dnode == v2->dnode;
-}
-
-static void inode_free(void *k, void *v, void *d)
-{
- g_free(k);
+ return res;
}
static CamelFolderInfo *
-get_folder_info (CamelStore *store, const char *topin, guint32 flags, CamelException *ex)
+get_folder_info (CamelStore *store, const char *top, guint32 flags, CamelException *ex)
{
CamelFolderInfo *fi = NULL;
CamelLocalStore *local_store = (CamelLocalStore *)store;
- GHashTable *visited;
CamelURL *url;
- const char *top;
- char *topa = NULL;
-
- visited = g_hash_table_new(inode_hash, inode_equal);
url = camel_url_new("maildir:", NULL);
camel_url_set_path(url, ((CamelService *)local_store)->url->path);
- /* HACK: if we try to get folderinfo from "/xxx", convert it to implicit "./xxx" as getfolderinfo will later */
- if (topin == NULL || topin[0] == 0)
- top = ".";
- else if (topin[0] == '/')
- top = topa = g_strdup_printf(".%s", topin);
- else
- top = topin;
+ if (top == NULL || top[0] == 0) {
+ CamelFolderInfo *scan;
- if (scan_dir(store, visited, url, top, flags, NULL, &fi, ex) == -1 && fi != NULL) {
- camel_store_free_folder_info_full(store, fi);
- fi = NULL;
- }
+ /* create a dummy "." parent inbox, use to scan, then put back at the top level */
+ fi = scan_fi(store, flags, url, ".", _("Inbox"));
+ if (scan_dirs(store, flags, fi, url, ex) == -1)
+ goto fail;
+ fi->next = fi->child;
+ scan = fi->child;
+ fi->child = NULL;
+ while (scan) {
+ scan->parent = NULL;
+ scan = scan->next;
+ }
+ fi->flags &= ~CAMEL_FOLDER_CHILDREN;
+ fi->flags |= CAMEL_FOLDER_SYSTEM|CAMEL_FOLDER_NOCHILDREN|CAMEL_FOLDER_NOINFERIORS;
+ } else if (!strcmp(top, ".")) {
+ fi = scan_fi(store, flags, url, ".", _("Inbox"));
+ fi->flags |= CAMEL_FOLDER_SYSTEM|CAMEL_FOLDER_NOCHILDREN|CAMEL_FOLDER_NOINFERIORS;
+ } else {
+ const char *name = strrchr(top, '/');
- g_free(topa);
+ fi = scan_fi(store, flags, url, top, name?name+1:top);
+ if (scan_dirs(store, flags, fi, url, ex) == -1)
+ goto fail;
+ }
camel_url_free(url);
- g_hash_table_foreach(visited, inode_free, NULL);
- g_hash_table_destroy(visited);
return fi;
+
+fail:
+ if (fi)
+ camel_store_free_folder_info_full(store, fi);
+
+ camel_url_free(url);
+
+ return NULL;
}
Index: camel/tests/folder/Makefile.am
===================================================================
RCS file: /cvs/gnome/evolution-data-server/camel/tests/folder/Makefile.am,v
retrieving revision 1.14
diff -u -p -r1.14 Makefile.am
--- camel/tests/folder/Makefile.am 13 Dec 2004 03:08:51 -0000 1.14
+++ camel/tests/folder/Makefile.am 17 Jan 2005 05:57:14 -0000
@@ -20,9 +20,9 @@ check_PROGRAMS = \
test1 test2 test3 \
test4 test5 test6 \
test7 test8 test9 \
- test10
+ test10 test11
TESTS = test1 test2 test3 \
test4 test5 test6 \
test7 test8 test9 \
- test10
+ test10 test11
Index: camel/tests/folder/README
===================================================================
RCS file: /cvs/gnome/evolution-data-server/camel/tests/folder/README,v
retrieving revision 1.5
diff -u -p -r1.5 README
--- camel/tests/folder/README 23 Apr 2003 01:34:03 -0000 1.5
+++ camel/tests/folder/README 17 Jan 2005 05:57:14 -0000
@@ -10,3 +10,5 @@ test7 basic folder operations, NNTP
test8 multithreaded folder torture test, local
test9 filtering
test10 multithreaded folder/store object bag torture test
+
+test11 old format maildir name compatability
Index: camel/tests/folder/test11.c
===================================================================
RCS file: camel/tests/folder/test11.c
diff -N camel/tests/folder/test11.c
--- /dev/null 1 Jan 1970 00:00:00 -0000
+++ camel/tests/folder/test11.c 17 Jan 2005 05:57:14 -0000
@@ -0,0 +1,192 @@
+
+/* threaded folder testing */
+
+#include <string.h>
+#include <pthread.h>
+
+#include "camel-test.h"
+#include "camel-test-provider.h"
+#include "session.h"
+
+#include <camel/camel-exception.h>
+#include <camel/camel-service.h>
+#include <camel/camel-store.h>
+
+#define MAX_LOOP (10000)
+#define MAX_THREADS (5)
+
+#define d(x)
+
+#define ARRAY_LEN(x) (sizeof(x)/sizeof(x[0]))
+
+static const char *local_drivers[] = { "local" };
+
+static CamelSession *session;
+
+/* FIXME: flags aren't really right yet */
+/* ASCII sorted on full_name */
+static CamelFolderInfo fi_list_1[] = {
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#.", "Inbox", ".", CAMEL_FOLDER_SYSTEM|CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#Junk", "Junk", "Junk", CAMEL_FOLDER_SYSTEM|CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#Trash", "Trash", "Trash", CAMEL_FOLDER_SYSTEM|CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#testbox", "testbox", "testbox", CAMEL_FOLDER_CHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#testbox/foo", "foo", "testbox/foo", CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#testbox2", "testbox2", "testbox2", CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+};
+
+static CamelFolderInfo fi_list_2[] = {
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#.", "Inbox", ".", CAMEL_FOLDER_SYSTEM|CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#Junk", "Junk", "Junk", CAMEL_FOLDER_SYSTEM|CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#Trash", "Trash", "Trash", CAMEL_FOLDER_SYSTEM|CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#testbox", "testbox", "testbox", CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#testbox2", "testbox2", "testbox2", CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+};
+
+static CamelFolderInfo fi_list_3[] = {
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#testbox", "testbox", "testbox", CAMEL_FOLDER_CHILDREN, -1, -1 },
+ { NULL, NULL, NULL, "maildir:/tmp/camel-test/maildir#testbox/foo", "foo", "testbox/foo", CAMEL_FOLDER_NOCHILDREN, -1, -1 },
+};
+
+static int
+cmp_fi(const void *a, const void *b)
+{
+ const CamelFolderInfo *fa = ((const CamelFolderInfo **)a)[0];
+ const CamelFolderInfo *fb = ((const CamelFolderInfo **)b)[0];
+
+ return strcmp(fa->full_name, fb->full_name);
+}
+
+static void
+add_fi(GPtrArray *folders, CamelFolderInfo *fi)
+{
+ while (fi) {
+ g_ptr_array_add(folders, fi);
+ if (fi->child)
+ add_fi(folders, fi->child);
+ fi = fi->next;
+ }
+}
+
+static void
+check_fi(CamelFolderInfo *fi, CamelFolderInfo *list, int len)
+{
+ GPtrArray *folders = g_ptr_array_new();
+ int i;
+
+ add_fi(folders, fi);
+ check_msg(folders->len == len, "unexpected number of folders returned from folderinfo");
+ qsort(folders->pdata, folders->len, sizeof(folders->pdata[0]), cmp_fi);
+ for (i=0;i<len;i++) {
+ CamelFolderInfo *f = folders->pdata[i];
+
+ camel_test_push("checking folder '%s'", list[i].uri);
+
+ check_msg(!strcmp(f->uri, list[i].uri), "got '%s' expecting '%s'", f->uri, list[i].uri);
+ check(!strcmp(f->full_name, list[i].full_name));
+
+ /* this might be translated, but we can't know */
+ camel_test_nonfatal("Inbox not english");
+ check(!strcmp(f->name, list[i].name));
+ camel_test_fatal();
+
+ camel_test_nonfatal("Flags mismatch");
+ check(f->flags == list[i].flags);
+ camel_test_fatal();
+
+ camel_test_pull();
+ }
+
+ g_ptr_array_free(folders, TRUE);
+}
+
+int main(int argc, char **argv)
+{
+ CamelException *ex;
+ CamelFolder *f1, *f2;
+ CamelStore *store;
+ CamelFolderInfo *fi;
+
+ camel_test_init(argc, argv);
+ camel_test_provider_init(1, local_drivers);
+
+ ex = camel_exception_new();
+
+ /* clear out any camel-test data */
+ system("/bin/rm -rf /tmp/camel-test");
+
+ session = camel_test_session_new("/tmp/camel-test");
+ store = camel_session_get_store(session, "maildir:///tmp/camel-test/maildir", ex);
+ camel_exception_clear(ex);
+
+ camel_test_start("Maildir backward compatability tests");
+
+ camel_test_push("./ prefix path, one level");
+ f1 = camel_store_get_folder(store, "testbox", CAMEL_STORE_FOLDER_CREATE, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ f2 = camel_store_get_folder(store, "./testbox", CAMEL_STORE_FOLDER_CREATE, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ check(f1 == f2);
+ check_unref(f2, 2);
+ check_unref(f1, 1);
+ camel_test_pull();
+
+ camel_test_push("./ prefix path, one level, no create");
+ f1 = camel_store_get_folder(store, "testbox2", CAMEL_STORE_FOLDER_CREATE, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ f2 = camel_store_get_folder(store, "./testbox2", 0, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ check(f1 == f2);
+ check_unref(f2, 2);
+ check_unref(f1, 1);
+ camel_test_pull();
+
+ camel_test_push("./ prefix path, two levels");
+ f1 = camel_store_get_folder(store, "testbox/foo", CAMEL_STORE_FOLDER_CREATE, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ f2 = camel_store_get_folder(store, "./testbox/foo", CAMEL_STORE_FOLDER_CREATE, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ check(f1 == f2);
+ check_unref(f2, 2);
+ check_unref(f1, 1);
+ camel_test_pull();
+
+ camel_test_push("'.' == Inbox");
+ f2 = camel_store_get_inbox(store, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ f1 = camel_store_get_folder(store, ".", 0, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ check(f1 == f2);
+ check_unref(f2, 2);
+ check_unref(f1, 1);
+ camel_test_pull();
+
+ camel_test_push("folder info, recursive");
+ fi = camel_store_get_folder_info(store, "", CAMEL_STORE_FOLDER_INFO_RECURSIVE, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ check(fi != NULL);
+ check_fi(fi, fi_list_1, ARRAY_LEN(fi_list_1));
+ camel_test_pull();
+
+ camel_test_push("folder info, flat");
+ fi = camel_store_get_folder_info(store, "", 0, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ check(fi != NULL);
+ check_fi(fi, fi_list_2, ARRAY_LEN(fi_list_2));
+ camel_test_pull();
+
+ camel_test_push("folder info, recursive, non root");
+ fi = camel_store_get_folder_info(store, "testbox", CAMEL_STORE_FOLDER_INFO_RECURSIVE, ex);
+ check_msg(!camel_exception_is_set(ex), "%s", camel_exception_get_description(ex));
+ check(fi != NULL);
+ check_fi(fi, fi_list_3, ARRAY_LEN(fi_list_3));
+ camel_test_pull();
+
+ check_unref(store, 1);
+ check_unref(session, 1);
+
+ camel_exception_free(ex);
+
+ camel_test_end();
+
+ return 0;
+}
? mail/GNOME_Evolution_Mail.oaf
? mail/GNOME_Evolution_Mail.oafinfo
? mail/Mail-common.c
? mail/Mail-skels.c
? mail/Mail-stubs.c
? mail/Mail.h
? mail/Mailer-common.c
? mail/Mailer-skels.c
? mail/Mailer-stubs.c
? mail/Mailer.h
? mail/a
? mail/a.diff
? mail/a.out
? mail/a.txt
? mail/all.txt
? mail/b
? mail/blah.c
? mail/blah.ps
? mail/changes.diff
? mail/conf.keys
? mail/day.diff
? mail/diff
? mail/e-plugin.html
? mail/econf.h
? mail/em-message-view.h
? mail/em-security-info.c
? mail/em-store-view.c
? mail/evolution-mail
? mail/evolution-mail-ops.log
? mail/evolution-mbox-upgrade
? mail/f.diff
? mail/fix.ed
? mail/foo
? mail/hide-empty.diff
? mail/html.diff
? mail/lockdown.txt
? mail/m.diff
? mail/mail-config.c.save
? mail/mail-config.evolution
? mail/mail-search.gladep
? mail/mail-security.gladep
? mail/mail-send-recv.c.save
? mail/mail.diff
? mail/map.sed
? mail/message-list.c.save
? mail/ml.diff
? mail/old
? mail/out
? mail/output.ps
? mail/plugins
? mail/search-types.xml
? mail/store_change.diff
? mail/subscribe-dialog-new.c
? mail/subscribe-dialog.c.new
? mail/subscribe-dialog.glade.backup
? mail/subscribe-dialog.glade.save
? mail/subscribe-dialog.gladep
? mail/test-mt
? mail/today-1.diff
? mail/today-2.diff
? mail/today-3.diff
? mail/today-4.diff
? mail/today-5.diff
? mail/today.diff
? mail/typescript
? mail/week.diff
? mail/default/zh_CN/Makefile
? mail/default/zh_CN/Makefile.in
? mail/importers/GNOME_Evolution_Mail_Elm_Intelligent_Importer.oaf
? mail/importers/GNOME_Evolution_Mail_Elm_Intelligent_Importer.oaf.in
? mail/importers/GNOME_Evolution_Mail_Mbox_Importer.oaf
? mail/importers/GNOME_Evolution_Mail_Netscape_Intelligent_Importer.oaf
? mail/importers/GNOME_Evolution_Mail_Netscape_Intelligent_Importer.oaf.in
? mail/importers/GNOME_Evolution_Mail_Outlook_Importer.oaf
? mail/importers/GNOME_Evolution_Mail_Pine_Intelligent_Importer.oaf
? mail/importers/GNOME_Evolution_Mail_Pine_Intelligent_Importer.oaf.in
? mail/importers/am.diff
? mail/importers/b
? mail/importers/elm-importer.c.new
? mail/importers/elm.ps
? mail/importers/im.diff
Index: mail/ChangeLog
===================================================================
RCS file: /cvs/gnome/evolution/mail/ChangeLog,v
retrieving revision 1.3529
diff -u -p -r1.3529 ChangeLog
--- mail/ChangeLog 12 Jan 2005 13:35:25 -0000 1.3529
+++ mail/ChangeLog 17 Jan 2005 05:21:56 -0000
@@ -1,3 +1,16 @@
+2005-01-14 Not Zed <NotZed Ximian com>
+
+ ** Related to changes for #65178.
+
+ * mail-vfolder.c (mv_find_folder): changed from my_list_find, take
+ store arg and use camel_Store_Folder_uri_equal() instead of wrong
+ uri_cmp function.
+ (mail_vfolder_delete_uri): same.
+ (mail_vfolder_rename_uri): same.
+ (uri_is_ignore): take store arg, and same as above. use the right
+ uri for checking sent folder.
+ (mail_vfolder_*): convert to storing uri's in camel form.
+
2005-01-12 Not Zed <NotZed Ximian com>
** See bug #70990.
Index: mail/mail-vfolder.c
===================================================================
RCS file: /cvs/gnome/evolution/mail/mail-vfolder.c,v
retrieving revision 1.127
diff -u -p -r1.127 mail-vfolder.c
--- mail/mail-vfolder.c 12 Nov 2004 05:54:07 -0000 1.127
+++ mail/mail-vfolder.c 17 Jan 2005 05:21:56 -0000
@@ -321,12 +321,11 @@ vfolder_adduri(const char *uri, GList *f
/* ********************************************************************** */
-/* So, uh, apparently g_list_find_custom expect the compare func to return 0 to mean true? */
static GList *
-my_list_find(GList *l, const char *uri, GCompareFunc cmp)
+mv_find_folder(GList *l, CamelStore *store, const char *uri)
{
while (l) {
- if (cmp(l->data, uri))
+ if (camel_store_folder_uri_equal(store, l->data, uri))
break;
l = l->next;
}
@@ -335,7 +334,7 @@ my_list_find(GList *l, const char *uri,
/* uri is a camel uri */
static int
-uri_is_ignore(const char *uri, GCompareFunc uri_cmp)
+uri_is_ignore(CamelStore *store, const char *uri)
{
EAccountList *accounts;
EAccount *account;
@@ -347,9 +346,9 @@ uri_is_ignore(const char *uri, GCompareF
mail_component_get_folder_uri(NULL, MAIL_COMPONENT_FOLDER_SENT),
mail_component_get_folder_uri(NULL, MAIL_COMPONENT_FOLDER_DRAFTS)));
- found = uri_cmp(mail_component_get_folder_uri(NULL, MAIL_COMPONENT_FOLDER_OUTBOX), uri)
- || uri_cmp(mail_component_get_folder_uri(NULL, MAIL_COMPONENT_FOLDER_SENT), uri)
- || uri_cmp(mail_component_get_folder_uri(NULL, MAIL_COMPONENT_FOLDER_DRAFTS), uri);
+ found = camel_store_folder_uri_equal(store, mail_component_get_folder_uri(NULL, MAIL_COMPONENT_FOLDER_OUTBOX), uri)
+ || camel_store_folder_uri_equal(store, mail_component_get_folder_uri(NULL, MAIL_COMPONENT_FOLDER_SENT), uri)
+ || camel_store_folder_uri_equal(store, mail_component_get_folder_uri(NULL, MAIL_COMPONENT_FOLDER_DRAFTS), uri);
if (found)
return found;
@@ -357,14 +356,24 @@ uri_is_ignore(const char *uri, GCompareF
accounts = mail_config_get_accounts ();
iter = e_list_get_iterator ((EList *) accounts);
while (e_iterator_is_valid (iter)) {
+ char *curi;
+
account = (EAccount *) e_iterator_get (iter);
-
+
d(printf("checking sent_folder_uri '%s' == '%s'\n",
account->sent_folder_uri ? account->sent_folder_uri : "empty", uri));
-
- found = (account->sent_folder_uri && uri_cmp (account->sent_folder_uri, uri))
- || (account->drafts_folder_uri && uri_cmp (account->drafts_folder_uri, uri));
-
+
+ if (account->sent_folder_uri) {
+ curi = em_uri_to_camel(account->sent_folder_uri);
+ found = camel_store_folder_uri_equal(store, uri, curi);
+ g_free(curi);
+ }
+ if (!found && account->drafts_folder_uri) {
+ curi = em_uri_to_camel(account->drafts_folder_uri);
+ found = camel_store_folder_uri_equal(store, uri, curi);
+ g_free(curi);
+ }
+
if (found)
break;
@@ -420,7 +429,6 @@ mail_vfolder_add_uri(CamelStore *store,
CamelVeeFolder *vf;
GList *folders = NULL, *link;
int remote = (((CamelService *)store)->provider->flags & CAMEL_PROVIDER_IS_REMOTE) != 0;
- GCompareFunc uri_cmp = CAMEL_STORE_CLASS(CAMEL_OBJECT_GET_CLASS(store))->compare_folder_name;
int is_ignore;
char *uri;
@@ -432,7 +440,7 @@ mail_vfolder_add_uri(CamelStore *store,
g_assert(pthread_self() == mail_gui_thread);
- is_ignore = uri_is_ignore(curi, uri_cmp);
+ is_ignore = uri_is_ignore(store, curi);
LOCK();
@@ -443,12 +451,12 @@ mail_vfolder_add_uri(CamelStore *store,
is_ignore = TRUE;
} else if (remove) {
if (remote) {
- if ((link = my_list_find(source_folders_remote, (void *)uri, uri_cmp)) != NULL) {
+ if ((link = mv_find_folder(source_folders_remote, store, curi)) != NULL) {
g_free(link->data);
source_folders_remote = g_list_remove_link(source_folders_remote, link);
}
} else {
- if ((link = my_list_find(source_folders_local, (void *)uri, uri_cmp)) != NULL) {
+ if ((link = mv_find_folder(source_folders_local, store, curi)) != NULL) {
g_free(link->data);
source_folders_local = g_list_remove_link(source_folders_local, link);
}
@@ -456,11 +464,11 @@ mail_vfolder_add_uri(CamelStore *store,
} else if (!is_ignore) {
/* we ignore drafts/sent/outbox here */
if (remote) {
- if (my_list_find(source_folders_remote, (void *)uri, uri_cmp) == NULL)
- source_folders_remote = g_list_prepend(source_folders_remote, g_strdup(uri));
+ if (mv_find_folder(source_folders_remote, store, curi) == NULL)
+ source_folders_remote = g_list_prepend(source_folders_remote, g_strdup(curi));
} else {
- if (my_list_find(source_folders_local, (void *)uri, uri_cmp) == NULL)
- source_folders_local = g_list_prepend(source_folders_local, g_strdup(uri));
+ if (mv_find_folder(source_folders_local, store, curi) == NULL)
+ source_folders_local = g_list_prepend(source_folders_local, g_strdup(curi));
}
}
@@ -481,15 +489,14 @@ mail_vfolder_add_uri(CamelStore *store,
|| (((EMVFolderRule *)rule)->with == EM_VFOLDER_RULE_WITH_LOCAL_REMOTE_ACTIVE)))
found = TRUE;
- /* we check using the store uri_cmp since its more accurate */
source = NULL;
while (!found && (source = em_vfolder_rule_next_source((EMVFolderRule *)rule, source))) {
- char *esource;
+ char *csource;
- esource = em_uri_from_camel(source);
- found = uri_cmp(uri, esource);
- d(printf(found?" '%s' == '%s'?\n":" '%s' != '%s'\n", uri, esource));
- g_free(esource);
+ csource = em_uri_to_camel(source);
+ found = camel_store_folder_uri_equal(store, curi, csource);
+ d(printf(found?" '%s' == '%s'?\n":" '%s' != '%s'\n", curi, csource));
+ g_free(csource);
}
if (found) {
@@ -512,7 +519,6 @@ mail_vfolder_add_uri(CamelStore *store,
void
mail_vfolder_delete_uri(CamelStore *store, const char *curi)
{
- GCompareFunc uri_cmp = CAMEL_STORE_CLASS(CAMEL_OBJECT_GET_CLASS(store))->compare_folder_name;
FilterRule *rule;
const char *source;
CamelVeeFolder *vf;
@@ -538,9 +544,11 @@ mail_vfolder_delete_uri(CamelStore *stor
while ((rule = rule_context_next_rule ((RuleContext *) context, rule, NULL))) {
source = NULL;
while ((source = em_vfolder_rule_next_source ((EMVFolderRule *) rule, source))) {
+ char *csource = em_uri_to_camel(source);
+
/* Remove all sources that match, ignore changed events though
because the adduri call above does the work async */
- if (uri_cmp (uri, source)) {
+ if (camel_store_folder_uri_equal(store, curi, csource)) {
vf = g_hash_table_lookup (vfolder_hash, rule->name);
g_assert (vf != NULL);
g_signal_handlers_disconnect_matched (rule, G_SIGNAL_MATCH_FUNC|G_SIGNAL_MATCH_DATA, 0,
@@ -550,15 +558,16 @@ mail_vfolder_delete_uri(CamelStore *stor
g_string_append_printf (changed, " %s\n", rule->name);
source = NULL;
}
+ g_free(csource);
}
}
- if ((link = my_list_find(source_folders_remote, (void *)uri, uri_cmp)) != NULL) {
+ if ((link = mv_find_folder(source_folders_remote, store, curi)) != NULL) {
g_free(link->data);
source_folders_remote = g_list_remove_link(source_folders_remote, link);
}
- if ((link = my_list_find(source_folders_local, (void *)uri, uri_cmp)) != NULL) {
+ if ((link = mv_find_folder(source_folders_local, store, curi)) != NULL) {
g_free(link->data);
source_folders_local = g_list_remove_link(source_folders_local, link);
}
@@ -588,7 +597,6 @@ mail_vfolder_delete_uri(CamelStore *stor
void
mail_vfolder_rename_uri(CamelStore *store, const char *cfrom, const char *cto)
{
- GCompareFunc uri_cmp = CAMEL_STORE_CLASS(CAMEL_OBJECT_GET_CLASS(store))->compare_folder_name;
FilterRule *rule;
const char *source;
CamelVeeFolder *vf;
@@ -612,9 +620,11 @@ mail_vfolder_rename_uri(CamelStore *stor
while ( (rule = rule_context_next_rule((RuleContext *)context, rule, NULL)) ) {
source = NULL;
while ( (source = em_vfolder_rule_next_source((EMVFolderRule *)rule, source)) ) {
+ char *csource = em_uri_to_camel(source);
+
/* Remove all sources that match, ignore changed events though
because the adduri call above does the work async */
- if (uri_cmp(from, source)) {
+ if (camel_store_folder_uri_equal(store, cfrom, csource)) {
d(printf("Vfolder '%s' used '%s' ('%s') now uses '%s'\n", rule->name, source, from, to));
vf = g_hash_table_lookup(vfolder_hash, rule->name);
g_assert(vf);
@@ -626,6 +636,7 @@ mail_vfolder_rename_uri(CamelStore *stor
changed++;
source = NULL;
}
+ g_free(csource);
}
}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]