/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/* Evolution Conduits - Pilot Map routines
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
* Authors: JP Rosevear <jpr@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., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include <string.h>
#include <stdlib.h>
#include <glib.h>
#include <libxml/parser.h>
#include <libedataserver/e-xml-utils.h>
#include "e-pilot-map.h"
typedef struct
{
char *uid;
gboolean archived;
gboolean touched;
} EPilotMapPidNode;
typedef struct
{
guint32 pid;
gboolean archived;
gboolean touched;
} EPilotMapUidNode;
typedef struct
{
gboolean touched_only;
xmlNodePtr root;
} EPilotMapWriteData;
static void
real_e_pilot_map_insert (EPilotMap *map, guint32 pid, const char *uid, gboolean archived, gboolean touch)
{
char *new_uid;
guint32 *new_pid;
EPilotMapPidNode *pnode;
EPilotMapUidNode *unode;
g_return_if_fail (map != NULL);
g_return_if_fail (uid != NULL);
/* Keys */
if (pid != 0) {
new_pid = g_new (guint32, 1);
*new_pid = pid;
}
new_uid = g_strdup (uid);
/* Values */
if (pid != 0) {
pnode = g_new0 (EPilotMapPidNode, 1);
pnode->uid = new_uid;
pnode->archived = archived;
if (touch)
pnode->touched = TRUE;
}
unode = g_new0 (EPilotMapUidNode, 1);
unode->pid = pid;
unode->archived = archived;
if (touch)
unode->touched = TRUE;
/* Insertion */
if (pid != 0)
g_hash_table_insert (map->pid_map, new_pid, pnode);
g_hash_table_insert (map->uid_map, new_uid, unode);
}
static void
map_set_node_timet (xmlNodePtr node, const char *name, time_t t)
{
char *tstring;
tstring = g_strdup_printf ("%ld", t);
xmlSetProp (node, (unsigned char *)name, (unsigned char *)tstring);
g_free (tstring);
}
static void
map_sax_start_element (void *data, const xmlChar *name,
const xmlChar **attrs)
{
EPilotMap *map = (EPilotMap *)data;
if (!strcmp ((char *)name, "PilotMap")) {
while (attrs && *attrs != NULL) {
const xmlChar **val = attrs;
val++;
if (!strcmp ((char *)*attrs, "timestamp"))
map->since = (time_t)strtoul ((char *)*val, NULL, 0);
attrs = ++val;
}
}
if (!strcmp ((char *)name, "map")) {
const char *uid = NULL;
guint32 pid = 0;
gboolean archived = FALSE;
while (attrs && *attrs != NULL) {
const xmlChar **val = attrs;
val++;
if (!strcmp ((char *)*attrs, "uid"))
uid = (char *)*val;
if (!strcmp ((char *)*attrs, "pilot_id"))
pid = strtoul ((char *)*val, NULL, 0);
if (!strcmp ((char *)*attrs, "archived"))
archived = strtoul ((char *)*val, NULL, 0)== 1 ? TRUE : FALSE;
attrs = ++val;
}
g_return_if_fail (uid != NULL);
g_return_if_fail (pid != 0 || archived);
real_e_pilot_map_insert (map, pid, uid, archived, FALSE);
}
}
static void
map_write_foreach (gpointer key, gpointer value, gpointer data)
{
EPilotMapWriteData *wd = data;
xmlNodePtr root = wd->root;
char *uid = key;
EPilotMapUidNode *unode = value;
xmlNodePtr mnode;
if (wd->touched_only && !unode->touched)
return;
mnode = xmlNewChild (root, NULL, (const unsigned char *)"map", NULL);
xmlSetProp (mnode, (const unsigned char *)"uid", (unsigned char *)uid);
if (unode->archived) {
xmlSetProp (mnode, (const unsigned char *)"archived", (const unsigned char *)"1");
} else {
char *pidstr;
pidstr = g_strdup_printf ("%d", unode->pid);
xmlSetProp (mnode, (const unsigned char *)"pilot_id", (unsigned char *)pidstr);
g_free (pidstr);
xmlSetProp (mnode, (const unsigned char *)"archived", (const unsigned char *)"0");
}
}
gboolean
e_pilot_map_pid_is_archived (EPilotMap *map, guint32 pid)
{
EPilotMapPidNode *pnode;
g_return_val_if_fail (map != NULL, FALSE);
pnode = g_hash_table_lookup (map->pid_map, &pid);
if (pnode == NULL)
return FALSE;
return pnode->archived;
}
gboolean
e_pilot_map_uid_is_archived (EPilotMap *map, const char *uid)
{
EPilotMapUidNode *unode;
g_return_val_if_fail (map != NULL, FALSE);
g_return_val_if_fail (uid != NULL, FALSE);
unode = g_hash_table_lookup (map->uid_map, uid);
if (unode == NULL)
return FALSE;
return unode->archived;
}
void
e_pilot_map_insert (EPilotMap *map, guint32 pid, const char *uid, gboolean archived)
{
EPilotMapPidNode *pnode;
EPilotMapUidNode *unode;
pnode = g_hash_table_lookup (map->pid_map, &pid);
if (pnode != NULL) {
/* In case the pid<->uid mapping is not the same anymore */
g_hash_table_remove (map->uid_map, pnode->uid);
g_hash_table_remove (map->pid_map, &pid);
}
unode = g_hash_table_lookup (map->uid_map, uid);
if (unode != NULL) {
/* In case the pid<->uid mapping is not the same anymore */
g_hash_table_remove (map->pid_map, &unode->pid);
g_hash_table_remove (map->uid_map, uid);
}
real_e_pilot_map_insert (map, pid, uid, archived, TRUE);
}
void
e_pilot_map_remove_by_pid (EPilotMap *map, guint32 pid)
{
EPilotMapPidNode *pnode;
EPilotMapUidNode *unode;
g_return_if_fail (map != NULL);
pnode = g_hash_table_lookup (map->pid_map, &pid);
if (pnode == NULL)
return;
unode = g_hash_table_lookup (map->uid_map, pnode->uid);
g_return_if_fail (unode != NULL);
g_hash_table_remove (map->uid_map, pnode->uid);
g_hash_table_remove (map->pid_map, &pid);
}
void
e_pilot_map_remove_by_uid (EPilotMap *map, const char *uid)
{
EPilotMapPidNode *pnode;
EPilotMapUidNode *unode;
g_return_if_fail (map != NULL);
g_return_if_fail (uid != NULL);
unode = g_hash_table_lookup (map->uid_map, uid);
if (unode == NULL)
return;
pnode = g_hash_table_lookup (map->pid_map, &unode->pid);
g_hash_table_remove (map->pid_map, &unode->pid);
g_hash_table_remove (map->uid_map, uid);
}
guint32
e_pilot_map_lookup_pid (EPilotMap *map, const char *uid, gboolean touch)
{
EPilotMapUidNode *unode = NULL;
g_return_val_if_fail (map != NULL, 0);
g_return_val_if_fail (uid != NULL, 0);
unode = g_hash_table_lookup (map->uid_map, uid);
if (unode == NULL)
return 0;
if (touch) {
EPilotMapPidNode *pnode = NULL;
pnode = g_hash_table_lookup (map->pid_map, &unode->pid);
if (pnode != NULL)
pnode->touched = TRUE;
unode->touched = TRUE;
}
return unode->pid;
}
const char *
e_pilot_map_lookup_uid (EPilotMap *map, guint32 pid, gboolean touch)
{
EPilotMapPidNode *pnode = NULL;
g_return_val_if_fail (map != NULL, NULL);
pnode = g_hash_table_lookup (map->pid_map, &pid);
if (pnode == NULL)
return NULL;
if (touch) {
EPilotMapUidNode *unode = NULL;
unode = g_hash_table_lookup (map->uid_map, pnode->uid);
g_return_val_if_fail (unode != NULL, NULL);
unode->touched = TRUE;
pnode->touched = TRUE;
}
return pnode->uid;
}
int
e_pilot_map_read (const char *filename, EPilotMap **map)
{
xmlSAXHandler handler;
EPilotMap *new_map;
g_return_val_if_fail (filename != NULL, -1);
g_return_val_if_fail (map != NULL, -1);
*map = NULL;
new_map = g_new0 (EPilotMap, 1);
memset (&handler, 0, sizeof (xmlSAXHandler));
handler.startElement = map_sax_start_element;
new_map->pid_map = g_hash_table_new_full (
g_int_hash, g_int_equal,
(GDestroyNotify) g_free,
(GDestroyNotify) g_free);
new_map->uid_map = g_hash_table_new_full (
g_str_hash, g_str_equal,
(GDestroyNotify) g_free,
(GDestroyNotify) g_free);
if (g_file_test (filename, G_FILE_TEST_EXISTS)) {
if (xmlSAXUserParseFile (&handler, new_map, filename) < 0) {
g_free (new_map);
return -1;
}
}
new_map->write_touched_only = FALSE;
*map = new_map;
return 0;
}
int
e_pilot_map_write (const char *filename, EPilotMap *map)
{
EPilotMapWriteData wd;
xmlDocPtr doc;
int ret;
g_return_val_if_fail (filename != NULL, -1);
g_return_val_if_fail (map != NULL, -1);
doc = xmlNewDoc ((const unsigned char *)"1.0");
if (doc == NULL) {
g_warning ("Pilot map file could not be created\n");
return -1;
}
xmlDocSetRootElement (doc, xmlNewDocNode(doc, NULL, (const unsigned char *)"PilotMap", NULL));
map->since = time (NULL);
map_set_node_timet (xmlDocGetRootElement (doc), "timestamp", map->since);
wd.touched_only = map->write_touched_only;
wd.root = xmlDocGetRootElement(doc);
g_hash_table_foreach (map->uid_map, map_write_foreach, &wd);
/* Write the file */
xmlSetDocCompressMode (doc, 0);
ret = e_xml_save_file (filename, doc);
if (ret < 0) {
g_warning ("Pilot map file '%s' could not be saved\n", filename);
return -1;
}
xmlFreeDoc (doc);
return 0;
}
void
e_pilot_map_clear (EPilotMap *map)
{
g_return_if_fail (map != NULL);
g_hash_table_remove_all (map->pid_map);
g_hash_table_remove_all (map->uid_map);
map->since = 0;
map->write_touched_only = FALSE;
}
void
e_pilot_map_destroy (EPilotMap *map)
{
g_return_if_fail (map != NULL);
g_hash_table_destroy (map->pid_map);
g_hash_table_destroy (map->uid_map);
g_free (map);
}