/* * Copyright (C) 2000 Helix Code Inc. * * Authors: Not Zed * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public License * as published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "camel-mh-summary.h" #include #include #include #include #include #include #include #include #include #include #define d(x) #define CAMEL_MH_SUMMARY_VERSION (0x2000) static CamelMessageInfo *message_info_new(CamelFolderSummary *, struct _header_raw *); static void camel_mh_summary_class_init (CamelMhSummaryClass *class); static void camel_mh_summary_init (CamelMhSummary *gspaper); static void camel_mh_summary_finalise (CamelObject *obj); #define _PRIVATE(x) (((CamelMhSummary *)(x))->priv) struct _CamelMhSummaryPrivate { char *current_uid; }; static CamelFolderSummaryClass *parent_class; CamelType camel_mh_summary_get_type (void) { static CamelType type = CAMEL_INVALID_TYPE; if (type == CAMEL_INVALID_TYPE) { type = camel_type_register(camel_folder_summary_get_type (), "CamelMhSummary", sizeof(CamelMhSummary), sizeof(CamelMhSummaryClass), (CamelObjectClassInitFunc)camel_mh_summary_class_init, NULL, (CamelObjectInitFunc)camel_mh_summary_init, (CamelObjectFinalizeFunc)camel_mh_summary_finalise); } return type; } static void camel_mh_summary_class_init (CamelMhSummaryClass *class) { CamelFolderSummaryClass *sklass = (CamelFolderSummaryClass *) class; parent_class = CAMEL_FOLDER_SUMMARY_CLASS (camel_type_get_global_classfuncs(camel_folder_summary_get_type ())); /* override methods */ sklass->message_info_new = message_info_new; } static void camel_mh_summary_init (CamelMhSummary *o) { struct _CamelFolderSummary *s = (CamelFolderSummary *) o; o->priv = g_malloc0(sizeof(*o->priv)); /* set unique file version */ s->version += CAMEL_MH_SUMMARY_VERSION; } static void camel_mh_summary_finalise(CamelObject *obj) { CamelMhSummary *o = (CamelMhSummary *)obj; g_free(o->mh_path); g_free(o->priv); } /** * camel_mh_summary_new: * * Create a new CamelMhSummary object. * * Return value: A new #CamelMhSummary object. **/ CamelMhSummary *camel_mh_summary_new (const char *filename, const char *mhdir, ibex *index) { CamelMhSummary *o = (CamelMhSummary *)camel_object_new(camel_mh_summary_get_type ()); camel_folder_summary_set_build_content((CamelFolderSummary *)o, TRUE); camel_folder_summary_set_filename((CamelFolderSummary *)o, filename); o->mh_path = g_strdup(mhdir); o->index = index; return o; } static CamelMessageInfo *message_info_new(CamelFolderSummary * s, struct _header_raw *h) { CamelMessageInfo *mi; CamelMhSummary *mhs = (CamelMhSummary *)s; mi = ((CamelFolderSummaryClass *) parent_class)->message_info_new(s, h); if (mi) { /* it only ever indexes 1 message at a time */ mi->uid = g_strdup(mhs->priv->current_uid); } return mi; } int camel_mh_summary_load(CamelMhSummary * mhs, int forceindex) { CamelFolderSummary *s = CAMEL_FOLDER_SUMMARY(mhs); d(printf("loading summary ...\n")); if (forceindex || camel_folder_summary_load(s) == -1) { camel_folder_summary_clear(s); } return camel_mh_summary_check(mhs, forceindex); } int camel_mh_summary_add(CamelMhSummary * mhs, const char *name, int forceindex) { char *filename = g_strdup_printf("%s/%s", mhs->mh_path, name); int fd; CamelMimeParser *mp; d(printf("summarising: %s\n", name)); fd = open(filename, O_RDONLY); if (fd == -1) { g_warning("Cannot summarise/index: %s: %s", filename, strerror(errno)); g_free(filename); return -1; } mp = camel_mime_parser_new(); camel_mime_parser_scan_from(mp, FALSE); camel_mime_parser_init_with_fd(mp, fd); if (mhs->index && (forceindex || !ibex_contains_name(mhs->index, (char *)name))) { d(printf("forcing indexing of message content\n")); camel_folder_summary_set_index((CamelFolderSummary *)mhs, mhs->index); } else { camel_folder_summary_set_index((CamelFolderSummary *)mhs, NULL); } mhs->priv->current_uid = (char *)name; camel_folder_summary_add_from_parser((CamelFolderSummary *)mhs, mp); camel_object_unref((CamelObject *)mp); mhs->priv->current_uid = NULL; camel_folder_summary_set_index((CamelFolderSummary *)mhs, NULL); g_free(filename); return 0; } static void remove_summary(char *key, CamelMessageInfo *info, CamelMhSummary *mhs) { d(printf("removing message %s from summary\n", key)); if (mhs->index) ibex_unindex(mhs->index, info->uid); camel_folder_summary_remove((CamelFolderSummary *)mhs, info); } int camel_mh_summary_check(CamelMhSummary * mhs, int forceindex) { DIR *dir; struct dirent *d; char *p, c; CamelMessageInfo *info; GHashTable *left; int i, count; d(printf("checking summary ...\n")); /* scan the directory, check for mail files not in the index, or index entries that no longer exist */ dir = opendir(mhs->mh_path); if (dir == NULL) return -1; /* keeps track of all uid's that have not been processed */ left = g_hash_table_new(g_str_hash, g_str_equal); count = camel_folder_summary_count((CamelFolderSummary *)mhs); for (i=0;iuid, info); } } while ( (d = readdir(dir)) ) { /* FIXME: also run stat to check for regular file */ p = d->d_name; while ( (c = *p++) ) { if (!isdigit(c)) break; } if (c==0) { info = camel_folder_summary_uid((CamelFolderSummary *)mhs, d->d_name); if (info == NULL || (mhs->index && (!ibex_contains_name(mhs->index, d->d_name)))) { /* need to add this file to the summary */ if (info != NULL) { g_hash_table_remove(left, info->uid); camel_folder_summary_remove((CamelFolderSummary *)mhs, info); } camel_mh_summary_add(mhs, d->d_name, forceindex); } else { g_hash_table_remove(left, info->uid); } } } closedir(dir); g_hash_table_foreach(left, (GHFunc)remove_summary, mhs); g_hash_table_destroy(left); /* force a save of the index, just to make sure */ /* note this could be expensive so possibly shouldn't be here as such */ if (mhs->index) { ibex_save(mhs->index); } return 0; } /* sync the summary with the ondisk files. It doesnt store the state in the file, the summary only, == MUCH faster */ int camel_mh_summary_sync(CamelMhSummary * mhs, int expunge, CamelFolderChangeInfo *changes, CamelException *ex) { int count, i; CamelMessageInfo *info; char *name; printf("summary_sync(expunge=%s)\n", expunge?"true":"false"); if (mhs->index) { ibex_save(mhs->index); } if (!expunge) return 0; count = camel_folder_summary_count((CamelFolderSummary *)mhs); for (i=count-1;i>=0;i--) { info = camel_folder_summary_index((CamelFolderSummary *)mhs, i); if (info && info->flags & CAMEL_MESSAGE_DELETED) { name = g_strdup_printf("%s/%s", mhs->mh_path, info->uid); d(printf("deleting %s\n", name)); if (unlink(name) == 0 || errno==ENOENT) { camel_folder_change_info_remove_uid(changes, info->uid); camel_folder_summary_remove((CamelFolderSummary *)mhs, info); } } } return 0; }