aboutsummaryrefslogblamecommitdiffstats
path: root/e-util/e-bconf-map.c
blob: 3e445260719a1661a429942db52700b2bca533e4 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
11
  
  



                                                                
  



                                                                    
  
                                                                   
                                                                             
  




                                                        


   



















                             
                            










                                                        

                                     
 

                                               
 
                                                  
                                                             

                                                          
 


                   

                                     
 

                                               
 










                                                                  
 




                   
                                                   

                        

                   
 
                             
                                                                 


                                                         
 

                              

                                                                                 






                                                               
 



                    
                                                      

                                         

                   
 
                      

                                                                                 






                                                               
 


                    

                                                      

                                                         
                                 
 
                                                                                   


                                      
 


                   

                                                     
 
                         
 




                                                                  
 


                   

                                                     
 
                         
 




                                                     
 


                   

                                                       
 
                         
 




                                                     
 




                      
                                                                                               
 

                                                                          



                                             

                                                                          



                                             

                                                                            



                                               

                                                                          
 

                          
 



                                                     
 



                                                             
 







                                                                                    

                                      

                                         
                      
 














                                                                          
 

                                   
 



                   
                                                                              
 
                            
                        
 

                                                                          
                                                                                 



                                                                                                     
 
                                                                   
 

                                                              
                                                                                                         
                                
                                                                                      
                         
 







                                       
    
                                                                                        

                                                                          

                          
                          
                         
                   
 






                                                                
 
                                                                       
 



                                             
                               
 

                                                                               
                                                         
 

                                                                       
                                              
 
                                                       
                                                                                    
                         
 
                                                         
 

                                                            
 

                                                             
 
                                                                                          
 








                                                                                               
 



                 
                                                                                                                           
 
    

                                                                                           
                                


                           

                      
 




                                                                                    
 





















                                                                                                                   
 



                                                             
 







                                                                                                                                    
 






                                                                                     
 




                                                               
 






















                                                                                               

                                                                            
 




                                                                               
 







                                                                                                    
 

                                                         

                                                                                     
 

                                                                                                                 




                                                                      
 

                                                          
 



                                                              
                                                                                             



                                                                  
 



                                                                 
                                                                                               



                                                                  
 


                                                              
                                                                                               



                                                                                                               
 









                                                                                                            
 





                                                                                 
 




                                                  
 

                 
/*
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) version 3.
 *
 * 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with the program; if not, see <http://www.gnu.org/licenses/>
 *
 *
 * Authors:
 *      Jeffrey Stedfast <fejj@ximian.com>
 *
 * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
 *
 */

#ifdef HAVE_CONFIG_H
#include <config.h>
#endif

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#include <glib.h>

#include <libxml/tree.h>
#include <libxml/parser.h>
#include <libxml/xmlmemory.h>

#include "e-bconf-map.h"


#define d(x)


static gchar 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,
};

gchar *
e_bconf_hex_decode (const gchar *val)
{
    const guchar *p = (const guchar *) val;
    gchar *o, *res;

    o = res = g_malloc (strlen (val) / 2 + 1);
    for (p = (const guchar *)val; (p[0] && p[1]); p += 2)
        *o++ = (hexnib[p[0]] << 4) | hexnib[p[1]];
    *o = 0;

    return res;
}

gchar *
e_bconf_url_decode (const gchar *val)
{
    const guchar *p = (const guchar *) val;
    gchar *o, *res, c;

    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;
}


xmlNodePtr
e_bconf_get_path (xmlDocPtr doc, const gchar *path)
{
    xmlNodePtr root;
    gchar *val;
    gint found;

    root = doc->children;
    if (strcmp ((gchar *)root->name, "bonobo-config") != 0) {
        g_warning ("not bonobo-config xml file");
        return NULL;
    }

    root = root->children;
    while (root) {
        if (!strcmp ((gchar *)root->name, "section")) {
            val = (gchar *)xmlGetProp (root, (const guchar *)"path");
            found = val && strcmp (val, path) == 0;
            xmlFree (val);
            if (found)
                break;
        }
        root = root->next;
    }

    return root;
}

xmlNodePtr
e_bconf_get_entry (xmlNodePtr root, const gchar *name)
{
    xmlNodePtr node = root->children;
    gint found;
    gchar *val;

    while (node) {
        if (!strcmp ((gchar *)node->name, "entry")) {
            val = (gchar *)xmlGetProp (node, (const guchar *)"name");
            found = val && strcmp (val, name) == 0;
            xmlFree (val);
            if (found)
                break;
        }
        node = node->next;
    }

    return node;
}

gchar *
e_bconf_get_value (xmlNodePtr root, const gchar *name)
{
    xmlNodePtr node = e_bconf_get_entry (root, name);
    gchar *prop, *val = NULL;

    if (node && (prop = (gchar *)xmlGetProp (node, (const guchar *)"value"))) {
        val = g_strdup (prop);
        xmlFree (prop);
    }

    return val;
}

gchar *
e_bconf_get_bool (xmlNodePtr root, const gchar *name)
{
    gchar *val, *res;

    if ((val = e_bconf_get_value (root, name))) {
        res = g_strdup (val[0] == '1' ? "true" : "false");
        g_free (val);
    } else
        res = NULL;

    return res;
}

gchar *
e_bconf_get_long (xmlNodePtr root, const gchar *name)
{
    gchar *val, *res;

    if ((val = e_bconf_get_value (root, name))) {
        res = g_strdup (val);
        g_free (val);
    } else
        res = NULL;

    return res;
}

gchar *
e_bconf_get_string (xmlNodePtr root, const gchar *name)
{
    gchar *val, *res;

    if ((val = e_bconf_get_value (root, name))) {
        res = e_bconf_hex_decode (val);
        g_free (val);
    } else
        res = NULL;

    return res;
}


/* lookup functions */
typedef gchar * (* bconf_lookup_func) (xmlNodePtr root, const gchar *name, e_bconf_map_t *nap);

static gchar *
bconf_lookup_bool (xmlNodePtr root, const gchar *name, e_bconf_map_t *map)
{
    return e_bconf_get_bool (root, name);
}

static gchar *
bconf_lookup_long (xmlNodePtr root, const gchar *name, e_bconf_map_t *map)
{
    return e_bconf_get_long (root, name);
}

static gchar *
bconf_lookup_string (xmlNodePtr root, const gchar *name, e_bconf_map_t *map)
{
    return e_bconf_get_string (root, name);
}

static gchar *
bconf_lookup_enum (xmlNodePtr root, const gchar *name, e_bconf_map_t *map)
{
    gint index = 0, i;
    gchar *val;

    if ((val = e_bconf_get_value (root, name))) {
        index = atoi (val);
        g_free (val);
    }

    for (i = 0; map->child[i].from; i++) {
        if (i == index)
            return g_strdup (map->child[i].from);
    }

    return NULL;
}


static bconf_lookup_func lookup_table[] = {
    bconf_lookup_bool, bconf_lookup_long, bconf_lookup_string, bconf_lookup_enum
};

static gchar *
get_name (const gchar *in, gint index)
{
    GString *out = g_string_new ("");
    gchar c, *res;

    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 void
build_xml (xmlNodePtr root, e_bconf_map_t *map, gint index, xmlNodePtr source)
{
    gchar *name, *value;
    xmlNodePtr node;

    while (map->type != E_BCONF_MAP_END) {
        if ((map->type & E_BCONF_MAP_MASK) == E_BCONF_MAP_CHILD) {
            node = xmlNewChild (root, NULL, (guchar *)map->to, NULL);
            build_xml (node, map->child, index, source);
        } else {
            name = get_name (map->from, index);
            value = lookup_table[(map->type & E_BCONF_MAP_MASK) - 1] (source, name, map);

            d(printf ("key '%s=%s' -> ", name, value));

            if (map->type & E_BCONF_MAP_CONTENT) {
                if (value && value[0])
                    xmlNewTextChild (root, NULL, (guchar *)map->to, (guchar *)value);
            } else {
                xmlSetProp (root, (guchar *)map->to, (guchar *)value);
            }

            g_free (value);
            g_free (name);
        }
        map++;
    }
}


gint
e_bconf_import_xml_blob (GConfClient *gconf, xmlDocPtr config_xmldb, e_bconf_map_t *map,
             const gchar *bconf_path, const gchar *gconf_path,
             const gchar *name, const gchar *idparam)
{
    xmlNodePtr source;
    gint count = 0, i;
    GSList *list, *l;
    gchar *val;

    source = e_bconf_get_path (config_xmldb, bconf_path);
    if (source) {
        list = NULL;
        if ((val = e_bconf_get_value (source, "num"))) {
            count = atoi (val);
            g_free (val);
        }

        d(printf("Found %d blobs at %s\n", count, bconf_path));

        for (i = 0; i < count; i++) {
            xmlDocPtr doc;
            xmlNodePtr root;
            xmlChar *xmlbuf;
            gint n;

            doc = xmlNewDoc ((const guchar *)"1.0");
            root = xmlNewDocNode (doc, NULL, (guchar *)name, NULL);
            xmlDocSetRootElement (doc, root);

            /* This could be set with a MAP_UID type ... */
            if (idparam) {
                gchar buf[16];

                sprintf (buf, "%d", i);
                xmlSetProp (root, (guchar *)idparam, (guchar *)buf);
            }

            build_xml (root, map, i, source);

            xmlDocDumpMemory (doc, &xmlbuf, &n);
            xmlFreeDoc (doc);

            list = g_slist_append (list, xmlbuf);
        }

        gconf_client_set_list (gconf, gconf_path, 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", bconf_path);
    }

    return 0;
}


static gint gconf_type[] = { GCONF_VALUE_BOOL, GCONF_VALUE_BOOL, GCONF_VALUE_INT, GCONF_VALUE_STRING, GCONF_VALUE_STRING };

gint
e_bconf_import (GConfClient *gconf, xmlDocPtr config_xmldb, e_gconf_map_list_t *remap_list)
{
    gchar *path, *val, *tmp;
    e_gconf_map_t *map;
    xmlNodePtr source;
    GSList *list, *l;
    gchar buf[32];
    gint i, j, k;

    /* process all flat config */
    for (i = 0; remap_list[i].root; i++) {
        d(printf ("Path: %s\n", remap_list[i].root));
        if (!(source = e_bconf_get_path (config_xmldb, remap_list[i].root)))
            continue;

        map = remap_list[i].map;
        for (j = 0; map[j].from; j++) {
            if (map[j].type & E_GCONF_MAP_LIST) {
                /* collapse a multi-entry indexed field into a list */
                list = NULL;
                k = 0;
                do {
                    path = get_name (map[j].from, k);
                    val = e_bconf_get_value (source, path);
                    d(printf ("finding path '%s' = '%s'\n", path, val));
                    g_free (path);
                    if (val) {
                        switch (map[j].type & E_GCONF_MAP_MASK) {
                        case E_GCONF_MAP_BOOL:
                        case E_GCONF_MAP_INT:
                            list = g_slist_append (list, GINT_TO_POINTER (atoi (val)));
                            break;
                        case E_GCONF_MAP_STRING:
                            d(printf (" -> '%s'\n", e_bconf_hex_decode (val)));
                            list = g_slist_append (list, e_bconf_hex_decode (val));
                            break;
                        }

                        g_free (val);
                        k++;
                    }
                } while (val);

                if (list) {
                    path = g_strdup_printf ("/apps/evolution/%s", map[j].to);
                    gconf_client_set_list (gconf, path, gconf_type[map[j].type & E_GCONF_MAP_MASK], list, NULL);
                    g_free (path);
                    if ((map[j].type & E_GCONF_MAP_MASK) == E_GCONF_MAP_STRING)
                        g_slist_foreach (list, (GFunc) g_free, NULL);
                    g_slist_free (list);
                }

                continue;
            } else if (map[j].type == E_GCONF_MAP_ANYLIST) {
                val = NULL;
            } else {
                if (!(val = e_bconf_get_value (source, map[j].from)))
                    continue;
            }

            d(printf (" %s = '%s' -> %s [%d]\n",
                  map[j].from,
                  val == NULL ? "(null)" : val,
                  map[j].to,
                  map[j].type));

            path = g_strdup_printf ("/apps/evolution/%s", map[j].to);
            switch (map[j].type) {
            case E_GCONF_MAP_BOOL:
                gconf_client_set_bool (gconf, path, atoi (val), NULL);
                break;
            case E_GCONF_MAP_BOOLNOT:
                gconf_client_set_bool (gconf, path, !atoi (val), NULL);
                break;
            case E_GCONF_MAP_INT:
                gconf_client_set_int (gconf, path, atoi (val), NULL);
                break;
            case E_GCONF_MAP_STRING:
                tmp = e_bconf_hex_decode (val);
                gconf_client_set_string (gconf, path, tmp, NULL);
                g_free (tmp);
                break;
            case E_GCONF_MAP_SIMPLESTRING:
                gconf_client_set_string (gconf, path, val, NULL);
                break;
            case E_GCONF_MAP_FLOAT:
                gconf_client_set_float (gconf, path, strtod (val, NULL), NULL);
                break;
            case E_GCONF_MAP_STRLIST: {
                gchar *v = e_bconf_hex_decode (val);
                gchar **t = g_strsplit (v, " !<-->!", 8196);

                list = NULL;
                for (k = 0; t[k]; k++) {
                    list = g_slist_append (list, t[k]);
                    d(printf ("  [%d] = '%s'\n", k, t[k]));
                }

                gconf_client_set_list (gconf, path, GCONF_VALUE_STRING, list, NULL);
                g_slist_free (list);
                g_strfreev (t);
                g_free (v);
                break; }
            case E_GCONF_MAP_ANYLIST: {
                xmlNodePtr node = source->children;
                list = NULL;

                /* find the entry node */
                while (node) {
                    if (!strcmp ((gchar *)node->name, "entry")) {
                        gint found;

                        if ((tmp = (gchar *)xmlGetProp (node, (const guchar *)"name"))) {
                            found = strcmp ((gchar *)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 ((gchar *)node->name, "any") == 0)
                            break;
                        node = node->next;
                    }
                }

                /* skip to the value inside it */
                if (node) {
                    node = node->children;
                    while (node) {
                        if (strcmp ((gchar *)node->name, "value") == 0)
                            break;
                        node = node->next;
                    }
                }

                if (node) {
                    node = node->children;
                    while (node) {
                        if (strcmp ((gchar *)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 E_GCONF_MAP_COLOUR:
                sprintf (buf, "#%06x", atoi (val) & 0xffffff);
                gconf_client_set_string (gconf, path, buf, NULL);
                break;
            }

            /* FIXME: handle errors */
            g_free (path);
            g_free (val);
        }
    }

    return 0;
}