Remove the CONF_DEFAULT_PATH entry, as this is handled by the url config
2002-06-04 Not Zed <NotZed@Ximian.com> * providers/local/camel-local-provider.c: Remove the CONF_DEFAULT_PATH entry, as this is handled by the url config stuff. This made it basically impossible to configure any local folder from the gui. * providers/local/camel-local-folder.c (local_refresh_info): Force a refresh. Should work for all local folders to force them to refresh. * providers/local/camel-maildir-folder.c (maildir_refresh_info): Removed, moved into camel-local-folder. * providers/local/camel-mh-summary.c (mh_summary_next_uid_string): Once we assign or get a uid, tell the summary of this, so the next uid we get will be higher than any previously. * camel-object.c (camel_object_ref, camel_object_unref): I got sick of casting, these now take void * like they should, and perform their own run-time type checking. * providers/local/camel-mh-store.c (recursive_scan): Changed to use stat, not lstat (*shrug* maybe someone wants to softlink their maildir tree?). (inode_hash): (inode_equal): (inode_free): Copied from camel-maildir store (should put into camel-local-store or utils?). (recursive_scan): Changed to check for re-visiting inodes. Also, it builds the tree itself, rather than using folder_info_build. (add_folder): Changed to folder_info_new. (recursive_scan): Properly honour the recursive flag. Also, lookup unread count from folder. (folder_info_new): Init unread message count to -1, since we dont know yet. (folder_info_new): Take the name as an argument, and perform the merging here. (folders_update): Util func to add/remove folders from .folders file. I'm assuming its sorted. (get_folder): Add the folder to .folders if we created a new one, and if it exists. (delete_folder): Remove from .folders, etc. (folders_scan): If we have a .folders file, read and use that instead. (recursive_scan): Handle scanning from a particular directory properly. (rename_folder): Implement so we can track any changes to the .folders file if its turned on. ** Applied patch below from Greg Hudson. 2002-05-10 Greg Hudson <ghudson@mit.edu> * camel-mh-store.c (get_inbox, get_folder_info, recursive_scan, add_folder): Implement support for MH stores. * camel-mh-summary.c (mh_summary_check, sort_uid_cmp): Sort MH messages by message number (uid), like we sort maildir messages by date. * camel-local-provider.c (mh_provider): Turn on source and store flags. svn path=/trunk/; revision=17130
This commit is contained in:
@ -1,3 +1,67 @@
|
|||||||
|
2002-06-04 Not Zed <NotZed@Ximian.com>
|
||||||
|
|
||||||
|
* providers/local/camel-local-provider.c: Remove the
|
||||||
|
CONF_DEFAULT_PATH entry, as this is handled by the url config
|
||||||
|
stuff. This made it basically impossible to configure any local
|
||||||
|
folder from the gui.
|
||||||
|
|
||||||
|
* providers/local/camel-local-folder.c (local_refresh_info): Force
|
||||||
|
a refresh. Should work for all local folders to force them to
|
||||||
|
refresh.
|
||||||
|
|
||||||
|
* providers/local/camel-maildir-folder.c (maildir_refresh_info):
|
||||||
|
Removed, moved into camel-local-folder.
|
||||||
|
|
||||||
|
* providers/local/camel-mh-summary.c (mh_summary_next_uid_string):
|
||||||
|
Once we assign or get a uid, tell the summary of this, so the next
|
||||||
|
uid we get will be higher than any previously.
|
||||||
|
|
||||||
|
* camel-object.c (camel_object_ref, camel_object_unref): I got
|
||||||
|
sick of casting, these now take void * like they should, and
|
||||||
|
perform their own run-time type checking.
|
||||||
|
|
||||||
|
* providers/local/camel-mh-store.c (recursive_scan): Changed to
|
||||||
|
use stat, not lstat (*shrug* maybe someone wants to softlink their
|
||||||
|
maildir tree?).
|
||||||
|
(inode_hash):
|
||||||
|
(inode_equal):
|
||||||
|
(inode_free): Copied from camel-maildir store (should put into
|
||||||
|
camel-local-store or utils?).
|
||||||
|
(recursive_scan): Changed to check for re-visiting inodes. Also,
|
||||||
|
it builds the tree itself, rather than using folder_info_build.
|
||||||
|
(add_folder): Changed to folder_info_new.
|
||||||
|
(recursive_scan): Properly honour the recursive flag. Also,
|
||||||
|
lookup unread count from folder.
|
||||||
|
(folder_info_new): Init unread message count to -1, since we dont
|
||||||
|
know yet.
|
||||||
|
(folder_info_new): Take the name as an argument, and perform the
|
||||||
|
merging here.
|
||||||
|
(folders_update): Util func to add/remove folders from .folders
|
||||||
|
file. I'm assuming its sorted.
|
||||||
|
(get_folder): Add the folder to .folders if we created a new one,
|
||||||
|
and if it exists.
|
||||||
|
(delete_folder): Remove from .folders, etc.
|
||||||
|
(folders_scan): If we have a .folders file, read and use that
|
||||||
|
instead.
|
||||||
|
(recursive_scan): Handle scanning from a particular directory
|
||||||
|
properly.
|
||||||
|
(rename_folder): Implement so we can track any changes to the
|
||||||
|
.folders file if its turned on.
|
||||||
|
|
||||||
|
** Applied patch below from Greg Hudson.
|
||||||
|
|
||||||
|
2002-05-10 Greg Hudson <ghudson@mit.edu>
|
||||||
|
|
||||||
|
* camel-mh-store.c (get_inbox, get_folder_info, recursive_scan,
|
||||||
|
add_folder): Implement support for MH stores.
|
||||||
|
|
||||||
|
* camel-mh-summary.c (mh_summary_check, sort_uid_cmp): Sort MH
|
||||||
|
messages by message number (uid), like we sort maildir messages by
|
||||||
|
date.
|
||||||
|
|
||||||
|
* camel-local-provider.c (mh_provider): Turn on source and store
|
||||||
|
flags.
|
||||||
|
|
||||||
2002-06-03 Not Zed <NotZed@Ximian.com>
|
2002-06-03 Not Zed <NotZed@Ximian.com>
|
||||||
|
|
||||||
* camel-vee-folder.c (camel_vee_folder_add_folder):
|
* camel-vee-folder.c (camel_vee_folder_add_folder):
|
||||||
|
@ -409,17 +409,26 @@ camel_object_new(CamelType type)
|
|||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
camel_object_ref(CamelObject *o)
|
camel_object_ref(void *vo)
|
||||||
{
|
{
|
||||||
|
register CamelObject *o = vo;
|
||||||
|
|
||||||
|
g_return_if_fail(CAMEL_IS_OBJECT(o));
|
||||||
|
|
||||||
CLASS_LOCK(o->klass);
|
CLASS_LOCK(o->klass);
|
||||||
o->ref_count++;
|
o->ref_count++;
|
||||||
CLASS_UNLOCK(o->klass);
|
CLASS_UNLOCK(o->klass);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
camel_object_unref(CamelObject *o)
|
camel_object_unref(void *vo)
|
||||||
{
|
{
|
||||||
register CamelObjectClass *klass = o->klass, *k;
|
register CamelObject *o = vo;
|
||||||
|
register CamelObjectClass *klass, *k;
|
||||||
|
|
||||||
|
g_return_if_fail(CAMEL_IS_OBJECT(o));
|
||||||
|
|
||||||
|
klass = o->klass;
|
||||||
|
|
||||||
CLASS_LOCK(klass);
|
CLASS_LOCK(klass);
|
||||||
o->ref_count--;
|
o->ref_count--;
|
||||||
|
@ -192,8 +192,8 @@ CamelType camel_object_get_type (void);
|
|||||||
CamelObject *camel_object_new (CamelType type);
|
CamelObject *camel_object_new (CamelType type);
|
||||||
CamelObject *camel_object_new_name (const char *name);
|
CamelObject *camel_object_new_name (const char *name);
|
||||||
|
|
||||||
void camel_object_ref (CamelObject *obj);
|
void camel_object_ref(void *);
|
||||||
void camel_object_unref (CamelObject *obj);
|
void camel_object_unref(void *);
|
||||||
|
|
||||||
#ifdef CAMEL_DEBUG
|
#ifdef CAMEL_DEBUG
|
||||||
#define camel_object_ref(o) (printf("%s (%s:%d):ref (%p)\n", __FUNCTION__, __FILE__, __LINE__, o), camel_object_ref(o))
|
#define camel_object_ref(o) (printf("%s (%s:%d):ref (%p)\n", __FUNCTION__, __FILE__, __LINE__, o), camel_object_ref(o))
|
||||||
|
@ -170,6 +170,7 @@ CamelStream *
|
|||||||
camel_stream_buffer_new (CamelStream *stream, CamelStreamBufferMode mode)
|
camel_stream_buffer_new (CamelStream *stream, CamelStreamBufferMode mode)
|
||||||
{
|
{
|
||||||
CamelStreamBuffer *sbf;
|
CamelStreamBuffer *sbf;
|
||||||
|
|
||||||
sbf = CAMEL_STREAM_BUFFER (camel_object_new (camel_stream_buffer_get_type ()));
|
sbf = CAMEL_STREAM_BUFFER (camel_object_new (camel_stream_buffer_get_type ()));
|
||||||
CAMEL_STREAM_BUFFER_CLASS (CAMEL_OBJECT_GET_CLASS(sbf))->init (sbf, stream, mode);
|
CAMEL_STREAM_BUFFER_CLASS (CAMEL_OBJECT_GET_CLASS(sbf))->init (sbf, stream, mode);
|
||||||
|
|
||||||
|
@ -70,6 +70,8 @@ static int local_getv(CamelObject *object, CamelException *ex, CamelArgGetV *arg
|
|||||||
static int local_lock(CamelLocalFolder *lf, CamelLockType type, CamelException *ex);
|
static int local_lock(CamelLocalFolder *lf, CamelLockType type, CamelException *ex);
|
||||||
static void local_unlock(CamelLocalFolder *lf);
|
static void local_unlock(CamelLocalFolder *lf);
|
||||||
|
|
||||||
|
static void local_refresh_info(CamelFolder *folder, CamelException *ex);
|
||||||
|
|
||||||
static void local_sync(CamelFolder *folder, gboolean expunge, CamelException *ex);
|
static void local_sync(CamelFolder *folder, gboolean expunge, CamelException *ex);
|
||||||
static void local_expunge(CamelFolder *folder, CamelException *ex);
|
static void local_expunge(CamelFolder *folder, CamelException *ex);
|
||||||
|
|
||||||
@ -94,6 +96,7 @@ camel_local_folder_class_init(CamelLocalFolderClass * camel_local_folder_class)
|
|||||||
/* virtual method overload */
|
/* virtual method overload */
|
||||||
oklass->getv = local_getv;
|
oklass->getv = local_getv;
|
||||||
|
|
||||||
|
camel_folder_class->refresh_info = local_refresh_info;
|
||||||
camel_folder_class->sync = local_sync;
|
camel_folder_class->sync = local_sync;
|
||||||
camel_folder_class->expunge = local_expunge;
|
camel_folder_class->expunge = local_expunge;
|
||||||
|
|
||||||
@ -371,6 +374,21 @@ local_unlock(CamelLocalFolder *lf)
|
|||||||
/* nothing */
|
/* nothing */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* for auto-check to work */
|
||||||
|
static void
|
||||||
|
local_refresh_info(CamelFolder *folder, CamelException *ex)
|
||||||
|
{
|
||||||
|
CamelLocalFolder *lf = (CamelLocalFolder *)folder;
|
||||||
|
|
||||||
|
if (camel_local_summary_check((CamelLocalSummary *)folder->summary, lf->changes, ex) == -1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (camel_folder_change_info_changed(lf->changes)) {
|
||||||
|
camel_object_trigger_event((CamelObject *)folder, "folder_changed", lf->changes);
|
||||||
|
camel_folder_change_info_clear(lf->changes);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
local_sync(CamelFolder *folder, gboolean expunge, CamelException *ex)
|
local_sync(CamelFolder *folder, gboolean expunge, CamelException *ex)
|
||||||
{
|
{
|
||||||
|
@ -40,6 +40,8 @@
|
|||||||
|
|
||||||
static CamelProviderConfEntry mh_conf_entries[] = {
|
static CamelProviderConfEntry mh_conf_entries[] = {
|
||||||
CAMEL_PROVIDER_CONF_DEFAULT_PATH,
|
CAMEL_PROVIDER_CONF_DEFAULT_PATH,
|
||||||
|
{ CAMEL_PROVIDER_CONF_CHECKBOX, "dotfolders", NULL,
|
||||||
|
N_("Use the `.folders' folder summary file (exmh)"), "0" },
|
||||||
{ CAMEL_PROVIDER_CONF_END }
|
{ CAMEL_PROVIDER_CONF_END }
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -48,7 +50,7 @@ static CamelProvider mh_provider = {
|
|||||||
N_("MH-format mail directories"),
|
N_("MH-format mail directories"),
|
||||||
N_("For storing local mail in MH-like mail directories."),
|
N_("For storing local mail in MH-like mail directories."),
|
||||||
"mail",
|
"mail",
|
||||||
CAMEL_PROVIDER_IS_LOCAL,
|
CAMEL_PROVIDER_IS_SOURCE | CAMEL_PROVIDER_IS_STORAGE | CAMEL_PROVIDER_IS_LOCAL,
|
||||||
CAMEL_URL_NEED_PATH | CAMEL_URL_PATH_IS_ABSOLUTE,
|
CAMEL_URL_NEED_PATH | CAMEL_URL_PATH_IS_ABSOLUTE,
|
||||||
mh_conf_entries,
|
mh_conf_entries,
|
||||||
/* ... */
|
/* ... */
|
||||||
@ -118,7 +120,7 @@ static CamelProvider spoold_provider = {
|
|||||||
"mail",
|
"mail",
|
||||||
CAMEL_PROVIDER_IS_SOURCE | CAMEL_PROVIDER_IS_STORAGE | CAMEL_PROVIDER_IS_LOCAL,
|
CAMEL_PROVIDER_IS_SOURCE | CAMEL_PROVIDER_IS_STORAGE | CAMEL_PROVIDER_IS_LOCAL,
|
||||||
CAMEL_URL_NEED_PATH | CAMEL_URL_PATH_IS_ABSOLUTE,
|
CAMEL_URL_NEED_PATH | CAMEL_URL_PATH_IS_ABSOLUTE,
|
||||||
spoold_conf_entries,
|
NULL,
|
||||||
/* ... */
|
/* ... */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -54,7 +54,6 @@ static CamelLocalSummary *maildir_create_summary(const char *path, const char *f
|
|||||||
|
|
||||||
static void maildir_append_message(CamelFolder * folder, CamelMimeMessage * message, const CamelMessageInfo *info, char **appended_uid, CamelException * ex);
|
static void maildir_append_message(CamelFolder * folder, CamelMimeMessage * message, const CamelMessageInfo *info, char **appended_uid, CamelException * ex);
|
||||||
static CamelMimeMessage *maildir_get_message(CamelFolder * folder, const gchar * uid, CamelException * ex);
|
static CamelMimeMessage *maildir_get_message(CamelFolder * folder, const gchar * uid, CamelException * ex);
|
||||||
static void maildir_refresh_info(CamelFolder *folder, CamelException *ex);
|
|
||||||
|
|
||||||
static void maildir_finalize(CamelObject * object);
|
static void maildir_finalize(CamelObject * object);
|
||||||
|
|
||||||
@ -68,7 +67,6 @@ static void camel_maildir_folder_class_init(CamelObjectClass * camel_maildir_fol
|
|||||||
/* virtual method definition */
|
/* virtual method definition */
|
||||||
|
|
||||||
/* virtual method overload */
|
/* virtual method overload */
|
||||||
camel_folder_class->refresh_info = maildir_refresh_info;
|
|
||||||
camel_folder_class->append_message = maildir_append_message;
|
camel_folder_class->append_message = maildir_append_message;
|
||||||
camel_folder_class->get_message = maildir_get_message;
|
camel_folder_class->get_message = maildir_get_message;
|
||||||
|
|
||||||
@ -127,20 +125,6 @@ static CamelLocalSummary *maildir_create_summary(const char *path, const char *f
|
|||||||
return (CamelLocalSummary *)camel_maildir_summary_new(path, folder, index);
|
return (CamelLocalSummary *)camel_maildir_summary_new(path, folder, index);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
maildir_refresh_info(CamelFolder *folder, CamelException *ex)
|
|
||||||
{
|
|
||||||
CamelLocalFolder *lf = (CamelLocalFolder *)folder;
|
|
||||||
|
|
||||||
if (camel_local_summary_check((CamelLocalSummary *)folder->summary, lf->changes, ex) == -1)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (camel_folder_change_info_changed(lf->changes)) {
|
|
||||||
camel_object_trigger_event((CamelObject *)folder, "folder_changed", lf->changes);
|
|
||||||
camel_folder_change_info_clear(lf->changes);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
static void
|
||||||
maildir_append_message (CamelFolder *folder, CamelMimeMessage *message, const CamelMessageInfo *info, char **appended_uid, CamelException *ex)
|
maildir_append_message (CamelFolder *folder, CamelMimeMessage *message, const CamelMessageInfo *info, char **appended_uid, CamelException *ex)
|
||||||
{
|
{
|
||||||
|
@ -27,32 +27,48 @@
|
|||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
#include <dirent.h>
|
||||||
|
|
||||||
#include "camel-mh-store.h"
|
#include "camel-mh-store.h"
|
||||||
#include "camel-mh-folder.h"
|
#include "camel-mh-folder.h"
|
||||||
#include "camel-exception.h"
|
#include "camel-exception.h"
|
||||||
#include "camel-url.h"
|
#include "camel-url.h"
|
||||||
|
#include "camel-private.h"
|
||||||
|
|
||||||
|
#include <camel/camel-stream-fs.h>
|
||||||
|
#include <camel/camel-stream-buffer.h>
|
||||||
|
|
||||||
static CamelLocalStoreClass *parent_class = NULL;
|
static CamelLocalStoreClass *parent_class = NULL;
|
||||||
|
|
||||||
|
#define d(x)
|
||||||
|
|
||||||
/* Returns the class for a CamelMhStore */
|
/* Returns the class for a CamelMhStore */
|
||||||
#define CMHS_CLASS(so) CAMEL_MH_STORE_CLASS (CAMEL_OBJECT_GET_CLASS(so))
|
#define CMHS_CLASS(so) CAMEL_MH_STORE_CLASS (CAMEL_OBJECT_GET_CLASS(so))
|
||||||
#define CF_CLASS(so) CAMEL_FOLDER_CLASS (CAMEL_OBJECT_GET_CLASS(so))
|
#define CF_CLASS(so) CAMEL_FOLDER_CLASS (CAMEL_OBJECT_GET_CLASS(so))
|
||||||
#define CMHF_CLASS(so) CAMEL_MH_FOLDER_CLASS (CAMEL_OBJECT_GET_CLASS(so))
|
#define CMHF_CLASS(so) CAMEL_MH_FOLDER_CLASS (CAMEL_OBJECT_GET_CLASS(so))
|
||||||
|
|
||||||
|
static void construct (CamelService *service, CamelSession *session, CamelProvider *provider, CamelURL *url, CamelException *ex);
|
||||||
static CamelFolder *get_folder(CamelStore * store, const char *folder_name, guint32 flags, CamelException * ex);
|
static CamelFolder *get_folder(CamelStore * store, const char *folder_name, guint32 flags, CamelException * ex);
|
||||||
|
static CamelFolder *get_inbox (CamelStore *store, CamelException *ex);
|
||||||
static void delete_folder(CamelStore * store, const char *folder_name, CamelException * ex);
|
static void delete_folder(CamelStore * store, const char *folder_name, CamelException * ex);
|
||||||
|
static void rename_folder(CamelStore *store, const char *old, const char *new, CamelException *ex);
|
||||||
|
static CamelFolderInfo * get_folder_info (CamelStore *store, const char *top, guint32 flags, CamelException *ex);
|
||||||
|
|
||||||
static void camel_mh_store_class_init(CamelObjectClass * camel_mh_store_class)
|
static void camel_mh_store_class_init(CamelObjectClass * camel_mh_store_class)
|
||||||
{
|
{
|
||||||
CamelStoreClass *camel_store_class = CAMEL_STORE_CLASS(camel_mh_store_class);
|
CamelStoreClass *camel_store_class = CAMEL_STORE_CLASS(camel_mh_store_class);
|
||||||
/*CamelServiceClass *camel_service_class = CAMEL_SERVICE_CLASS(camel_mh_store_class);*/
|
CamelServiceClass *camel_service_class = CAMEL_SERVICE_CLASS(camel_mh_store_class);
|
||||||
|
|
||||||
parent_class = (CamelLocalStoreClass *)camel_type_get_global_classfuncs(camel_local_store_get_type());
|
parent_class = (CamelLocalStoreClass *)camel_type_get_global_classfuncs(camel_local_store_get_type());
|
||||||
|
|
||||||
/* virtual method overload, use defaults for most */
|
/* virtual method overload, use defaults for most */
|
||||||
|
camel_service_class->construct = construct;
|
||||||
|
|
||||||
camel_store_class->get_folder = get_folder;
|
camel_store_class->get_folder = get_folder;
|
||||||
|
camel_store_class->get_inbox = get_inbox;
|
||||||
camel_store_class->delete_folder = delete_folder;
|
camel_store_class->delete_folder = delete_folder;
|
||||||
|
camel_store_class->rename_folder = rename_folder;
|
||||||
|
camel_store_class->get_folder_info = get_folder_info;
|
||||||
}
|
}
|
||||||
|
|
||||||
CamelType camel_mh_store_get_type(void)
|
CamelType camel_mh_store_get_type(void)
|
||||||
@ -72,6 +88,103 @@ CamelType camel_mh_store_get_type(void)
|
|||||||
return camel_mh_store_type;
|
return camel_mh_store_type;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
construct (CamelService *service, CamelSession *session, CamelProvider *provider, CamelURL *url, CamelException *ex)
|
||||||
|
{
|
||||||
|
CamelMhStore *mh_store = (CamelMhStore *)service;
|
||||||
|
|
||||||
|
CAMEL_SERVICE_CLASS (parent_class)->construct (service, session, provider, url, ex);
|
||||||
|
if (camel_exception_is_set (ex))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (camel_url_get_param(url, "dotfolders"))
|
||||||
|
mh_store->flags |= CAMEL_MH_DOTFOLDERS;
|
||||||
|
}
|
||||||
|
|
||||||
|
enum {
|
||||||
|
UPDATE_NONE,
|
||||||
|
UPDATE_ADD,
|
||||||
|
UPDATE_REMOVE,
|
||||||
|
};
|
||||||
|
|
||||||
|
/* update the .folders file if it exists, or create it if it doesn't */
|
||||||
|
static void
|
||||||
|
folders_update(const char *root, const char *folder, int mode)
|
||||||
|
{
|
||||||
|
char *tmp, *tmpnew, *line = NULL;
|
||||||
|
CamelStream *stream, *in = NULL, *out = NULL;
|
||||||
|
|
||||||
|
tmpnew = alloca(strlen(root)+16);
|
||||||
|
sprintf(tmpnew, "%s.folders~", root);
|
||||||
|
|
||||||
|
out = camel_stream_fs_new_with_name(tmpnew, O_WRONLY|O_CREAT|O_TRUNC, 0666);
|
||||||
|
if (out == NULL)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
tmp = alloca(strlen(root)+16);
|
||||||
|
sprintf(tmp, "%s.folders", root);
|
||||||
|
stream = camel_stream_fs_new_with_name(tmp, O_RDONLY, 0);
|
||||||
|
if (stream) {
|
||||||
|
in = camel_stream_buffer_new(stream, CAMEL_STREAM_BUFFER_READ);
|
||||||
|
camel_object_unref(stream);
|
||||||
|
}
|
||||||
|
if (in == NULL || stream == NULL) {
|
||||||
|
if (mode == UPDATE_ADD && camel_stream_printf(out, "%s\n", folder) == -1)
|
||||||
|
goto fail;
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
|
||||||
|
while ((line = camel_stream_buffer_read_line((CamelStreamBuffer *)in))) {
|
||||||
|
int copy = TRUE;
|
||||||
|
|
||||||
|
switch (mode) {
|
||||||
|
case UPDATE_REMOVE:
|
||||||
|
if (strcmp(line, folder) == 0)
|
||||||
|
copy = FALSE;
|
||||||
|
break;
|
||||||
|
case UPDATE_ADD: {
|
||||||
|
int cmp = strcmp(line, folder);
|
||||||
|
|
||||||
|
if (cmp > 0) {
|
||||||
|
/* found insertion point */
|
||||||
|
if (camel_stream_printf(out, "%s\n", folder) == -1)
|
||||||
|
goto fail;
|
||||||
|
mode = UPDATE_NONE;
|
||||||
|
} else if (tmp == 0) {
|
||||||
|
/* already there */
|
||||||
|
mode = UPDATE_NONE;
|
||||||
|
}
|
||||||
|
break; }
|
||||||
|
case UPDATE_NONE:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (copy && camel_stream_printf(out, "%s\n", line) == -1)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
g_free(line);
|
||||||
|
line = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add to end? */
|
||||||
|
if (mode == UPDATE_ADD && camel_stream_printf(out, "%s\n", folder) == -1)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
if (camel_stream_close(out) == -1)
|
||||||
|
goto fail;
|
||||||
|
|
||||||
|
done:
|
||||||
|
/* should we care if this fails? I suppose so ... */
|
||||||
|
rename(tmpnew, tmp);
|
||||||
|
fail:
|
||||||
|
unlink(tmpnew); /* remove it if its there */
|
||||||
|
g_free(line);
|
||||||
|
if (in)
|
||||||
|
camel_object_unref(in);
|
||||||
|
if (out)
|
||||||
|
camel_object_unref(out);
|
||||||
|
}
|
||||||
|
|
||||||
static CamelFolder *get_folder(CamelStore * store, const char *folder_name, guint32 flags, CamelException * ex)
|
static CamelFolder *get_folder(CamelStore * store, const char *folder_name, guint32 flags, CamelException * ex)
|
||||||
{
|
{
|
||||||
char *name;
|
char *name;
|
||||||
@ -97,6 +210,7 @@ static CamelFolder *get_folder(CamelStore * store, const char *folder_name, guin
|
|||||||
g_free (name);
|
g_free (name);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mkdir(name, 0700) != 0) {
|
if (mkdir(name, 0700) != 0) {
|
||||||
camel_exception_setv(ex, CAMEL_EXCEPTION_SYSTEM,
|
camel_exception_setv(ex, CAMEL_EXCEPTION_SYSTEM,
|
||||||
_("Could not create folder `%s':\n%s"),
|
_("Could not create folder `%s':\n%s"),
|
||||||
@ -104,6 +218,12 @@ static CamelFolder *get_folder(CamelStore * store, const char *folder_name, guin
|
|||||||
g_free (name);
|
g_free (name);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* add to .folders if we are supposed to */
|
||||||
|
/* FIXME: throw exception on error */
|
||||||
|
if (((CamelMhStore *)store)->flags & CAMEL_MH_DOTFOLDERS)
|
||||||
|
folders_update(((CamelLocalStore *)store)->toplevel_dir, folder_name, UPDATE_ADD);
|
||||||
|
|
||||||
} else if (!S_ISDIR(st.st_mode)) {
|
} else if (!S_ISDIR(st.st_mode)) {
|
||||||
camel_exception_setv(ex, CAMEL_EXCEPTION_STORE_NO_FOLDER,
|
camel_exception_setv(ex, CAMEL_EXCEPTION_STORE_NO_FOLDER,
|
||||||
_("`%s' is not a directory."), name);
|
_("`%s' is not a directory."), name);
|
||||||
@ -115,6 +235,12 @@ static CamelFolder *get_folder(CamelStore * store, const char *folder_name, guin
|
|||||||
return camel_mh_folder_new(store, folder_name, flags, ex);
|
return camel_mh_folder_new(store, folder_name, flags, ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static CamelFolder *
|
||||||
|
get_inbox (CamelStore *store, CamelException *ex)
|
||||||
|
{
|
||||||
|
return get_folder (store, "inbox", 0, ex);
|
||||||
|
}
|
||||||
|
|
||||||
static void delete_folder(CamelStore * store, const char *folder_name, CamelException * ex)
|
static void delete_folder(CamelStore * store, const char *folder_name, CamelException * ex)
|
||||||
{
|
{
|
||||||
char *name;
|
char *name;
|
||||||
@ -130,6 +256,279 @@ static void delete_folder(CamelStore * store, const char *folder_name, CamelExce
|
|||||||
}
|
}
|
||||||
g_free(name);
|
g_free(name);
|
||||||
|
|
||||||
|
/* remove from .folders if we are supposed to */
|
||||||
|
if (((CamelMhStore *)store)->flags & CAMEL_MH_DOTFOLDERS)
|
||||||
|
folders_update(((CamelLocalStore *)store)->toplevel_dir, folder_name, UPDATE_REMOVE);
|
||||||
|
|
||||||
/* and remove metadata */
|
/* and remove metadata */
|
||||||
((CamelStoreClass *)parent_class)->delete_folder(store, folder_name, ex);
|
((CamelStoreClass *)parent_class)->delete_folder(store, folder_name, ex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
rename_folder(CamelStore *store, const char *old, const char *new, CamelException *ex)
|
||||||
|
{
|
||||||
|
CamelException e;
|
||||||
|
|
||||||
|
camel_exception_init(&e);
|
||||||
|
((CamelStoreClass *)parent_class)->rename_folder(store, old, new, &e);
|
||||||
|
if (camel_exception_is_set(&e)) {
|
||||||
|
camel_exception_xfer(ex, &e);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
camel_exception_clear(&e);
|
||||||
|
|
||||||
|
if (((CamelMhStore *)store)->flags & CAMEL_MH_DOTFOLDERS) {
|
||||||
|
/* yeah this is messy, but so is mh! */
|
||||||
|
folders_update(((CamelLocalStore *)store)->toplevel_dir, new, UPDATE_ADD);
|
||||||
|
folders_update(((CamelLocalStore *)store)->toplevel_dir, old, UPDATE_REMOVE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static CamelFolderInfo *folder_info_new(CamelStore *store, const char *root, const char *path)
|
||||||
|
{
|
||||||
|
CamelFolderInfo *fi;
|
||||||
|
char *base;
|
||||||
|
CamelFolder *folder;
|
||||||
|
|
||||||
|
base = strrchr(path, '/');
|
||||||
|
|
||||||
|
/* Build the folder info structure. */
|
||||||
|
fi = g_malloc0(sizeof(*fi));
|
||||||
|
fi->url = g_strdup_printf("mh:%s#%s", root, path);
|
||||||
|
fi->full_name = g_strdup(path);
|
||||||
|
fi->name = g_strdup(base?base+1:path);
|
||||||
|
fi->unread_message_count = 0;
|
||||||
|
|
||||||
|
/* check unread count if open */
|
||||||
|
CAMEL_STORE_LOCK(store, cache_lock);
|
||||||
|
folder = g_hash_table_lookup(store->folders, path);
|
||||||
|
if (folder) {
|
||||||
|
if ((((CamelMhStore *)store)->flags & CAMEL_STORE_FOLDER_INFO_FAST) == 0)
|
||||||
|
camel_folder_refresh_info(folder, NULL);
|
||||||
|
fi->unread_message_count = camel_folder_get_unread_message_count(folder);
|
||||||
|
}
|
||||||
|
CAMEL_STORE_UNLOCK(store, cache_lock);
|
||||||
|
|
||||||
|
/* We could: if we have no folder, and FAST isn't specified, perform a full
|
||||||
|
scan of all messages for their status flags. But its probably not worth
|
||||||
|
it as we need to read the top of every file, i.e. very very slow */
|
||||||
|
|
||||||
|
camel_folder_info_build_path(fi, '/');
|
||||||
|
|
||||||
|
d(printf("New folderinfo:\n '%s'\n '%s'\n '%s'\n", fi->full_name, fi->url, fi->path));
|
||||||
|
|
||||||
|
return fi;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* used to find out where we've visited already */
|
||||||
|
struct _inode {
|
||||||
|
dev_t dnode;
|
||||||
|
ino_t inode;
|
||||||
|
};
|
||||||
|
|
||||||
|
/* Scan path, under root, for directories to add folders for. Both
|
||||||
|
* root and path should have a trailing "/" if they aren't empty. */
|
||||||
|
static void recursive_scan(CamelStore *store, CamelFolderInfo **fip, CamelFolderInfo *parent, GHashTable *visited, const char *root, const char *path)
|
||||||
|
{
|
||||||
|
char *fullpath, *tmp;
|
||||||
|
DIR *dp;
|
||||||
|
struct dirent *d;
|
||||||
|
struct stat st;
|
||||||
|
CamelFolderInfo *fi;
|
||||||
|
struct _inode in, *inew;
|
||||||
|
|
||||||
|
/* Open the specified directory. */
|
||||||
|
if (path[0]) {
|
||||||
|
fullpath = alloca(strlen(root)+strlen(path)+2);
|
||||||
|
sprintf(fullpath, "%s/%s", root, path);
|
||||||
|
} else
|
||||||
|
fullpath = (char *)root;
|
||||||
|
|
||||||
|
if (stat(fullpath, &st) == -1 || !S_ISDIR(st.st_mode))
|
||||||
|
return;
|
||||||
|
|
||||||
|
in.dnode = st.st_dev;
|
||||||
|
in.inode = st.st_ino;
|
||||||
|
|
||||||
|
/* see if we've visited already */
|
||||||
|
if (g_hash_table_lookup(visited, &in) != NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
inew = g_malloc(sizeof(*inew));
|
||||||
|
*inew = in;
|
||||||
|
g_hash_table_insert(visited, inew, inew);
|
||||||
|
|
||||||
|
/* link in ... */
|
||||||
|
fi = folder_info_new(store, root, path);
|
||||||
|
fi->parent = parent;
|
||||||
|
fi->sibling = *fip;
|
||||||
|
*fip = fi;
|
||||||
|
|
||||||
|
if ((( ((CamelMhStore *)store)->flags & CAMEL_STORE_FOLDER_INFO_RECURSIVE) || parent == NULL)) {
|
||||||
|
/* now check content for possible other directories */
|
||||||
|
dp = opendir(fullpath);
|
||||||
|
if (dp == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* Look for subdirectories to add and scan. */
|
||||||
|
while ((d = readdir(dp)) != NULL) {
|
||||||
|
/* Skip current and parent directory. */
|
||||||
|
if (strcmp(d->d_name, ".") == 0
|
||||||
|
|| strcmp(d->d_name, "..") == 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* skip fully-numerical entries (i.e. mh messages) */
|
||||||
|
strtoul(d->d_name, &tmp, 10);
|
||||||
|
if (*tmp == 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* otherwise, treat at potential node, and recurse, a bit more expensive than needed, but tough! */
|
||||||
|
if (path[0]) {
|
||||||
|
tmp = g_strdup_printf("%s/%s", path, d->d_name);
|
||||||
|
recursive_scan(store, &fi->child, fi, visited, root, tmp);
|
||||||
|
g_free(tmp);
|
||||||
|
} else {
|
||||||
|
recursive_scan(store, &fi->child, fi, visited, root, d->d_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
closedir(dp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* scan a .folders file */
|
||||||
|
static void
|
||||||
|
folders_scan(CamelStore *store, const char *root, const char *top, CamelFolderInfo **fip)
|
||||||
|
{
|
||||||
|
CamelFolderInfo *fi;
|
||||||
|
char line[512], *path, *tmp;
|
||||||
|
CamelStream *stream, *in;
|
||||||
|
struct stat st;
|
||||||
|
GPtrArray *folders;
|
||||||
|
GHashTable *visited;
|
||||||
|
int len;
|
||||||
|
|
||||||
|
tmp = alloca(strlen(root)+16);
|
||||||
|
sprintf(tmp, "%s/.folders", root);
|
||||||
|
stream = camel_stream_fs_new_with_name(tmp, 0, O_RDONLY);
|
||||||
|
if (stream == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
in = camel_stream_buffer_new(stream, CAMEL_STREAM_BUFFER_READ);
|
||||||
|
camel_object_unref(stream);
|
||||||
|
if (in == NULL)
|
||||||
|
return;
|
||||||
|
|
||||||
|
visited = g_hash_table_new(g_str_hash, g_str_equal);
|
||||||
|
folders = g_ptr_array_new();
|
||||||
|
|
||||||
|
while ( (len = camel_stream_buffer_gets((CamelStreamBuffer *)in, line, sizeof(line))) > 0) {
|
||||||
|
/* ignore blank lines */
|
||||||
|
if (len <= 1)
|
||||||
|
continue;
|
||||||
|
/* check for invalidly long lines, we abort evreything and fallback */
|
||||||
|
if (line[len-1] != '\n') {
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i=0;i<folders->len;i++)
|
||||||
|
camel_folder_info_free(folders->pdata[i]);
|
||||||
|
g_ptr_array_set_size(folders, 0);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
line[len-1] = 0;
|
||||||
|
|
||||||
|
/* check for \r ? */
|
||||||
|
|
||||||
|
if (top && top[0]) {
|
||||||
|
int toplen = strlen(top);
|
||||||
|
|
||||||
|
/* check is subdir */
|
||||||
|
if (strncmp(top, line, len) != 0)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
/* check is not sub-subdir if not recursive */
|
||||||
|
if (( ((CamelMhStore *)store)->flags & CAMEL_STORE_FOLDER_INFO_RECURSIVE) == 0
|
||||||
|
&& (tmp = strrchr(line, '/'))
|
||||||
|
&& tmp > line+toplen)
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (g_hash_table_lookup(visited, line) != NULL)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
tmp = g_strdup(line);
|
||||||
|
g_hash_table_insert(visited, tmp, tmp);
|
||||||
|
|
||||||
|
path = g_strdup_printf("%s/%s", root, line);
|
||||||
|
if (stat(path, &st) == 0 && S_ISDIR(st.st_mode)) {
|
||||||
|
fi = folder_info_new(store, root, line);
|
||||||
|
g_ptr_array_add(folders, fi);
|
||||||
|
}
|
||||||
|
g_free(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (folders->len)
|
||||||
|
*fip = camel_folder_info_build(folders, NULL, '/', TRUE);
|
||||||
|
g_ptr_array_free(folders, TRUE);
|
||||||
|
|
||||||
|
g_hash_table_foreach(visited, (GHFunc)g_free, NULL);
|
||||||
|
g_hash_table_destroy(visited);
|
||||||
|
|
||||||
|
camel_object_unref(in);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* FIXME: move to camel-local, this is shared with maildir code */
|
||||||
|
static guint inode_hash(const void *d)
|
||||||
|
{
|
||||||
|
const struct _inode *v = d;
|
||||||
|
|
||||||
|
return v->inode ^ v->dnode;
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
static CamelFolderInfo *
|
||||||
|
get_folder_info (CamelStore *store, const char *top, guint32 flags, CamelException *ex)
|
||||||
|
{
|
||||||
|
CamelFolderInfo *fi = NULL;
|
||||||
|
char *root;
|
||||||
|
|
||||||
|
root = ((CamelService *)store)->url->path;
|
||||||
|
|
||||||
|
/* use .folders if we are supposed to */
|
||||||
|
if (((CamelMhStore *)store)->flags & CAMEL_MH_DOTFOLDERS) {
|
||||||
|
folders_scan(store, root, top, &fi);
|
||||||
|
} else {
|
||||||
|
GHashTable *visited = g_hash_table_new(inode_hash, inode_equal);
|
||||||
|
|
||||||
|
if (top == NULL)
|
||||||
|
top = "";
|
||||||
|
|
||||||
|
recursive_scan(store, &fi, NULL, visited, root, top);
|
||||||
|
|
||||||
|
/* if we actually scanned from root, we have a "" root node we dont want */
|
||||||
|
if (fi != NULL && top[0] == 0) {
|
||||||
|
CamelFolderInfo *rfi;
|
||||||
|
|
||||||
|
rfi = fi;
|
||||||
|
fi = rfi->child;
|
||||||
|
rfi->child = NULL;
|
||||||
|
camel_folder_info_free(rfi);
|
||||||
|
}
|
||||||
|
|
||||||
|
g_hash_table_foreach(visited, inode_free, NULL);
|
||||||
|
g_hash_table_destroy(visited);
|
||||||
|
}
|
||||||
|
|
||||||
|
return fi;
|
||||||
|
}
|
||||||
|
@ -34,9 +34,14 @@ extern "C" {
|
|||||||
#define CAMEL_MH_STORE_CLASS(k) (CAMEL_CHECK_CLASS_CAST ((k), CAMEL_MH_STORE_TYPE, CamelMhStoreClass))
|
#define CAMEL_MH_STORE_CLASS(k) (CAMEL_CHECK_CLASS_CAST ((k), CAMEL_MH_STORE_TYPE, CamelMhStoreClass))
|
||||||
#define CAMEL_IS_MH_STORE(o) (CAMEL_CHECK_TYPE((o), CAMEL_MH_STORE_TYPE))
|
#define CAMEL_IS_MH_STORE(o) (CAMEL_CHECK_TYPE((o), CAMEL_MH_STORE_TYPE))
|
||||||
|
|
||||||
|
enum {
|
||||||
|
CAMEL_MH_DOTFOLDERS = (1<<0), /* update/use .folders file */
|
||||||
|
};
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
CamelLocalStore parent_object;
|
CamelLocalStore parent_object;
|
||||||
|
|
||||||
|
guint32 flags;
|
||||||
} CamelMhStore;
|
} CamelMhStore;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
|
@ -39,6 +39,8 @@
|
|||||||
#include "camel-mh-summary.h"
|
#include "camel-mh-summary.h"
|
||||||
#include <camel/camel-mime-message.h>
|
#include <camel/camel-mime-message.h>
|
||||||
|
|
||||||
|
#include "camel-private.h"
|
||||||
|
|
||||||
#define d(x) /*(printf("%s(%d): ", __FILE__, __LINE__),(x))*/
|
#define d(x) /*(printf("%s(%d): ", __FILE__, __LINE__),(x))*/
|
||||||
|
|
||||||
#define CAMEL_MH_SUMMARY_VERSION (0x2000)
|
#define CAMEL_MH_SUMMARY_VERSION (0x2000)
|
||||||
@ -135,11 +137,14 @@ static char *mh_summary_next_uid_string(CamelFolderSummary *s)
|
|||||||
int fd = -1;
|
int fd = -1;
|
||||||
guint32 uid;
|
guint32 uid;
|
||||||
char *name;
|
char *name;
|
||||||
|
char *uidstr;
|
||||||
|
|
||||||
/* if we are working to add an existing file, then use current_uid */
|
/* if we are working to add an existing file, then use current_uid */
|
||||||
if (mhs->priv->current_uid)
|
if (mhs->priv->current_uid) {
|
||||||
return g_strdup(mhs->priv->current_uid);
|
uidstr = g_strdup(mhs->priv->current_uid);
|
||||||
|
/* tell the summary of this, so we always append numbers to the end */
|
||||||
|
camel_folder_summary_set_uid(s, strtoul(uidstr, NULL, 10)+1);
|
||||||
|
} else {
|
||||||
/* else scan for one - and create it too, to make sure */
|
/* else scan for one - and create it too, to make sure */
|
||||||
do {
|
do {
|
||||||
close(fd);
|
close(fd);
|
||||||
@ -152,7 +157,10 @@ static char *mh_summary_next_uid_string(CamelFolderSummary *s)
|
|||||||
|
|
||||||
close(fd);
|
close(fd);
|
||||||
|
|
||||||
return g_strdup_printf("%u", uid);
|
uidstr = g_strdup_printf("%u", uid);
|
||||||
|
}
|
||||||
|
|
||||||
|
return uidstr;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int camel_mh_summary_add(CamelLocalSummary *cls, const char *name, int forceindex)
|
static int camel_mh_summary_add(CamelLocalSummary *cls, const char *name, int forceindex)
|
||||||
@ -198,6 +206,20 @@ remove_summary(char *key, CamelMessageInfo *info, CamelLocalSummary *cls)
|
|||||||
camel_folder_summary_info_free((CamelFolderSummary *)cls, info);
|
camel_folder_summary_info_free((CamelFolderSummary *)cls, info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
sort_uid_cmp(const void *ap, const void *bp)
|
||||||
|
{
|
||||||
|
const CamelMessageInfo
|
||||||
|
*a = *((CamelMessageInfo **)ap),
|
||||||
|
*b = *((CamelMessageInfo **)bp);
|
||||||
|
const char
|
||||||
|
*auid = camel_message_info_uid(a),
|
||||||
|
*buid = camel_message_info_uid(b);
|
||||||
|
int aval = atoi(auid), bval = atoi(buid);
|
||||||
|
|
||||||
|
return (aval < bval) ? -1 : (aval > bval) ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
mh_summary_check(CamelLocalSummary *cls, CamelFolderChangeInfo *changeinfo, CamelException *ex)
|
mh_summary_check(CamelLocalSummary *cls, CamelFolderChangeInfo *changeinfo, CamelException *ex)
|
||||||
{
|
{
|
||||||
@ -205,6 +227,7 @@ mh_summary_check(CamelLocalSummary *cls, CamelFolderChangeInfo *changeinfo, Came
|
|||||||
struct dirent *d;
|
struct dirent *d;
|
||||||
char *p, c;
|
char *p, c;
|
||||||
CamelMessageInfo *info;
|
CamelMessageInfo *info;
|
||||||
|
CamelFolderSummary *s = (CamelFolderSummary *)cls;
|
||||||
GHashTable *left;
|
GHashTable *left;
|
||||||
int i, count;
|
int i, count;
|
||||||
int forceindex;
|
int forceindex;
|
||||||
@ -265,6 +288,11 @@ mh_summary_check(CamelLocalSummary *cls, CamelFolderChangeInfo *changeinfo, Came
|
|||||||
g_hash_table_foreach(left, (GHFunc)remove_summary, cls);
|
g_hash_table_foreach(left, (GHFunc)remove_summary, cls);
|
||||||
g_hash_table_destroy(left);
|
g_hash_table_destroy(left);
|
||||||
|
|
||||||
|
/* sort the summary based on message number (uid), since the directory order is not useful */
|
||||||
|
CAMEL_SUMMARY_LOCK(s, summary_lock);
|
||||||
|
qsort(s->messages->pdata, s->messages->len, sizeof(CamelMessageInfo *), sort_uid_cmp);
|
||||||
|
CAMEL_SUMMARY_UNLOCK(s, summary_lock);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user