aboutsummaryrefslogtreecommitdiffstats
path: root/shell/e-config-upgrade.c
diff options
context:
space:
mode:
Diffstat (limited to 'shell/e-config-upgrade.c')
-rw-r--r--shell/e-config-upgrade.c1699
1 files changed, 1699 insertions, 0 deletions
diff --git a/shell/e-config-upgrade.c b/shell/e-config-upgrade.c
new file mode 100644
index 0000000000..5cddd842c2
--- /dev/null
+++ b/shell/e-config-upgrade.c
@@ -0,0 +1,1699 @@
+/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
+/* e-upgrade.c - upgrade previous config versions
+ *
+ * Copyright (C) 2003 Ximian, Inc.
+ *
+ * Authors: Michael Zucchi <notzed@ximian.com>
+ * Jeffery Stedfast <fejj@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.
+ *
+ */
+
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <unistd.h>
+
+#include <stdio.h>
+#include <errno.h>
+#include <regex.h>
+#include <string.h>
+
+#include <glib.h>
+#include <gconf/gconf.h>
+#include <gconf/gconf-client.h>
+
+#include <libxml/xmlmemory.h>
+#include <libxml/parser.h>
+#include <libxml/tree.h>
+
+#include "e-config-upgrade.h"
+
+#define d(x) x
+
+/* output revision of configuration */
+#define CONF_MAJOR (1)
+#define CONF_MINOR (3)
+#define CONF_REVISION (0)
+
+/* major/minor/revision of existing config */
+static unsigned int major = -1;
+static unsigned int minor = -1;
+static unsigned int revision = -1;
+
+/* 1.0 details, if required */
+static GHashTable *accounts_1_0 = NULL;
+static GHashTable *accounts_name_1_0 = NULL;
+
+/* where files are stored */
+static const char *evolution_dir;
+
+static char hexnib[256] = {
+ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,
+ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,
+ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,
+ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,-1,-1,-1,-1,-1,-1,
+ -1,10,11,12,13,14,15,16,-1,-1,-1,-1,-1,-1,-1,-1,
+ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,
+ -1,10,11,12,13,14,15,16,-1,-1,-1,-1,-1,-1,-1,-1,
+ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,
+ -1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,-1,
+};
+
+static char *hex_decode(const char *val)
+{
+ char *o, *res;
+ const unsigned char *p = (const unsigned char *)val;
+
+ o = res = g_malloc(strlen(val)/2 + 1);
+ for (p=val;(p[0] && p[1]);p+=2)
+ *o++ = (hexnib[p[0]] << 4) | hexnib[p[1]];
+ *o = 0;
+
+ return res;
+}
+
+static char *url_decode(const char *val)
+{
+ char *o, *res, c;
+ const unsigned char *p = (const unsigned char *)val;
+
+ o = res = g_malloc(strlen(val) + 1);
+ while (*p) {
+ c = *p++;
+ if (c == '%'
+ && hexnib[p[0]] != -1 && hexnib[p[1]] != -1) {
+ *o++ = (hexnib[p[0]] << 4) | hexnib[p[1]];
+ p+=2;
+ } else
+ *o++ = c;
+ }
+ *o = 0;
+
+ return res;
+}
+
+/* so we dont need camel, just copy here */
+static int
+camel_file_util_decode_uint32 (FILE *in, guint32 *dest)
+{
+ guint32 value = 0;
+ int v;
+
+ /* until we get the last byte, keep decoding 7 bits at a time */
+ while ( ((v = fgetc (in)) & 0x80) == 0 && v!=EOF) {
+ value |= v;
+ value <<= 7;
+ }
+ if (v == EOF) {
+ *dest = value >> 7;
+ return -1;
+ }
+ *dest = value | (v & 0x7f);
+
+ return 0;
+}
+
+static int
+camel_file_util_decode_string (FILE *in, char **str)
+{
+ guint32 len;
+ register char *ret;
+
+ if (camel_file_util_decode_uint32 (in, &len) == -1) {
+ *str = NULL;
+ return -1;
+ }
+
+ len--;
+ if (len > 65536) {
+ *str = NULL;
+ return -1;
+ }
+
+ ret = g_malloc (len+1);
+ if (len > 0 && fread (ret, len, 1, in) != 1) {
+ g_free (ret);
+ *str = NULL;
+ return -1;
+ }
+
+ ret[len] = 0;
+ *str = ret;
+ return 0;
+}
+
+/* For 1.0.8 conversion */
+
+/* as much info as we have on a given account */
+struct _account_info {
+ char *name;
+ char *uri;
+ char *base_uri;
+ union {
+ struct {
+ /* for imap */
+ char *namespace;
+ char *namespace_full;
+ guint32 capabilities;
+ GHashTable *folders;
+ char dir_sep;
+ } imap;
+ } u;
+};
+
+struct _imap_folder_info {
+ char *folder;
+ /* encoded? decoded? canonicalised? */
+ char dir_sep;
+};
+
+static char *parse_lsub(const char *lsub, char *dir_sep)
+{
+ static int comp;
+ static regex_t pat;
+ regmatch_t match[3];
+ char *m = "^\\* LSUB \\([^)]*\\) \"?([^\" ]+)\"? \"?(.*)\"?$";
+
+ if (!comp) {
+ if (regcomp(&pat, m, REG_EXTENDED|REG_ICASE) == -1) {
+ g_warning("reg comp '%s' failed: %s", m, g_strerror(errno));
+ return NULL;
+ }
+ comp = 1;
+ }
+
+ if (regexec(&pat, lsub, 3, match, 0) == 0) {
+ if (match[1].rm_so != -1 && match[2].rm_so != -1) {
+ if (dir_sep)
+ *dir_sep = (match[1].rm_eo - match[1].rm_so == 1) ? lsub[match[1].rm_so] : 0;
+ return g_strndup(lsub + match[2].rm_so, match[2].rm_eo - match[2].rm_so);
+ }
+ }
+
+ return NULL;
+}
+
+static int read_imap_storeinfo(struct _account_info *si)
+{
+ FILE *storeinfo;
+ guint32 tmp;
+ char *buf, *folder, dir_sep, *path, *name, *p;
+ struct _imap_folder_info *fi;
+
+ si->u.imap.folders = g_hash_table_new(g_str_hash, g_str_equal);
+
+ /* get details from uri first */
+ name = strstr(si->uri, ";override_namespace");
+ if (name) {
+ name = strstr(si->uri, ";namespace=");
+ if (name) {
+ char *end;
+
+ name += strlen(";namespace=");
+ if (*name == '\"') {
+ name++;
+ end = strchr(name, '\"');
+ } else {
+ end = strchr(name, ';');
+ }
+
+ if (end) {
+ /* try get the dir_sep from the namespace */
+ si->u.imap.namespace = g_strndup(name, end-name);
+
+ p = si->u.imap.namespace;
+ while ((dir_sep = *p++)) {
+ if (dir_sep < '0'
+ || (dir_sep > '9' && dir_sep < 'A')
+ || (dir_sep > 'Z' && dir_sep < 'a')
+ || (dir_sep > 'z')) {
+ si->u.imap.dir_sep = dir_sep;
+ break;
+ }
+ p++;
+ }
+ }
+ }
+ }
+
+ /* now load storeinfo if it exists */
+ path = g_build_filename(evolution_dir, "mail/imap", si->base_uri+7, "storeinfo", NULL);
+ storeinfo = fopen(path, "r");
+ g_free(path);
+ if (storeinfo == NULL) {
+ g_warning("could not find imap store info '%s'", path);
+ return -1;
+ }
+
+ /* ignore version */
+ camel_file_util_decode_uint32(storeinfo, &tmp);
+ camel_file_util_decode_uint32(storeinfo, &si->u.imap.capabilities);
+ g_free(si->u.imap.namespace);
+ camel_file_util_decode_string (storeinfo, &si->u.imap.namespace);
+ camel_file_util_decode_uint32 (storeinfo, &tmp);
+ si->u.imap.dir_sep = tmp;
+ /* strip trailing dir_sep or / */
+ if (si->u.imap.namespace
+ && (si->u.imap.namespace[strlen(si->u.imap.namespace)-1] == si->u.imap.dir_sep
+ || si->u.imap.namespace[strlen(si->u.imap.namespace)-1] == '/')) {
+ si->u.imap.namespace[strlen(si->u.imap.namespace)-1] = 0;
+ }
+
+ d(printf("namespace '%s' dir_sep '%c'\n", si->u.imap.namespace, si->u.imap.dir_sep?si->u.imap.dir_sep:'?'));
+
+ while (camel_file_util_decode_string (storeinfo, &buf) == 0) {
+ folder = parse_lsub(buf, &dir_sep);
+ if (folder) {
+ fi = g_malloc0(sizeof(*fi));
+ fi->folder = folder;
+ fi->dir_sep = dir_sep;
+#if d(!)0
+ printf(" add folder '%s' ", folder);
+ if (dir_sep)
+ printf("'%c'\n", dir_sep);
+ else
+ printf("NIL\n");
+#endif
+ g_hash_table_insert(si->u.imap.folders, fi->folder, fi);
+ } else {
+ g_warning("Could not parse LIST result '%s'\n", buf);
+ }
+ }
+ fclose(storeinfo);
+
+ return 0;
+}
+
+static char *get_base_uri(const char *val)
+{
+ const char *tmp;
+
+ tmp = strchr(val, ':');
+ if (tmp) {
+ tmp++;
+ if (strncmp(tmp, "//", 2) == 0)
+ tmp += 2;
+ tmp = strchr(tmp, '/');
+ }
+
+ if (tmp)
+ return g_strndup(val, tmp-val);
+ else
+ return g_strdup(val);
+}
+
+static char *upgrade_uri(const char *uri)
+{
+ char *out = NULL;
+
+ /* upgrades camel uri's */
+
+ if (major <=1 && minor < 2) {
+ if (strncmp(uri, "imap:", 5) == 0) {
+ char *base_uri, dir_sep, *folder, *p;
+ struct _account_info *ai;
+
+ /* add namespace, canonicalise dir_sep to / */
+ base_uri = get_base_uri(uri);
+ ai = g_hash_table_lookup(accounts_1_0, base_uri);
+
+ if (ai == NULL) {
+ g_free(base_uri);
+ return NULL;
+ }
+
+ dir_sep = ai->u.imap.dir_sep;
+ if (dir_sep == 0) {
+ /* no dir_sep listed, try get it from the namespace, if set */
+ if (ai->u.imap.namespace != NULL) {
+ p = ai->u.imap.namespace;
+ while ((dir_sep = *p++)) {
+ if (dir_sep < '0'
+ || (dir_sep > '9' && dir_sep < 'A')
+ || (dir_sep > 'Z' && dir_sep < 'a')
+ || (dir_sep > 'z')) {
+ break;
+ }
+ p++;
+ }
+ }
+
+ /* give up ... */
+ if (dir_sep == 0) {
+ g_free(base_uri);
+ return NULL;
+ }
+ }
+
+ folder = g_strdup(uri + strlen(base_uri)+1);
+
+ /* Add the namespace before the mailbox name, unless the mailbox is INBOX */
+ if (ai->u.imap.namespace && strcmp(folder, "INBOX") != 0)
+ out = g_strdup_printf("%s/%s/%s", base_uri, ai->u.imap.namespace, folder);
+ else
+ out = g_strdup_printf("%s/%s", base_uri, folder);
+
+ p = out;
+ while (*p) {
+ if (*p == dir_sep)
+ *p = '/';
+ p++;
+ }
+
+ g_free(folder);
+ g_free(base_uri);
+ } else if (strncmp(uri, "exchange:", 9) == 0) {
+ char *base_uri, *folder, *p;
+
+ /* exchange://user@host/exchange/ * -> exchange://user@host/personal/ * */
+ /* Any url encoding (%xx) in the folder name is also removed */
+ base_uri = get_base_uri(uri);
+ uri += strlen(base_uri) + 1;
+ if (strncmp(uri, "exchange/", 9) == 0) {
+ folder = url_decode(uri+9);
+ p = strchr(folder, '/');
+ out = g_strdup_printf("%s/personal%s", base_uri, p?p:"/");
+ g_free(folder);
+ }
+ } else if (strncmp(uri, "exchanget:", 10) == 0) {
+ /* these should be converted in the accounts table when it is loaded */
+ g_warning("exchanget: uri not converted: '%s'", uri);
+ }
+ }
+
+ return out;
+}
+
+static char *upgrade_evolution_uri(const char *uri)
+{
+ char *out = NULL;
+
+ if (!strcmp (uri, "evolution:/local/Inbox")) {
+ return g_strdup("default:mail");
+ } else if (!strcmp (uri, "evolution:/local/Calendar")) {
+ return g_strdup("default:calendar");
+ } else if (!strcmp (uri, "evolution:/local/Contacts")) {
+ return g_strdup("default:contacts");
+ } else if (!strcmp (uri, "evolution:/local/Tasks")) {
+ return g_strdup("default:tasks");
+ } if (!strncmp(uri, "evolution:/", 11)) {
+ char *account, *tmp, *folder, *p;
+ struct _account_info *ai;
+
+ d(printf("convert url '%s'\n", uri));
+ tmp = strchr(uri+11, '/');
+ if (tmp == NULL)
+ return NULL;
+
+ folder = *tmp?tmp+1:tmp;
+ account = g_strndup(uri+11, tmp-uri-11);
+ d(printf(" looking for account '%s'\n", account));
+ ai = g_hash_table_lookup(accounts_name_1_0, account);
+ if (ai && !strncmp(ai->base_uri, "imap:", 5)) {
+ /* Add namespace to evolution url's of imap accounts, if the account uses a namespace */
+ d(printf("found account ... '%s', folder = '%s'\n", ai->name, folder));
+
+ if (ai->u.imap.namespace && strcmp(folder, "INBOX") != 0)
+ out = g_strdup_printf("evolution:/%s/%s/%s", account, ai->u.imap.namespace, folder);
+ else
+ out = g_strdup_printf("evolution:/%s/%s", account, folder);
+
+ if (ai->u.imap.dir_sep) {
+ p = out + strlen("evolution://") + strlen(account);
+ while (*p) {
+ if (*p == ai->u.imap.dir_sep)
+ *p = '/';
+ p++;
+ }
+ }
+ } else if (ai && !strncmp(ai->base_uri, "exchange:", 9)) {
+ /* add personal to exchange url's */
+ folder = url_decode(folder);
+ out = g_strdup_printf("evolution:/%s/personal/%s", account, folder);
+ g_free(folder);
+ }
+ g_free(account);
+ }
+
+ return out;
+}
+
+static char *upgrade_type(const char *type)
+{
+ char *res = NULL;
+
+ /*
+ <item type="ldap-contacts ...>
+ to
+ <item type="contacts/ldap ...>
+ */
+
+ if (strcmp(type, "ldap-contacts") == 0)
+ res = g_strdup("contacts/ldap");
+
+ return res;
+}
+
+enum {
+ CONVERT_CONTENT,
+ CONVERT_ARG,
+};
+
+struct _convert {
+ int type;
+ char *tag;
+ char *arg;
+ char *(*convert)(const char *val);
+} convert_table [] = {
+ /* fix evolution uri's in shortcuts file */
+ { CONVERT_CONTENT, "item", NULL, upgrade_evolution_uri },
+ /* ldap item/type's converted from ldap-contacts to contacts/ldap */
+ { CONVERT_ARG, "item", "type", upgrade_type },
+ /* fix folder uri's in filters/vfolders */
+ { CONVERT_ARG, "folder", "uri", upgrade_uri },
+};
+
+#define CONVERT_SIZEOF (sizeof(convert_table)/sizeof(convert_table[0]))
+
+static int upgrade_xml_1_0_rec(xmlNodePtr node)
+{
+ int i;
+ struct _convert *ct;
+ int scan = TRUE;
+ int work = FALSE;
+ char *txt, *newtxt;
+
+ for (i=0;i<CONVERT_SIZEOF;i++) {
+ ct = &convert_table[i];
+ if (!strcmp(node->name, ct->tag)) {
+ switch(ct->type) {
+ case CONVERT_CONTENT:
+ txt = xmlNodeGetContent(node);
+ newtxt = ct->convert(txt);
+ d(printf("Upgrade content '%s': '%s' -> '%s'\n", ct->tag, txt, newtxt?newtxt:"unchanged"));
+ xmlFree(txt);
+ if (newtxt) {
+ xmlNodeSetContent(node, newtxt);
+ g_free(newtxt);
+ work = TRUE;
+ }
+ scan = FALSE;
+ break;
+ case CONVERT_ARG:
+ txt = xmlGetProp(node, ct->arg);
+ if (txt) {
+ newtxt = ct->convert(txt);
+ d(printf("Upgrade prop '%s' '%s': '%s' -> '%s'\n", ct->tag, ct->arg, txt, newtxt?newtxt:"unchanged"));
+ xmlFree(txt);
+ if (newtxt) {
+ xmlSetProp(node, ct->arg, newtxt);
+ g_free(newtxt);
+ work = TRUE;
+ }
+ }
+ break;
+ }
+ }
+ }
+
+ if (scan) {
+ node = node->children;
+ while (node) {
+ work |= upgrade_xml_1_0_rec(node);
+ node = node->next;
+ }
+ }
+
+ return work;
+}
+
+static int upgrade_xml_file_1_0(const char *filename)
+{
+ xmlDocPtr doc;
+ char *savename;
+ struct stat st;
+ int res;
+
+ /* FIXME: do something nicer with the errors */
+
+ savename = alloca(strlen(filename)+64);
+ sprintf(savename, "%s.save-%u.%u.%u", filename, major, minor, revision);
+ if (stat(savename, &st) == 0) {
+ fprintf(stderr, "xml file `%s' already upgraded\n", filename);
+ return 0;
+ }
+
+ /* no file, no error */
+ if (stat(filename, &st) == -1)
+ return 0;
+
+ doc = xmlParseFile (filename);
+ if (!doc || !doc->xmlRootNode) {
+ fprintf (stderr, "Failed to load %s\n", filename);
+ return -1;
+ }
+
+ if (!upgrade_xml_1_0_rec(doc->xmlRootNode)) {
+ xmlFreeDoc(doc);
+ printf("file %s contains no old urls\n", filename);
+ return 0;
+ }
+
+ d(printf("backing up %s to %s\n", filename, savename));
+
+ if (rename(filename, savename) == -1) {
+ xmlFreeDoc(doc);
+ fprintf(stderr, "could not rename '%s' to '%s': %s\n", filename, savename, strerror(errno));
+ return -1;
+ }
+
+ res = xmlSaveFormatFile(filename, doc, 1);
+
+ xmlFreeDoc(doc);
+
+ return res;
+}
+
+/* ********************************************************************** */
+/* Tables for converting flat bonobo conf -> gconf xml blob */
+/* ********************************************************************** */
+
+/* for remapping bonobo-conf account data into the new xml blob format */
+/* These are used in build_xml, and order must match the lookup_table */
+enum _map_t {
+ MAP_END = 0, /* end of line*/
+ MAP_BOOL, /* bool -> prop of name 'to' value true or false */
+ MAP_LONG, /* long -> prop of name 'to' value a long */
+ MAP_STRING, /* string -> prop of name 'to' */
+ MAP_ENUM, /* long/bool -> prop of name 'to', with the value indexed into the child map table's from field */
+ MAP_CHILD, /* a new child of name 'to' */
+ MAP_MASK = 0x3f,
+ MAP_URI_UPGRADE = 0x40, /* if from 1.0.x, upgrade any uri's present */
+ MAP_CONTENT = 0x80, /* if set, create a new node of name 'to' instead of a property */
+};
+
+struct _map_table {
+ char *from;
+ char *to;
+ int type;
+ struct _map_table *child;
+};
+
+/* Mail/Accounts/ * */
+struct _map_table cc_map[] = {
+ { "account_always_cc_%i", "always", MAP_BOOL },
+ { "account_always_cc_addrs_%i", "recipients", MAP_STRING|MAP_CONTENT },
+ { NULL },
+};
+
+struct _map_table bcc_map[] = {
+ { "account_always_cc_%i", "always", MAP_BOOL },
+ { "account_always_bcc_addrs_%i", "recipients", MAP_STRING|MAP_CONTENT },
+ { NULL },
+};
+
+struct _map_table pgp_map[] = {
+ { "account_pgp_encrypt_to_self_%i", "encrypt-to-self", MAP_BOOL },
+ { "account_pgp_always_trust_%i", "always-trust", MAP_BOOL },
+ { "account_pgp_always_sign_%i", "always-sign", MAP_BOOL },
+ { "account_pgp_no_imip_sign_%i", "no-imip-sign", MAP_BOOL },
+ { "account_pgp_key_%i", "key-id", MAP_STRING|MAP_CONTENT },
+ { NULL },
+};
+
+struct _map_table smime_map[] = {
+ { "account_smime_encrypt_to_self_%i", "encrypt-to-self", MAP_BOOL },
+ { "account_smime_always_sign_%i", "always-sign", MAP_BOOL },
+ { "account_smime_key_%i", "key-id", MAP_STRING|MAP_CONTENT },
+ { NULL },
+};
+
+struct _map_table identity_map[] = {
+ { "identity_name_%i", "name", MAP_STRING|MAP_CONTENT },
+ { "identity_address_%i", "addr-spec", MAP_STRING|MAP_CONTENT },
+ { "identity_reply_to_%i", "reply-to", MAP_STRING|MAP_CONTENT },
+ { "identity_organization_%i", "organization", MAP_STRING|MAP_CONTENT },
+ { "identity_autogenerated_signature_%i", "signature", MAP_BOOL },
+ { "identity_def_signature_%i", "default", MAP_LONG },
+ { NULL },
+};
+
+struct _map_table source_map[] = {
+ { "source_save_passwd_%i", "save-passwd", MAP_BOOL },
+ { "source_keep_on_server_%i", "keep-on-server", MAP_BOOL },
+ { "source_auto_check_%i", "auto-check", MAP_BOOL },
+ { "source_auto_check_time_%i", "auto-check-timeout", MAP_LONG },
+ { "source_url_%i", "url", MAP_STRING|MAP_CONTENT },
+ { NULL },
+};
+
+struct _map_table transport_map[] = {
+ { "transport_save_passwd_%i", "save-passwd", MAP_BOOL },
+ { "transport_url_%i", "url", MAP_STRING|MAP_CONTENT|MAP_URI_UPGRADE },
+ { NULL },
+};
+
+struct _map_table account_map[] = {
+ { "account_name_%i", "name", MAP_STRING },
+ { "source_enabled_%i", "enabled", MAP_BOOL },
+ { NULL, "identity", MAP_CHILD, identity_map },
+ { NULL, "source", MAP_CHILD, source_map },
+ { NULL, "transport", MAP_CHILD, transport_map },
+ { "account_drafts_folder_uri_%i", "drafts-folder", MAP_STRING|MAP_CONTENT|MAP_URI_UPGRADE },
+ { "account_sent_folder_uri_%i", "sent-folder", MAP_STRING|MAP_CONTENT|MAP_URI_UPGRADE },
+ { NULL, "auto-cc", MAP_CHILD, cc_map },
+ { NULL, "auto-bcc", MAP_CHILD, bcc_map },
+ { NULL, "pgp", MAP_CHILD, pgp_map },
+ { NULL, "smime", MAP_CHILD, smime_map },
+ { NULL },
+};
+
+/* /Mail/Signatures/ * */
+struct _map_table signature_format_map[] = {
+ { "text/plain", },
+ { "text/html", },
+ { NULL }
+};
+
+struct _map_table signature_map[] = {
+ { "name_%i", "name", MAP_STRING },
+ { "html_%i", "format", MAP_ENUM, signature_format_map },
+ { "filename_%i", "filename", MAP_STRING|MAP_CONTENT },
+ { "script_%i", "script", MAP_STRING|MAP_CONTENT },
+ { NULL },
+};
+
+
+static char *get_name(const char *in, int index)
+{
+ char c, *res;
+ GString *out = g_string_new("");
+
+ while ( (c = *in++) ) {
+ if (c == '%') {
+ c = *in++;
+ switch(c) {
+ case '%':
+ g_string_append_c(out, '%');
+ break;
+ case 'i':
+ g_string_append_printf(out, "%d", index);
+ break;
+ }
+ } else {
+ g_string_append_c(out, c);
+ }
+ }
+
+ res = out->str;
+ g_string_free(out, FALSE);
+
+ return res;
+}
+
+static xmlNodePtr lookup_bconf_entry(xmlNodePtr source, const char *name)
+{
+ xmlNodePtr node = source->children;
+ int found;
+ char *val;
+
+ while (node) {
+ if (!strcmp(node->name, "entry")) {
+ val = xmlGetProp(node, "name");
+ found = val && strcmp(val, name) == 0;
+ xmlFree(val);
+ if (found)
+ break;
+ }
+ node = node->next;
+ }
+
+ return node;
+}
+
+static char *lookup_bconf_value(xmlNodePtr source, const char *name)
+{
+ xmlNodePtr node = lookup_bconf_entry(source, name);
+
+ if (node)
+ return xmlGetProp(node, "value");
+ else
+ return NULL;
+}
+
+static xmlNodePtr lookup_bconf_path(xmlDocPtr doc, const char *path)
+{
+ xmlNodePtr root;
+ char *val;
+ int found;
+
+ root = doc->children;
+ if (strcmp(root->name, "bonobo-config") != 0) {
+ g_warning("not bonobo-config xml file\n");
+ return NULL;
+ }
+
+ root = root->children;
+ while (root) {
+ if (!strcmp(root->name, "section")) {
+ val = xmlGetProp(root, "path");
+ found = val && strcmp(val, path) == 0;
+ xmlFree(val);
+ if (found)
+ break;
+ }
+ root = root->next;
+ }
+
+ return root;
+}
+
+
+static char *lookup_bool(xmlNodePtr source, const char *name, struct _map_table *map)
+{
+ char *val, *res;
+
+ val = lookup_bconf_value(source, name);
+ if (val) {
+ res = g_strdup(val[0] == '1'?"true":"false");
+ xmlFree(val);
+ } else
+ res = NULL;
+
+ return res;
+}
+
+static char *lookup_long(xmlNodePtr source, const char *name, struct _map_table *map)
+{
+ char *val, *res;
+
+ val = lookup_bconf_value(source, name);
+ if (val) {
+ res = g_strdup(val);
+ xmlFree(val);
+ } else
+ res = NULL;
+
+ return res;
+}
+
+static char *lookup_string(xmlNodePtr source, const char *name, struct _map_table *map)
+{
+ char *val, *res;
+
+ val = lookup_bconf_value(source, name);
+ if (val) {
+ res = hex_decode(val);
+ xmlFree(val);
+ } else
+ res = NULL;
+
+ return res;
+}
+
+static char *lookup_enum(xmlNodePtr source, const char *name, struct _map_table *map)
+{
+ char *val;
+ int index = 0, i;
+
+ val = lookup_bconf_value(source, name);
+ if (val) {
+ index = atoi(val);
+ xmlFree(val);
+ }
+
+ for (i=0;map->child[i].from;i++)
+ if (i == index)
+ return g_strdup(map->child[i].from);
+
+ return NULL;
+}
+
+typedef char * (*lookup_func) (xmlNodePtr, const char *, struct _map_table *);
+
+static void
+build_xml(xmlNodePtr root, struct _map_table *map, int index, xmlNodePtr source)
+{
+ char *name, *value;
+ xmlNodePtr node;
+ lookup_func lookup_table[] = { lookup_bool, lookup_long, lookup_string, lookup_enum };
+
+ while (map->type != MAP_END) {
+ if ((map->type & MAP_MASK) == MAP_CHILD) {
+ node = xmlNewChild(root, NULL, map->to, NULL);
+ build_xml(node, map->child, index, source);
+ } else {
+ name = get_name(map->from, index);
+ value = lookup_table[(map->type&MAP_MASK)-1](source, name, map);
+
+ d(printf("key '%s=%s' -> ", name, value));
+
+ if (map->type & MAP_URI_UPGRADE) {
+ char *tmp = value;
+
+ value = upgrade_uri(tmp);
+ if (value)
+ g_free(tmp);
+ else
+ value = tmp;
+ }
+
+ d(printf("'%s=%s'\n", map->to, value));
+
+ if (map->type & MAP_CONTENT) {
+ if (value && value[0])
+ xmlNewTextChild(root, NULL, map->to, value);
+ } else {
+ xmlSetProp(root, map->to, value);
+ }
+ g_free(value);
+ g_free(name);
+ }
+ map++;
+ }
+}
+
+static int convert_xml_blob(GConfClient *gconf, xmlDocPtr doc, struct _map_table *map, const char *path, const char *outpath, const char *name, const char *idparam)
+{
+ xmlNodePtr source;
+ int count = 0, i;
+ GSList *list, *l;
+ char *val;
+
+ source = lookup_bconf_path(doc, path);
+ if (source) {
+ list = NULL;
+ val = lookup_bconf_value(source, "num");
+ if (val) {
+ count = atoi(val);
+ xmlFree(val);
+ }
+
+ d(printf("Found %d blobs at %s\n", count, path));
+
+ for (i = 0; i<count;i++) {
+ xmlDocPtr docout;
+ xmlChar *xmlbuf;
+ int n;
+ xmlNodePtr root;
+
+ docout = xmlNewDoc ("1.0");
+ root = xmlNewDocNode (docout, NULL, name, NULL);
+ xmlDocSetRootElement (docout, root);
+
+ /* This could be set with a MAP_UID type ... */
+ if (idparam) {
+ char buf[16];
+
+ sprintf(buf, "%d", i);
+ xmlSetProp(root, idparam, buf);
+ }
+
+ build_xml(root, map, i, source);
+
+ xmlDocDumpMemory (docout, &xmlbuf, &n);
+ xmlFreeDoc (docout);
+
+ list = g_slist_append(list, xmlbuf);
+ }
+
+ gconf_client_set_list(gconf, outpath, GCONF_VALUE_STRING, list, NULL);
+ while (list) {
+ l = list->next;
+ xmlFree(list->data);
+ g_slist_free_1(list);
+ list = l;
+ }
+ } else {
+ g_warning("could not find '%s' in old config database, skipping", path);
+ }
+
+ return 0;
+}
+
+/* ********************************************************************** */
+/* Tables for bonobo conf -> gconf conversion */
+/* ********************************************************************** */
+
+/* order important here, used to index a few tables below */
+enum {
+ BMAP_BOOL,
+ BMAP_BOOLNOT,
+ BMAP_INT,
+ BMAP_STRING,
+ BMAP_SIMPLESTRING, /* a non-encoded string */
+ BMAP_COLOUR,
+ BMAP_FLOAT, /* bloody floats, who uses floats ... idiots */
+ BMAP_STRLIST, /* strings separated to !<-->! -> gconf list */
+ BMAP_ANYLIST, /* corba sequence corba string -> gconf list */
+ BMAP_MASK = 0x7f,
+ BMAP_LIST = 0x80 /* from includes a %i field for the index of the key, to be converted to a list of type BMAP_* */
+};
+
+struct _gconf_map {
+ char *from;
+ char *to;
+ int type;
+};
+
+/* ********************************************************************** */
+
+static struct _gconf_map mail_accounts_map[] = {
+ /* /Mail/Accounts - most entries are processed via the xml blob routine */
+ /* This also works because the initial uid mapping is 1:1 with the list order */
+ { "default_account", "mail/default_account", BMAP_SIMPLESTRING },
+ { 0 },
+};
+
+static struct _gconf_map mail_display_map[] = {
+ /* /Mail/Display */
+ { "thread_list", "mail/display/thread_list", BMAP_BOOL },
+ { "thread_subject", "mail/display/thread_subject", BMAP_BOOL },
+ { "hide_deleted", "mail/display/show_deleted", BMAP_BOOLNOT },
+ { "preview_pane", "mail/display/show_preview", BMAP_BOOL },
+ { "paned_size", "mail/display/paned_size", BMAP_INT },
+ { "seen_timeout", "mail/display/mark_seen_timeout", BMAP_INT },
+ { "do_seen_timeout", "mail/display/mark_seen", BMAP_BOOL },
+ { "http_images", "mail/display/load_http_images", BMAP_BOOL },
+ { "citation_highlight", "mail/display/mark_citations", BMAP_BOOL },
+ { "citation_color", "mail/display/citation_colour", BMAP_COLOUR },
+ { "x_mailer_display_style", "mail/display/xmailer_mask", BMAP_INT },
+ { 0 },
+};
+
+static struct _gconf_map mail_format_map[] = {
+ /* /Mail/Format */
+ { "message_display_style", "mail/display/message_style", BMAP_INT },
+ { "send_html", "mail/composer/send_html", BMAP_BOOL },
+ { "default_reply_style", "mail/format/reply_style", BMAP_INT },
+ { "default_forward_style", "mail/format/forward_style", BMAP_INT },
+ { "default_charset", "mail/composer/charset", BMAP_STRING },
+ { "confirm_unwanted_html", "mail/prompts/unwanted_html", BMAP_BOOL },
+ { 0 },
+};
+
+static struct _gconf_map mail_trash_map[] = {
+ /* /Mail/Trash */
+ { "empty_on_exit", "mail/trash/empty_on_exit", BMAP_BOOL },
+ { 0 },
+};
+
+static struct _gconf_map mail_prompts_map[] = {
+ /* /Mail/Prompts */
+ { "confirm_expunge", "mail/prompts/expunge", BMAP_BOOL },
+ { "empty_subject", "mail/prompts/empty_subject", BMAP_BOOL },
+ { "only_bcc", "mail/prompts/only_bcc", BMAP_BOOL },
+ { 0 }
+};
+
+static struct _gconf_map mail_filters_map[] = {
+ /* /Mail/Filters */
+ { "log", "mail/filters/log", BMAP_BOOL },
+ { "log_path", "mail/filters/logfile", BMAP_STRING },
+ { 0 }
+};
+
+static struct _gconf_map mail_notify_map[] = {
+ /* /Mail/Notify */
+ { "new_mail_notification", "mail/notify/type", BMAP_INT },
+ { "new_mail_notification_sound_file", "mail/notify/sound", BMAP_STRING },
+ { 0 }
+};
+
+static struct _gconf_map mail_filesel_map[] = {
+ /* /Mail/Filesel */
+ { "last_filesel_dir", "mail/save_dir", BMAP_STRING },
+ { 0 }
+};
+
+static struct _gconf_map mail_composer_map[] = {
+ /* /Mail/Composer */
+ { "ViewFrom", "mail/composer/view/From", BMAP_BOOL },
+ { "ViewReplyTo", "mail/composer/view/ReplyTo", BMAP_BOOL },
+ { "ViewCC", "mail/composer/view/Cc", BMAP_BOOL },
+ { "ViewBCC", "mail/composer/view/Bcc", BMAP_BOOL },
+ { "ViewSubject", "mail/composer/view/Subject", BMAP_BOOL },
+ { 0 },
+};
+
+/* ********************************************************************** */
+
+static struct _gconf_map importer_elm_map[] = {
+ /* /Importer/Elm */
+ { "mail", "importer/elm/mail", BMAP_BOOL },
+ { "mail-imported", "importer/elm/mail-imported", BMAP_BOOL },
+ { 0 },
+};
+
+static struct _gconf_map importer_pine_map[] = {
+ /* /Importer/Pine */
+ { "mail", "importer/elm/mail", BMAP_BOOL },
+ { "address", "importer/elm/address", BMAP_BOOL },
+ { 0 },
+};
+
+static struct _gconf_map importer_netscape_map[] = {
+ /* /Importer/Netscape */
+ { "mail", "importer/netscape/mail", BMAP_BOOL },
+ { "settings", "importer/netscape/settings", BMAP_BOOL },
+ { "filters", "importer/netscape/filters", BMAP_BOOL },
+ { 0 },
+};
+
+/* ********************************************************************** */
+
+static struct _gconf_map myev_mail_map[] = {
+ /* /My-Evolution/Mail */
+ { "show_full_path", "summary/mail/show_full_paths", BMAP_BOOL },
+ { 0 },
+};
+
+static struct _gconf_map myev_rdf_map[] = {
+ /* /My-Evolution/RDF */
+ { "rdf_urls", "summary/rdf/uris", BMAP_STRLIST },
+ { "rdf_refresh_time", "summary/rdf/refresh_time", BMAP_INT },
+ { "limit", "summary/rdf/max_items", BMAP_INT },
+ { 0 },
+};
+
+static struct _gconf_map myev_weather_map[] = {
+ /* /My-Evolution/Weather */
+ { "stations", "summary/weather/stations", BMAP_STRLIST },
+ { "units", "summary/weather/use_metric", BMAP_BOOL }, /* this is use_metric bool in 1.3? */
+ { "weather_refresh_time", "summary/weather/refresh_time", BMAP_INT },
+ { 0 },
+};
+
+static struct _gconf_map myev_schedule_map[] = {
+ /* /My-Evolution/Shedule */
+ { "show_tasks", "summary/tasks/show_all", BMAP_BOOL }, /* this is show_all bool in 1.3? */
+ { 0 },
+};
+
+/* ********************************************************************** */
+
+/* This grabs the defaults from the first view ... (?) */
+static struct _gconf_map shell_views_map[] = {
+ /* /Shell/Views/0 */
+ { "Width", "shell/view_defaults/width", BMAP_INT },
+ { "Height", "shell/view_defaults/height", BMAP_INT },
+ { "CurrentShortcutsGroupNum", "shell/view_defaults/selected_shortcut_group", BMAP_INT },
+ { "FolderBarShown", "shell/view_defaults/show_folder_bar", BMAP_BOOL },
+ { "ShortcutBarShown", "shell/view_defaults/show_shortcut_bar", BMAP_BOOL },
+ { "HPanedPosition", "shell/view_defaults/shortcut_bar/width", BMAP_INT },
+ { "ViewPanedPosition", "shell/view_defaults/folder_bar/width", BMAP_INT },
+ { "DisplayedURI", "shell/view_defaults/folder_path", BMAP_STRING },
+ { "ShortcutBarGroup%iIconMode", "shell/view_defaults/shortcut_bar/icon_types", BMAP_INT|BMAP_LIST },
+ { 0 },
+};
+
+static struct _gconf_map offlinefolders_map[] = {
+ /* /OfflineFolders */
+ { "paths", "shell/offline/folder_paths", BMAP_ANYLIST },
+ { 0 },
+};
+
+static struct _gconf_map defaultfolders_map[] = {
+ /* /DefaultFolders */
+ { "mail_path", "shell/default_folders/mail_path", BMAP_STRING },
+ { "mail_uri", "shell/default_folders/mail_uri", BMAP_STRING },
+ { "contacts_path", "shell/default_folders/contacts_path", BMAP_STRING },
+ { "contacts_uri", "shell/default_folders/contacts_uri", BMAP_STRING },
+ { "calendar_path", "shell/default_folders/calendar_path", BMAP_STRING },
+ { "calendar_uri", "shell/default_folders/calendar_uri", BMAP_STRING },
+ { "tasks_path", "shell/default_folders/tasks_path", BMAP_STRING },
+ { "tasks_uri", "shell/default_folders/tasks_uri", BMAP_STRING },
+ { 0 },
+};
+
+static struct _gconf_map shell_map[] = {
+ /* /Shell */
+ { "StartOffline", "shell/start_offline", BMAP_BOOL },
+ { 0 },
+};
+
+/* ********************************************************************** */
+
+static struct _gconf_map addressbook_map[] = {
+ /* /Addressbook */
+ { "select_names_uri", "addressbook/select_names/last_used_uri", BMAP_STRING },
+ { 0 },
+};
+
+static struct _gconf_map addressbook_completion_map[] = {
+ /* /Addressbook/Completion */
+ { "uris", "addressbook/completion/uris", BMAP_STRING },
+ { 0 },
+};
+
+/* ********************************************************************** */
+
+static struct _gconf_map calendar_display_map[] = {
+ /* /Calendar/Display */
+ { "Timezone", "calendar/display/timezone", BMAP_STRING },
+ { "Use24HourFormat", "calendar/display/use_24hour_format", BMAP_BOOL },
+ { "WeekStartDay", "calendar/display/week_start_day", BMAP_INT },
+ { "DayStartHour", "calendar/display/day_start_hour", BMAP_INT },
+ { "DayStartMinute", "calendar/display/day_start_minute", BMAP_INT },
+ { "DayEndHour", "calendar/display/day_end_hour", BMAP_INT },
+ { "DayEndMinute", "calendar/display/day_end_minute", BMAP_INT },
+ { "TimeDivisions", "calendar/display/time_divisions", BMAP_INT },
+ { "View", "calendar/display/default_view", BMAP_INT },
+ { "HPanePosition", "calendar/display/hpane_position", BMAP_FLOAT },
+ { "VPanePosition", "calendar/display/vpane_position", BMAP_FLOAT },
+ { "MonthHPanePosition", "calendar/display/month_hpane_position", BMAP_FLOAT },
+ { "MonthVPanePosition", "calendar/display/month_vpane_position", BMAP_FLOAT },
+ { "CompressWeekend", "calendar/display/compress_weekend", BMAP_BOOL },
+ { "ShowEventEndTime", "calendar/display/show_event_end", BMAP_BOOL },
+ { "WorkingDays", "calendar/display/working_days", BMAP_INT },
+ { 0 },
+};
+
+static struct _gconf_map calendar_tasks_map[] = {
+ /* /Calendar/Tasks */
+ { "HideCompletedTasks", "calendar/tasks/hide_completed", BMAP_BOOL },
+ { "HideCompletedTasksUnits", "calendar/tasks/hide_completed_units", BMAP_STRING },
+ { "HideCompletedTasksValue", "calendar/tasks/hide_completed_value", BMAP_INT },
+ { 0 },
+};
+
+static struct _gconf_map calendar_tasks_colours_map[] = {
+ /* /Calendar/Tasks/Colors */
+ { "TasksDueToday", "calendar/tasks/colors/due_today", BMAP_STRING },
+ { "TasksOverDue", "calendar/tasks/colors/overdue", BMAP_STRING },
+ { "TasksDueToday", "calendar/tasks/colors/due_today", BMAP_STRING },
+ { 0 },
+};
+
+static struct _gconf_map calendar_other_map[] = {
+ /* /Calendar/Other */
+ { "ConfirmDelete", "calendar/prompts/confirm_delete", BMAP_BOOL },
+ { "ConfirmExpunge", "calendar/prompts/confirm_expunge", BMAP_BOOL },
+ { "UseDefaultReminder", "calendar/other/use_default_reminder", BMAP_BOOL },
+ { "DefaultReminderInterval", "calendar/other/default_reminder_interval", BMAP_INT },
+ { "DefaultReminderUnits", "calendar/other/default_reminder_units", BMAP_STRING },
+ { 0 },
+};
+
+static struct _gconf_map calendar_datenavigator_map[] = {
+ /* /Calendar/DateNavigator */
+ { "ShowWeekNumbers", "calendar/date_navigator/show_week_numbers", BMAP_BOOL },
+ { 0 },
+};
+
+static struct _gconf_map calendar_alarmnotify_map[] = {
+ /* /Calendar/AlarmNotify */
+ { "LastNotificationTime", "calendar/notify/last_notification_time", BMAP_INT },
+ { "CalendarToLoad%i", "calendar/notify/calendars", BMAP_STRING|BMAP_LIST },
+ { "BlessedProgram%i", "calendar/notify/programs", BMAP_STRING|BMAP_LIST },
+ { 0 },
+};
+
+/* ********************************************************************** */
+
+static struct {
+ char *root;
+ struct _gconf_map *map;
+} gconf_remap_list[] = {
+ { "/Mail/Accounts", mail_accounts_map },
+ { "/Mail/Display", mail_display_map },
+ { "/Mail/Format", mail_format_map },
+ { "/Mail/Trash", mail_trash_map },
+ { "/Mail/Prompts", mail_prompts_map },
+ { "/Mail/Filters", mail_filters_map },
+ { "/Mail/Notify", mail_notify_map },
+ { "/Mail/Filesel", mail_filesel_map },
+ { "/Mail/Composer", mail_composer_map },
+
+ { "/Importer/Elm", importer_elm_map },
+ { "/Importer/Pine", importer_pine_map },
+ { "/Importer/Netscape", importer_netscape_map },
+
+ { "/My-Evolution/Mail", myev_mail_map },
+ { "/My-Evolution/RDF", myev_rdf_map },
+ { "/My-Evolution/Weather", myev_weather_map },
+ { "/My-Evolution/Schedule", myev_schedule_map },
+
+ { "/Shell", shell_map },
+ { "/Shell/Views/0", shell_views_map },
+ { "/OfflineFolders", offlinefolders_map },
+ { "/DefaultFolders", defaultfolders_map },
+
+ { "/Addressbook", addressbook_map },
+ { "/Addressbook/Completion", addressbook_completion_map },
+
+ { "/Calendar/Display", calendar_display_map },
+ { "/Calendar/Tasks", calendar_tasks_map },
+ { "/Calendar/Tasks/Colors", calendar_tasks_colours_map },
+ { "/Calendar/Other/Map", calendar_other_map },
+ { "/Calendar/DateNavigator", calendar_datenavigator_map },
+ { "/Calendar/AlarmNotify", calendar_alarmnotify_map },
+
+ { 0 },
+};
+
+#define N_(x) x
+
+struct {
+ char *label;
+ char *colour;
+} label_default[5] = {
+ { N_("Important"), "#ff0000" }, /* red */
+ { N_("Work"), "#ff8c00" }, /* orange */
+ { N_("Personal"), "#008b00" }, /* forest green */
+ { N_("To Do"), "#0000ff" }, /* blue */
+ { N_("Later"), "#8b008b" } /* magenta */
+};
+
+/* remaps mail config from bconf to gconf */
+static int import_bonobo_config(xmlDocPtr config_doc, GConfClient *gconf)
+{
+ xmlNodePtr source;
+ int i, j, k;
+ struct _gconf_map *map;
+ char *path, *val, *tmp;
+ GSList *list, *l;
+ char buf[32];
+
+ /* process all flat config */
+ for (i=0;gconf_remap_list[i].root;i++) {
+ d(printf("Path: %s\n", gconf_remap_list[i].root));
+ source = lookup_bconf_path(config_doc, gconf_remap_list[i].root);
+ if (source == NULL)
+ continue;
+ map = gconf_remap_list[i].map;
+ for (j=0;map[j].from;j++) {
+ if (map[j].type & BMAP_LIST) {
+ /* collapse a multi-entry indexed field into a list */
+ list = NULL;
+ k = 0;
+ do {
+ path = get_name(map[j].from, k);
+ val = lookup_bconf_value(source, path);
+ d(printf("finding path '%s' = '%s'\n", path, val));
+ g_free(path);
+ if (val) {
+ switch(map[j].type & BMAP_MASK) {
+ case BMAP_BOOL:
+ case BMAP_INT:
+ list = g_slist_append(list, GINT_TO_POINTER(atoi(val)));
+ break;
+ case BMAP_STRING:
+ d(printf(" -> '%s'\n", hex_decode(val)));
+ list = g_slist_append(list, hex_decode(val));
+ break;
+ }
+ xmlFree(val);
+ k++;
+ }
+ } while (val);
+
+ if (list) {
+ const int gconf_type[] = { GCONF_VALUE_BOOL, GCONF_VALUE_BOOL, GCONF_VALUE_INT, GCONF_VALUE_STRING, GCONF_VALUE_STRING };
+
+ path = g_strdup_printf("/apps/evolution/%s", map[j].to);
+ gconf_client_set_list(gconf, path, gconf_type[map[j].type & BMAP_MASK], list, NULL);
+ g_free(path);
+ if ((map[j].type & BMAP_MASK) == BMAP_STRING)
+ g_slist_foreach(list, (GFunc)g_free, NULL);
+ g_slist_free(list);
+ }
+
+ continue;
+ } else if (map[j].type == BMAP_ANYLIST)
+ val = NULL;
+ else {
+ val = lookup_bconf_value(source, map[j].from);
+ if (val == NULL)
+ continue;
+ }
+ d(printf(" %s = '%s' -> %s [%d]\n", map[j].from, val, map[j].to, map[j].type));
+ path = g_strdup_printf("/apps/evolution/%s", map[j].to);
+ switch(map[j].type) {
+ case BMAP_BOOL:
+ gconf_client_set_bool(gconf, path, atoi(val), NULL);
+ break;
+ case BMAP_BOOLNOT:
+ gconf_client_set_bool(gconf, path, !atoi(val), NULL);
+ break;
+ case BMAP_INT:
+ gconf_client_set_int(gconf, path, atoi(val), NULL);
+ break;
+ case BMAP_STRING:
+ tmp = hex_decode(val);
+ gconf_client_set_string(gconf, path, tmp, NULL);
+ g_free(tmp);
+ break;
+ case BMAP_SIMPLESTRING:
+ gconf_client_set_string(gconf, path, val, NULL);
+ break;
+ case BMAP_FLOAT:
+ gconf_client_set_float(gconf, path, strtod(val, NULL), NULL);
+ break;
+ case BMAP_STRLIST:{
+ char **t = g_strsplit (val, " !<-->! ", 8196);
+
+ list = NULL;
+ for (k=0;t[k];k++)
+ list = g_slist_append(list, t[k]);
+ gconf_client_set_list(gconf, path, GCONF_VALUE_STRING, list, NULL);
+ g_slist_free(list);
+ g_strfreev(t);
+ break;}
+ case BMAP_ANYLIST:{
+ xmlNodePtr node = source->children;
+ list = NULL;
+
+ /* find the entry node */
+ while (node) {
+ if (!strcmp(node->name, "entry")) {
+ int found;
+
+ tmp = xmlGetProp(node, "name");
+ if (tmp) {
+ found = strcmp(tmp, map[j].from) == 0;
+ xmlFree(tmp);
+ if (found)
+ break;
+ }
+ }
+ node = node->next;
+ }
+
+ /* find the the any block */
+ if (node) {
+ node = node->children;
+ while (node) {
+ if (strcmp(node->name, "any") == 0)
+ break;
+ node = node->next;
+ }
+ }
+
+ /* skip to the value inside it */
+ if (node) {
+ node = node->children;
+ while (node) {
+ if (strcmp(node->name, "value") == 0)
+ break;
+ node = node->next;
+ }
+ }
+
+ if (node) {
+ node = node->children;
+ while (node) {
+ if (strcmp(node->name, "value") == 0)
+ list = g_slist_append(list, xmlNodeGetContent(node));
+ node = node->next;
+ }
+ }
+
+ /* & store */
+ if (list) {
+ gconf_client_set_list(gconf, path, GCONF_VALUE_STRING, list, NULL);
+ while (list) {
+ l = list->next;
+ xmlFree(list->data);
+ g_slist_free_1(list);
+ list = l;
+ }
+ }
+
+ break;}
+ case BMAP_COLOUR:
+ sprintf(buf, "#%06x", atoi(val) & 0xffffff);
+ gconf_client_set_string(gconf, path, buf, NULL);
+ break;
+ }
+ /* FIXME: handle errors */
+ g_free(path);
+ xmlFree(val);
+ }
+ }
+
+ /* Labels:
+ label string + label colour as integer
+ -> label string:# colour as hex */
+ source = lookup_bconf_path(config_doc, "/Mail/Labels");
+ if (source) {
+ list = NULL;
+ for (i=0;i<5;i++) {
+ char labx[16], colx[16];
+ char *lab, *col;
+
+ sprintf(labx, "label_%d", i);
+ sprintf(colx, "color_%d", i);
+ lab = lookup_string(source, labx, NULL);
+ col = lookup_bconf_value(source, colx);
+ if (col) {
+ sprintf(colx, "#%06x", atoi(col) & 0xffffff);
+ xmlFree(col);
+ } else
+ strcpy(colx, label_default[i].colour);
+ val = g_strdup_printf("%s:%s", lab?lab:label_default[i].label, colx);
+ list = g_slist_append(list, val);
+ g_free(lab);
+ }
+
+ gconf_client_set_list(gconf, "/apps/evolution/mail/labels", GCONF_VALUE_STRING, list, NULL);
+ while (list) {
+ l = list->next;
+ g_free(list->data);
+ g_slist_free_1(list);
+ list = l;
+ }
+ } else {
+ g_warning("could not find /Mail/Labels in old config database, skipping");
+ }
+
+ /* Accounts: The flat bonobo-config structure is remapped to a list of xml blobs. Upgrades as necessary */
+ convert_xml_blob(gconf, config_doc, account_map, "/Mail/Accounts", "/apps/evolution/mail/accounts", "account", "uid");
+ /* Same for signatures */
+ convert_xml_blob(gconf, config_doc, signature_map, "/Mail/Signatures", "/apps/evolution/mail/signatures", "signature", NULL);
+
+ /* My-Evolution folder lists */
+ source = lookup_bconf_path(config_doc, "/My-Evolution/Mail");
+ if (source) {
+ char **t;
+
+ list = NULL;
+ l = NULL;
+
+ val = lookup_string(source, "display_folders-1.2", NULL);
+ /* FIXME: what needs upgrading here? Anything? */
+ if (val == NULL)
+ val = lookup_string(source, "display_folders", NULL);
+
+ if (val) {
+ t = g_strsplit (val, " !<-->! ", 8196);
+ for (i=0;t[i] && t[i+1];i+=2) {
+ list = g_slist_append(list, t[i]);
+ l = g_slist_append(l, t[i+1]);
+ }
+ if (list) {
+ gconf_client_set_list(gconf, "/apps/evolution/summary/mail/folder_evolution_uris", GCONF_VALUE_STRING, list, NULL);
+ g_slist_free(list);
+ }
+ if (l) {
+ gconf_client_set_list(gconf, "/apps/evolution/summary/mail/folder_physical_uris", GCONF_VALUE_STRING, l, NULL);
+ g_slist_free(l);
+ }
+ g_strfreev(t);
+ }
+ }
+
+ return 0;
+}
+
+
+static int load_accounts_1_0(xmlDocPtr doc)
+{
+ xmlNodePtr source;
+ char *val, *tmp;
+ int count = 0, i;
+ char key[32];
+
+ source = lookup_bconf_path(doc, "/Mail/Accounts");
+ if (source == NULL)
+ return 0;
+
+ val = lookup_bconf_value(source, "num");
+ if (val) {
+ count = atoi(val);
+ xmlFree(val);
+ }
+
+ /* load account upgrade info for each account */
+ for (i=0;i<count;i++) {
+ struct _account_info *ai;
+ char *rawuri;
+
+ sprintf(key, "source_url_%d", i);
+ rawuri = lookup_bconf_value(source, key);
+ if (rawuri == NULL)
+ continue;
+ ai = g_malloc0(sizeof(*ai));
+ ai->uri = hex_decode(rawuri);
+ ai->base_uri = get_base_uri(ai->uri);
+ sprintf(key, "account_name_%d", i);
+ ai->name = lookup_string(source, key, NULL);
+
+ d(printf("load account '%s'\n", ai->uri));
+
+ if (!strncmp(ai->uri, "imap:", 5)) {
+ read_imap_storeinfo(ai);
+ } else if (!strncmp(ai->uri, "exchange:", 9)) {
+ xmlNodePtr node;
+
+ d(printf(" upgrade exchange account\n"));
+ /* small hack, poke the source_url into the transport_url for exchanget: transports
+ - this will be picked up later in the conversion */
+ sprintf(key, "transport_url_%d", i);
+ node = lookup_bconf_entry(source, key);
+ if (node
+ && (val = xmlGetProp(node, "value"))) {
+ tmp = hex_decode(val);
+ xmlFree(val);
+ if (strncmp(tmp, "exchanget:", 10) == 0)
+ xmlSetProp(node, "value", rawuri);
+ g_free(tmp);
+ } else {
+ d(printf(" couldn't find transport uri?\n"));
+ }
+ }
+ xmlFree(rawuri);
+
+ g_hash_table_insert(accounts_1_0, ai->base_uri, ai);
+ if (ai->name)
+ g_hash_table_insert(accounts_name_1_0, ai->name, ai);
+ }
+
+ return 0;
+}
+
+/**
+ * e_config_upgrade:
+ * @edir:
+ *
+ * Upgrade evolution configuration from prior versions of evolution to
+ * the current one.
+ *
+ * No work is performed if the configuration version is up to date.
+ *
+ * The tracked version is upgraded to the latest even if no
+ * configuration upgrades are required for that version.
+ *
+ * Return value: -1 on an error.
+ **/
+int
+e_config_upgrade(const char *edir)
+{
+ xmlNodePtr source;
+ xmlDocPtr config_doc = NULL;
+ int i;
+ char *val, *tmp;
+ GConfClient *gconf;
+ int res = -1;
+ struct stat st;
+
+ evolution_dir = edir;
+
+ /* determine existing version */
+ gconf = gconf_client_get_default();
+ val = gconf_client_get_string(gconf, "/apps/evolution/version", NULL);
+ if (val) {
+ sscanf(val, "%u.%u.%u", &major, &minor, &revision);
+ g_free(val);
+ } else {
+ char *filename = g_build_filename(evolution_dir, "config.xmldb", NULL);
+
+ if (lstat(filename, &st) == 0
+ && S_ISREG(st.st_mode))
+ config_doc = xmlParseFile (filename);
+ g_free(filename);
+
+ tmp = NULL;
+ if ( config_doc
+ && (source = lookup_bconf_path(config_doc, "/Shell"))
+ && (tmp = lookup_bconf_value(source, "upgrade_from_1_0_to_1_2_performed"))
+ && tmp[0] == '1' ) {
+ major = 1;
+ minor = 2;
+ revision = 0;
+ } else {
+ major = 1;
+ minor = 0;
+ revision = 0;
+ }
+ if (tmp)
+ xmlFree(tmp);
+ }
+
+ d(printf("current config version is '%u.%u.%u'\n", major, minor, revision));
+
+ /* For 1.0.x we need to load the accounts first, as data it initialises is used elsewhere */
+ if (major <=1 && minor < 2) {
+ char *xml_files[] = { "vfolders.xml", "filters.xml", "shortcuts.xml" };
+
+ /* load in 1.0 info */
+ if (config_doc) {
+ accounts_1_0 = g_hash_table_new(g_str_hash, g_str_equal);
+ accounts_name_1_0 = g_hash_table_new(g_str_hash, g_str_equal);
+ load_accounts_1_0(config_doc);
+ }
+
+ /* upgrade xml uri fields */
+ for (i=0;i<sizeof(xml_files)/sizeof(xml_files[0]);i++) {
+ char *path;
+
+ path = g_build_filename(evolution_dir, xml_files[i], NULL);
+ if (upgrade_xml_file_1_0(path) == -1)
+ g_warning("Could not upgrade xml file %s", xml_files[i]);
+
+ g_free(path);
+ }
+ }
+
+ if (config_doc && major <=1 && minor < 3) {
+ /* move bonobo config to gconf */
+ if (import_bonobo_config(config_doc, gconf) == -1) {
+ g_warning("Could not move config from bonobo-conf to gconf");
+ goto error;
+ }
+ }
+
+ /* we're done, update our version info if its changed */
+ if (major < CONF_MAJOR
+ || minor < CONF_MINOR
+ || revision < CONF_REVISION) {
+ val = g_strdup_printf("%u.%u.%u", CONF_MAJOR, CONF_MINOR, CONF_REVISION);
+ gconf_client_set_string(gconf, "/apps/evolution/version", val, NULL);
+ /* TODO: should this be translatable? */
+ g_message("Evolution configuration upgraded to version: %s", val);
+ g_free(val);
+ gconf_client_suggest_sync(gconf, NULL);
+ }
+
+ res = 0;
+
+error:
+ if (config_doc)
+ xmlFreeDoc(config_doc);
+
+ return res;
+}