aboutsummaryrefslogtreecommitdiffstats
path: root/camel/camel-store-summary.c
diff options
context:
space:
mode:
Diffstat (limited to 'camel/camel-store-summary.c')
-rw-r--r--camel/camel-store-summary.c899
1 files changed, 899 insertions, 0 deletions
diff --git a/camel/camel-store-summary.c b/camel/camel-store-summary.c
new file mode 100644
index 0000000000..8cd2bf37dd
--- /dev/null
+++ b/camel/camel-store-summary.c
@@ -0,0 +1,899 @@
+/*
+ * Copyright (C) 2001 Ximian Inc.
+ *
+ * Authors: Michael Zucchi <notzed@ximian.com>
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of version 2 of the GNU General Public
+ * License as published by the Free Software Foundation.
+ *
+ * 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, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <unistd.h>
+#include <ctype.h>
+#include <string.h>
+#include <errno.h>
+#include <stdlib.h>
+
+#include "camel-store-summary.h"
+
+#include "camel-file-utils.h"
+
+#include "hash-table-utils.h"
+#include "e-util/md5-utils.h"
+#include "e-util/e-memory.h"
+
+#include "camel-private.h"
+
+#define d(x)
+#define io(x) /* io debug */
+
+#define CAMEL_STORE_SUMMARY_VERSION (13)
+
+#define _PRIVATE(o) (((CamelStoreSummary *)(o))->priv)
+
+static int summary_header_load(CamelStoreSummary *, FILE *);
+static int summary_header_save(CamelStoreSummary *, FILE *);
+
+static CamelFolderInfo * folder_info_new(CamelStoreSummary *, const char *);
+static CamelFolderInfo * folder_info_load(CamelStoreSummary *, FILE *);
+static int folder_info_save(CamelStoreSummary *, FILE *, CamelFolderInfo *);
+static void folder_info_free(CamelStoreSummary *, CamelFolderInfo *);
+
+static const char *folder_info_string(CamelStoreSummary *, const CamelFolderInfo *, int);
+static void folder_info_set_string(CamelStoreSummary *, CamelFolderInfo *, int, const char *);
+
+static void camel_store_summary_class_init (CamelStoreSummaryClass *klass);
+static void camel_store_summary_init (CamelStoreSummary *obj);
+static void camel_store_summary_finalise (CamelObject *obj);
+
+static CamelObjectClass *camel_store_summary_parent;
+
+static void
+camel_store_summary_class_init (CamelStoreSummaryClass *klass)
+{
+ camel_store_summary_parent = camel_type_get_global_classfuncs (camel_object_get_type ());
+
+ klass->summary_header_load = summary_header_load;
+ klass->summary_header_save = summary_header_save;
+
+ klass->folder_info_new = folder_info_new;
+ klass->folder_info_load = folder_info_load;
+ klass->folder_info_save = folder_info_save;
+ klass->folder_info_free = folder_info_free;
+
+ klass->folder_info_string = folder_info_string;
+ klass->folder_info_set_string = folder_info_set_string;
+}
+
+static void
+camel_store_summary_init (CamelStoreSummary *s)
+{
+ struct _CamelStoreSummaryPrivate *p;
+
+ p = _PRIVATE(s) = g_malloc0(sizeof(*p));
+
+ s->folder_info_size = sizeof(CamelFolderInfo);
+
+ s->folder_info_chunks = NULL;
+
+ s->version = CAMEL_STORE_SUMMARY_VERSION;
+ s->flags = 0;
+ s->count = 0;
+ s->time = 0;
+
+ s->folders = g_ptr_array_new();
+ s->folders_full = g_hash_table_new(g_str_hash, g_str_equal);
+
+#ifdef ENABLE_THREADS
+ p->summary_lock = g_mutex_new();
+ p->io_lock = g_mutex_new();
+ p->alloc_lock = g_mutex_new();
+ p->ref_lock = g_mutex_new();
+#endif
+}
+
+static void
+camel_store_summary_finalise (CamelObject *obj)
+{
+ struct _CamelStoreSummaryPrivate *p;
+ CamelStoreSummary *s = (CamelStoreSummary *)obj;
+
+ p = _PRIVATE(obj);
+
+ camel_store_summary_clear(s);
+ g_ptr_array_free(s->folders, TRUE);
+ g_hash_table_destroy(s->folders_full);
+
+ g_free(s->summary_path);
+
+ if (s->folder_info_chunks)
+ e_memchunk_destroy(s->folder_info_chunks);
+
+#ifdef ENABLE_THREADS
+ g_mutex_free(p->summary_lock);
+ g_mutex_free(p->io_lock);
+ g_mutex_free(p->alloc_lock);
+ g_mutex_free(p->ref_lock);
+#endif
+
+ g_free(p);
+}
+
+CamelType
+camel_store_summary_get_type (void)
+{
+ static CamelType type = CAMEL_INVALID_TYPE;
+
+ if (type == CAMEL_INVALID_TYPE) {
+ type = camel_type_register (camel_object_get_type (), "CamelStoreSummary",
+ sizeof (CamelStoreSummary),
+ sizeof (CamelStoreSummaryClass),
+ (CamelObjectClassInitFunc) camel_store_summary_class_init,
+ NULL,
+ (CamelObjectInitFunc) camel_store_summary_init,
+ (CamelObjectFinalizeFunc) camel_store_summary_finalise);
+ }
+
+ return type;
+}
+
+/**
+ * camel_store_summary_new:
+ *
+ * Create a new CamelStoreSummary object.
+ *
+ * Return value: A new CamelStoreSummary widget.
+ **/
+CamelStoreSummary *
+camel_store_summary_new (void)
+{
+ CamelStoreSummary *new = CAMEL_STORE_SUMMARY ( camel_object_new (camel_store_summary_get_type ())); return new;
+}
+
+/**
+ * camel_store_summary_set_filename:
+ * @s:
+ * @name:
+ *
+ * Set the filename where the summary will be loaded to/saved from.
+ **/
+void camel_store_summary_set_filename(CamelStoreSummary *s, const char *name)
+{
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+
+ g_free(s->summary_path);
+ s->summary_path = g_strdup(name);
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+}
+
+void camel_store_summary_set_uri_prefix(CamelStoreSummary *s, const char *prefix)
+{
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+
+ g_free(s->uri_prefix);
+ s->uri_prefix = g_strdup(prefix);
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+}
+
+/**
+ * camel_store_summary_count:
+ * @s:
+ *
+ * Get the number of summary items stored in this summary.
+ *
+ * Return value: The number of items int he summary.
+ **/
+int
+camel_store_summary_count(CamelStoreSummary *s)
+{
+ return s->folders->len;
+}
+
+/**
+ * camel_store_summary_index:
+ * @s:
+ * @i:
+ *
+ * Retrieve a summary item by index number.
+ *
+ * A referenced to the summary item is returned, which may be
+ * ref'd or free'd as appropriate.
+ *
+ * Return value: The summary item, or NULL if the index @i is out
+ * of range.
+ * It must be freed using camel_store_summary_info_free().
+ **/
+CamelFolderInfo *
+camel_store_summary_index(CamelStoreSummary *s, int i)
+{
+ CamelFolderInfo *info = NULL;
+
+ CAMEL_STORE_SUMMARY_LOCK(s, ref_lock);
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+
+ if (i<s->folders->len)
+ info = g_ptr_array_index(s->folders, i);
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+
+ if (info)
+ info->refcount++;
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, ref_lock);
+
+ return info;
+}
+
+/**
+ * camel_store_summary_index:
+ * @s:
+ * @i:
+ *
+ * Obtain a copy of the summary array. This is done atomically,
+ * so cannot contain empty entries.
+ *
+ * It must be freed using camel_store_summary_array_free().
+ **/
+GPtrArray *
+camel_store_summary_array(CamelStoreSummary *s)
+{
+ CamelFolderInfo *info;
+ GPtrArray *res = g_ptr_array_new();
+ int i;
+
+ CAMEL_STORE_SUMMARY_LOCK(s, ref_lock);
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+
+ g_ptr_array_set_size(res, s->folders->len);
+ for (i=0;i<s->folders->len;i++) {
+ info = res->pdata[i] = g_ptr_array_index(s->folders, i);
+ info->refcount++;
+ }
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+ CAMEL_STORE_SUMMARY_UNLOCK(s, ref_lock);
+
+ return res;
+}
+
+/**
+ * camel_store_summary_array_free:
+ * @s:
+ * @array:
+ *
+ * Free the folder summary array.
+ **/
+void
+camel_store_summary_array_free(CamelStoreSummary *s, GPtrArray *array)
+{
+ int i;
+
+ for (i=0;i<array->len;i++)
+ camel_store_summary_info_free(s, array->pdata[i]);
+
+ g_ptr_array_free(array, TRUE);
+}
+
+/**
+ * camel_store_summary_full:
+ * @s:
+ * @full:
+ *
+ * Retrieve a summary item by full name.
+ *
+ * A referenced to the summary item is returned, which may be
+ * ref'd or free'd as appropriate.
+ *
+ * Return value: The summary item, or NULL if the @full name
+ * is not available.
+ * It must be freed using camel_store_summary_info_free().
+ **/
+CamelFolderInfo *
+camel_store_summary_full(CamelStoreSummary *s, const char *full)
+{
+ CamelFolderInfo *info;
+
+ CAMEL_STORE_SUMMARY_LOCK(s, ref_lock);
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+
+ info = g_hash_table_lookup(s->folders_full, full);
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+
+ if (info)
+ info->refcount++;
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, ref_lock);
+
+ return info;
+}
+
+int
+camel_store_summary_load(CamelStoreSummary *s)
+{
+ FILE *in;
+ int i;
+ CamelFolderInfo *mi;
+
+ g_assert(s->summary_path);
+
+ in = fopen(s->summary_path, "r");
+ if (in == NULL)
+ return -1;
+
+ CAMEL_STORE_SUMMARY_LOCK(s, io_lock);
+ if ( ((CamelStoreSummaryClass *)(CAMEL_OBJECT_GET_CLASS(s)))->summary_header_load(s, in) == -1)
+ goto error;
+
+ /* now read in each message ... */
+ for (i=0;i<s->count;i++) {
+ mi = ((CamelStoreSummaryClass *)(CAMEL_OBJECT_GET_CLASS(s)))->folder_info_load(s, in);
+
+ if (mi == NULL)
+ goto error;
+
+ camel_store_summary_add(s, mi);
+ }
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, io_lock);
+
+ if (fclose(in) == -1)
+ return -1;
+
+ s->flags &= ~CAMEL_STORE_SUMMARY_DIRTY;
+
+ return 0;
+
+error:
+ g_warning("Cannot load summary file: %s", strerror(ferror(in)));
+ CAMEL_STORE_SUMMARY_UNLOCK(s, io_lock);
+ fclose(in);
+ s->flags |= ~CAMEL_STORE_SUMMARY_DIRTY;
+
+ return -1;
+}
+
+/**
+ * camel_store_summary_save:
+ * @s:
+ *
+ * Writes the summary to disk. The summary is only written if changes
+ * have occured.
+ *
+ * Return value: Returns -1 on error.
+ **/
+int
+camel_store_summary_save(CamelStoreSummary *s)
+{
+ FILE *out;
+ int fd;
+ int i;
+ guint32 count;
+ CamelFolderInfo *mi;
+
+ g_assert(s->summary_path);
+
+ if ((s->flags & CAMEL_STORE_SUMMARY_DIRTY) == 0)
+ return 0;
+
+ fd = open(s->summary_path, O_RDWR|O_CREAT, 0600);
+ if (fd == -1)
+ return -1;
+ out = fdopen(fd, "w");
+ if ( out == NULL ) {
+ close(fd);
+ return -1;
+ }
+
+ io(printf("saving header\n"));
+
+ CAMEL_STORE_SUMMARY_LOCK(s, io_lock);
+
+ if ( ((CamelStoreSummaryClass *)(CAMEL_OBJECT_GET_CLASS(s)))->summary_header_save(s, out) == -1) {
+ fclose(out);
+ CAMEL_STORE_SUMMARY_UNLOCK(s, io_lock);
+ return -1;
+ }
+
+ /* now write out each message ... */
+
+ /* FIXME: Locking? */
+
+ count = s->folders->len;
+ for (i=0;i<count;i++) {
+ mi = s->folders->pdata[i];
+ ((CamelStoreSummaryClass *)(CAMEL_OBJECT_GET_CLASS(s)))->folder_info_save(s, out, mi);
+ }
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, io_lock);
+
+ if (fclose(out) == -1)
+ return -1;
+
+ s->flags &= ~CAMEL_STORE_SUMMARY_DIRTY;
+ return 0;
+}
+
+/**
+ * camel_store_summary_header_load:
+ * @s: Summary object.
+ *
+ * Only load the header information from the summary,
+ * keep the rest on disk. This should only be done on
+ * a fresh summary object.
+ *
+ * Return value: -1 on error.
+ **/
+int camel_store_summary_header_load(CamelStoreSummary *s)
+{
+ FILE *in;
+ int ret;
+
+ g_assert(s->summary_path);
+
+ in = fopen(s->summary_path, "r");
+ if (in == NULL)
+ return -1;
+
+ CAMEL_STORE_SUMMARY_LOCK(s, io_lock);
+ ret = ((CamelStoreSummaryClass *)(CAMEL_OBJECT_GET_CLASS(s)))->summary_header_load(s, in);
+ CAMEL_STORE_SUMMARY_UNLOCK(s, io_lock);
+
+ fclose(in);
+ s->flags &= ~CAMEL_STORE_SUMMARY_DIRTY;
+ return ret;
+}
+
+/**
+ * camel_store_summary_add:
+ * @s:
+ * @info:
+ *
+ * Adds a new @info record to the summary. If @info->uid is NULL, then a new
+ * uid is automatically re-assigned by calling :next_uid_string().
+ *
+ * The @info record should have been generated by calling one of the
+ * info_new_*() functions, as it will be free'd based on the summary
+ * class. And MUST NOT be allocated directly using malloc.
+ **/
+void camel_store_summary_add(CamelStoreSummary *s, CamelFolderInfo *info)
+{
+ if (info == NULL)
+ return;
+
+ if (camel_folder_info_full(s, info) == NULL) {
+ g_warning("Trying to add a folder info with missing required full name\n");
+ return;
+ }
+
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+
+ g_ptr_array_add(s->folders, info);
+ g_hash_table_insert(s->folders_full, (char *)camel_folder_info_full(s, info), info);
+ s->flags |= CAMEL_STORE_SUMMARY_DIRTY;
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+}
+
+/**
+ * camel_store_summary_add_from_full:
+ * @s:
+ * @h:
+ *
+ * Build a new info record based on the name, and add it to the summary.
+ *
+ * Return value: The newly added record.
+ **/
+CamelFolderInfo *camel_store_summary_add_from_full(CamelStoreSummary *s, const char *full)
+{
+ CamelFolderInfo *info;
+
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+
+ info = g_hash_table_lookup(s->folders_full, full);
+ if (info != NULL) {
+ g_warning("Trying to add folder '%s' to summary that already has it", full);
+ info = NULL;
+ } else {
+ info = camel_store_summary_info_new_from_full(s, full);
+ g_ptr_array_add(s->folders, info);
+ g_hash_table_insert(s->folders_full, (char *)camel_folder_info_full(s, info), info);
+ s->flags |= CAMEL_STORE_SUMMARY_DIRTY;
+ }
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+
+ return info;
+}
+
+/**
+ * camel_store_summary_info_new_from_full:
+ * @s:
+ * @h:
+ *
+ * Create a new info record from a name.
+ *
+ * Return value: Guess? This info record MUST be freed using
+ * camel_store_summary_info_free(), camel_folder_info_free() will not work.
+ **/
+CamelFolderInfo *camel_store_summary_info_new_from_full(CamelStoreSummary *s, const char *f)
+{
+ return ((CamelStoreSummaryClass *)(CAMEL_OBJECT_GET_CLASS(s))) -> folder_info_new(s, f);
+}
+
+/**
+ * camel_store_summary_info_free:
+ * @s:
+ * @mi:
+ *
+ * Unref and potentially free the message info @mi, and all associated memory.
+ **/
+void camel_store_summary_info_free(CamelStoreSummary *s, CamelFolderInfo *mi)
+{
+ g_assert(mi);
+ g_assert(s);
+
+ CAMEL_STORE_SUMMARY_LOCK(s, ref_lock);
+
+ g_assert(mi->refcount >= 1);
+
+ mi->refcount--;
+ if (mi->refcount > 0) {
+ CAMEL_STORE_SUMMARY_UNLOCK(s, ref_lock);
+ return;
+ }
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, ref_lock);
+
+ ((CamelStoreSummaryClass *)(CAMEL_OBJECT_GET_CLASS(s)))->folder_info_free(s, mi);
+}
+
+/**
+ * camel_store_summary_info_ref:
+ * @s:
+ * @mi:
+ *
+ * Add an extra reference to @mi.
+ **/
+void camel_store_summary_info_ref(CamelStoreSummary *s, CamelFolderInfo *mi)
+{
+ g_assert(mi);
+ g_assert(s);
+
+ CAMEL_STORE_SUMMARY_LOCK(s, ref_lock);
+ g_assert(mi->refcount >= 1);
+ mi->refcount++;
+ CAMEL_STORE_SUMMARY_UNLOCK(s, ref_lock);
+}
+
+/**
+ * camel_store_summary_touch:
+ * @s:
+ *
+ * Mark the summary as changed, so that a save will save it.
+ **/
+void
+camel_store_summary_touch(CamelStoreSummary *s)
+{
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+ s->flags |= CAMEL_STORE_SUMMARY_DIRTY;
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+}
+
+/**
+ * camel_store_summary_clear:
+ * @s:
+ *
+ * Empty the summary contents.
+ **/
+void
+camel_store_summary_clear(CamelStoreSummary *s)
+{
+ int i;
+
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+ if (camel_store_summary_count(s) == 0) {
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+ return;
+ }
+
+ for (i=0;i<s->folders->len;i++)
+ camel_store_summary_info_free(s, s->folders->pdata[i]);
+
+ g_ptr_array_set_size(s->folders, 0);
+ g_hash_table_destroy(s->folders_full);
+ s->folders_full = g_hash_table_new(g_str_hash, g_str_equal);
+ s->flags |= CAMEL_STORE_SUMMARY_DIRTY;
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+}
+
+/**
+ * camel_store_summary_remove:
+ * @s:
+ * @info:
+ *
+ * Remove a specific @info record from the summary.
+ **/
+void camel_store_summary_remove(CamelStoreSummary *s, CamelFolderInfo *info)
+{
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+ g_hash_table_remove(s->folders_full, camel_folder_info_full(s, info));
+ g_ptr_array_remove(s->folders, info);
+ s->flags |= CAMEL_STORE_SUMMARY_DIRTY;
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+
+ camel_store_summary_info_free(s, info);
+}
+
+/**
+ * camel_store_summary_remove_uid:
+ * @s:
+ * @full:
+ *
+ * Remove a specific info record from the summary, by @full.
+ **/
+void camel_store_summary_remove_full(CamelStoreSummary *s, const char *full)
+{
+ CamelFolderInfo *oldinfo;
+ char *oldfull;
+
+ CAMEL_STORE_SUMMARY_LOCK(s, ref_lock);
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+ if (g_hash_table_lookup_extended(s->folders_full, full, (void *)&oldfull, (void *)&oldinfo)) {
+ /* make sure it doesn't vanish while we're removing it */
+ oldinfo->refcount++;
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+ CAMEL_STORE_SUMMARY_UNLOCK(s, ref_lock);
+ camel_store_summary_remove(s, oldinfo);
+ camel_store_summary_info_free(s, oldinfo);
+ } else {
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+ CAMEL_STORE_SUMMARY_UNLOCK(s, ref_lock);
+ }
+}
+
+/**
+ * camel_store_summary_remove_index:
+ * @s:
+ * @index:
+ *
+ * Remove a specific info record from the summary, by index.
+ **/
+void camel_store_summary_remove_index(CamelStoreSummary *s, int index)
+{
+ CAMEL_STORE_SUMMARY_LOCK(s, summary_lock);
+ if (index < s->folders->len) {
+ CamelFolderInfo *info = s->folders->pdata[index];
+
+ g_hash_table_remove(s->folders_full, camel_folder_info_full(s, info));
+ g_ptr_array_remove_index(s->folders, index);
+ s->flags |= CAMEL_STORE_SUMMARY_DIRTY;
+
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+ camel_store_summary_info_free(s, info);
+ } else {
+ CAMEL_STORE_SUMMARY_UNLOCK(s, summary_lock);
+ }
+}
+
+static int
+summary_header_load(CamelStoreSummary *s, FILE *in)
+{
+ gint32 version, flags, count;
+ time_t time;
+
+ fseek(in, 0, SEEK_SET);
+
+ io(printf("Loading header\n"));
+
+ if (camel_file_util_decode_fixed_int32(in, &version) == -1
+ || camel_file_util_decode_fixed_int32(in, &flags) == -1
+ || camel_file_util_decode_time_t(in, &time) == -1
+ || camel_file_util_decode_fixed_int32(in, &count) == -1) {
+ return -1;
+ }
+
+ s->flags = flags;
+ s->time = time;
+ s->count = count;
+ if (s->version != version) {
+ g_warning("Summary header version mismatch");
+ return -1;
+ }
+ return 0;
+}
+
+static int
+summary_header_save(CamelStoreSummary *s, FILE *out)
+{
+ fseek(out, 0, SEEK_SET);
+
+ io(printf("Savining header\n"));
+
+ camel_file_util_encode_fixed_int32(out, s->version);
+ camel_file_util_encode_fixed_int32(out, s->flags);
+ camel_file_util_encode_time_t(out, s->time);
+ return camel_file_util_encode_fixed_int32(out, camel_store_summary_count(s));
+}
+
+/**
+ * camel_store_summary_info_new:
+ * @s:
+ *
+ * Allocate a new camel message info, suitable for adding
+ * to this summary.
+ *
+ * Return value:
+ **/
+CamelFolderInfo *
+camel_store_summary_info_new(CamelStoreSummary *s)
+{
+ CamelFolderInfo *mi;
+
+ CAMEL_STORE_SUMMARY_LOCK(s, alloc_lock);
+ if (s->folder_info_chunks == NULL)
+ s->folder_info_chunks = e_memchunk_new(32, s->folder_info_size);
+ mi = e_memchunk_alloc0(s->folder_info_chunks);
+ CAMEL_STORE_SUMMARY_UNLOCK(s, alloc_lock);
+ mi->refcount = 1;
+ return mi;
+}
+
+const char *camel_folder_info_string(CamelStoreSummary *s, const CamelFolderInfo *mi, int type)
+{
+ return ((CamelStoreSummaryClass *)(CAMEL_OBJECT_GET_CLASS(s)))->folder_info_string(s, mi, type);
+}
+
+void camel_folder_info_set_string(CamelStoreSummary *s, CamelFolderInfo *mi, int type, const char *value)
+{
+ return ((CamelStoreSummaryClass *)(CAMEL_OBJECT_GET_CLASS(s)))->folder_info_set_string(s, mi, type, value);
+}
+
+static CamelFolderInfo *
+folder_info_new(CamelStoreSummary *s, const char *f)
+{
+ CamelFolderInfo *mi;
+
+ mi = camel_store_summary_info_new(s);
+
+ mi->full = g_strdup(f);
+ mi->unread = CAMEL_STORE_SUMMARY_UNKNOWN;
+ mi->total = CAMEL_STORE_SUMMARY_UNKNOWN;
+
+ return mi;
+}
+
+static CamelFolderInfo *
+folder_info_load(CamelStoreSummary *s, FILE *in)
+{
+ CamelFolderInfo *mi;
+
+ mi = camel_store_summary_info_new(s);
+
+ io(printf("Loading folder info\n"));
+
+ camel_file_util_decode_string(in, &mi->full);
+ camel_file_util_decode_uint32(in, &mi->flags);
+ camel_file_util_decode_uint32(in, &mi->unread);
+ camel_file_util_decode_uint32(in, &mi->total);
+
+ if (!ferror(in))
+ return mi;
+
+ camel_store_summary_info_free(s, mi);
+
+ return NULL;
+}
+
+static int
+folder_info_save(CamelStoreSummary *s, FILE *out, CamelFolderInfo *mi)
+{
+ io(printf("Saving folder info\n"));
+
+ camel_file_util_encode_string(out, camel_folder_info_full(s, mi));
+ camel_file_util_encode_uint32(out, mi->flags);
+ camel_file_util_encode_uint32(out, mi->unread);
+ camel_file_util_encode_uint32(out, mi->total);
+
+ return ferror(out);
+}
+
+static void
+folder_info_free(CamelStoreSummary *s, CamelFolderInfo *mi)
+{
+ g_free(mi->full);
+ g_free(mi->uri);
+ e_memchunk_free(s->folder_info_chunks, mi);
+}
+
+static const char *
+folder_info_string(CamelStoreSummary *s, const CamelFolderInfo *mi, int type)
+{
+ const char *p;
+
+ /* FIXME: Locks? */
+
+ g_assert (mi != NULL);
+
+ switch (type) {
+ case CAMEL_STORE_SUMMARY_FULL:
+ return mi->full;
+ case CAMEL_STORE_SUMMARY_NAME:
+ p = strrchr(mi->full, '/');
+ if (p)
+ return p;
+ else
+ return mi->full;
+ case CAMEL_STORE_SUMMARY_URI:
+ if (mi->uri)
+ return mi->uri;
+ if (s->uri_prefix)
+ return (((CamelFolderInfo *)mi)->uri = g_strdup_printf("%s%s", s->uri_prefix, mi->full));
+ }
+
+ return "";
+}
+
+static void
+folder_info_set_string (CamelStoreSummary *s, CamelFolderInfo *mi, int type, const char *str)
+{
+ const char *p;
+ char *v;
+ int len;
+
+ g_assert (mi != NULL);
+
+ switch(type) {
+ case CAMEL_STORE_SUMMARY_FULL:
+ g_free(mi->full);
+ g_free(mi->uri);
+ mi->full = g_strdup(str);
+ break;
+ case CAMEL_STORE_SUMMARY_NAME:
+ p = strrchr(mi->full, '/');
+ if (p) {
+ len = p-mi->full+1;
+ v = g_malloc(len+strlen(str)+1);
+ memcpy(v, mi->full, len);
+ strcpy(v+len, str);
+ } else {
+ v = g_strdup(str);
+ }
+ g_free(mi->full);
+ mi->full = v;
+ break;
+ case CAMEL_STORE_SUMMARY_URI:
+ if (s->uri_prefix) {
+ len = strlen(s->uri_prefix);
+ if (len > strlen(str)
+ || strncmp(s->uri_prefix, str, len) != 0) {
+ g_warning("Trying to set folderinfo uri '%s' for summary with prefix '%s'",
+ str, s->uri_prefix);
+ return;
+ }
+ g_free(mi->full);
+ g_free(mi->uri);
+ mi->full = g_strdup(str + len);
+ mi->uri = g_strdup(str);
+ } else {
+ g_warning("Trying to set folderinfo uri '%s' for summary with no uri prefix", str);
+ }
+ break;
+ }
+}