2003-01-25 Chris Toshok <toshok@ximian.com> * e-pilot-map.c (e_pilot_map_read): g_file_exists -> g_file_test. (e_pilot_map_write): don't use doc->root. svn path=/trunk/; revision=19631
449 lines
9.7 KiB
C
449 lines
9.7 KiB
C
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
|
|
/* Evolution Conduits - Pilot Map routines
|
|
*
|
|
* Copyright (C) 2000 Ximian, Inc.
|
|
*
|
|
* 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., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include "e-pilot-map.h"
|
|
|
|
#include <string.h> /* memset(), strcmp() */
|
|
#include <stdlib.h>
|
|
#include <glib.h>
|
|
#include <libxml/parser.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, name, tstring);
|
|
g_free (tstring);
|
|
}
|
|
|
|
static void
|
|
map_sax_start_element (void *data, const xmlChar *name,
|
|
const xmlChar **attrs)
|
|
{
|
|
EPilotMap *map = (EPilotMap *)data;
|
|
|
|
if (!strcmp (name, "PilotMap")) {
|
|
while (attrs && *attrs != NULL) {
|
|
const xmlChar **val = attrs;
|
|
|
|
val++;
|
|
if (!strcmp (*attrs, "timestamp"))
|
|
map->since = (time_t)strtoul (*val, NULL, 0);
|
|
|
|
attrs = ++val;
|
|
}
|
|
}
|
|
|
|
if (!strcmp (name, "map")) {
|
|
const char *uid = NULL;
|
|
guint32 pid = 0;
|
|
gboolean archived = FALSE;
|
|
|
|
while (attrs && *attrs != NULL) {
|
|
const xmlChar **val = attrs;
|
|
|
|
val++;
|
|
if (!strcmp (*attrs, "uid"))
|
|
uid = *val;
|
|
|
|
if (!strcmp (*attrs, "pilot_id"))
|
|
pid = strtoul (*val, NULL, 0);
|
|
|
|
if (!strcmp (*attrs, "archived"))
|
|
archived = strtoul (*val, NULL, 0)== 1 ? TRUE : FALSE;
|
|
|
|
attrs = ++val;
|
|
}
|
|
|
|
g_assert (uid != NULL);
|
|
g_assert (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, "map", NULL);
|
|
xmlSetProp (mnode, "uid", uid);
|
|
|
|
if (unode->archived) {
|
|
xmlSetProp (mnode, "archived", "1");
|
|
} else {
|
|
char *pidstr;
|
|
|
|
pidstr = g_strdup_printf ("%d", unode->pid);
|
|
xmlSetProp (mnode, "pilot_id", pidstr);
|
|
g_free (pidstr);
|
|
xmlSetProp (mnode, "archived", "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)
|
|
{
|
|
gpointer key, value;
|
|
|
|
if (g_hash_table_lookup_extended (map->pid_map, &pid, &key, &value)) {
|
|
EPilotMapPidNode *pnode = value;
|
|
gpointer other_key, other_value;
|
|
|
|
g_hash_table_remove (map->pid_map, &pid);
|
|
|
|
/* In case the pid<->uid mapping is not the same anymore */
|
|
if (g_hash_table_lookup_extended (map->uid_map, pnode->uid, &other_key, &other_value)) {
|
|
g_hash_table_remove (map->uid_map, pnode->uid);
|
|
g_free (other_key);
|
|
g_free (other_value);
|
|
}
|
|
|
|
g_free (key);
|
|
g_free (value);
|
|
}
|
|
if (g_hash_table_lookup_extended (map->uid_map, uid, &key, &value)) {
|
|
EPilotMapUidNode *unode = value;
|
|
gpointer other_key, other_value;
|
|
|
|
g_hash_table_remove (map->uid_map, uid);
|
|
|
|
/* In case the pid<->uid mapping is not the same anymore */
|
|
if (g_hash_table_lookup_extended (map->pid_map, &unode->pid, &other_key, &other_value)) {
|
|
g_hash_table_remove (map->pid_map, &unode->pid);
|
|
g_free (other_key);
|
|
g_free (other_value);
|
|
}
|
|
|
|
g_free (key);
|
|
g_free (value);
|
|
}
|
|
|
|
real_e_pilot_map_insert (map, pid, uid, archived, TRUE);
|
|
}
|
|
|
|
void
|
|
e_pilot_map_remove_by_pid (EPilotMap *map, guint32 pid)
|
|
{
|
|
EPilotMapPidNode *pnode = NULL;
|
|
EPilotMapUidNode *unode = NULL;
|
|
gpointer pkey, ukey;
|
|
|
|
g_return_if_fail (map != NULL);
|
|
|
|
if (!g_hash_table_lookup_extended (map->pid_map, &pid,
|
|
&pkey, (gpointer *)&pnode))
|
|
return;
|
|
|
|
g_hash_table_lookup_extended (map->uid_map, pnode->uid, &ukey,
|
|
(gpointer *)&unode);
|
|
g_assert (unode != NULL);
|
|
|
|
g_hash_table_remove (map->pid_map, &pid);
|
|
g_hash_table_remove (map->uid_map, pnode->uid);
|
|
|
|
g_free (pkey);
|
|
g_free (ukey);
|
|
g_free (pnode);
|
|
g_free (unode);
|
|
}
|
|
|
|
void
|
|
e_pilot_map_remove_by_uid (EPilotMap *map, const char *uid)
|
|
{
|
|
EPilotMapPidNode *pnode = NULL;
|
|
EPilotMapUidNode *unode = NULL;
|
|
gpointer pkey, ukey;
|
|
|
|
g_return_if_fail (map != NULL);
|
|
g_return_if_fail (uid != NULL);
|
|
|
|
if (!g_hash_table_lookup_extended (map->uid_map, uid, &ukey, (gpointer *)&unode))
|
|
return;
|
|
|
|
g_hash_table_lookup_extended (map->pid_map, &unode->pid, &pkey, (gpointer *)&pnode);
|
|
|
|
g_hash_table_remove (map->uid_map, uid);
|
|
g_hash_table_remove (map->pid_map, &unode->pid);
|
|
|
|
if (unode->pid != 0)
|
|
g_free (pkey);
|
|
g_free (ukey);
|
|
|
|
if (unode->pid != 0)
|
|
g_free (pnode);
|
|
g_free (unode);
|
|
}
|
|
|
|
|
|
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_assert (unode != 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 (g_int_hash, g_int_equal);
|
|
new_map->uid_map = g_hash_table_new (g_str_hash, g_str_equal);
|
|
|
|
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 ("1.0");
|
|
if (doc == NULL) {
|
|
g_warning ("Pilot map file could not be created\n");
|
|
return -1;
|
|
}
|
|
xmlDocSetRootElement (doc, xmlNewDocNode(doc, NULL, "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 = xmlSaveFile (filename, doc);
|
|
if (ret < 0) {
|
|
g_warning ("Pilot map file '%s' could not be saved\n", filename);
|
|
return -1;
|
|
}
|
|
|
|
xmlFreeDoc (doc);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static gboolean
|
|
foreach_remove (gpointer key, gpointer value, gpointer data)
|
|
{
|
|
g_free (key);
|
|
g_free (value);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
e_pilot_map_clear (EPilotMap *map)
|
|
{
|
|
g_return_if_fail (map != NULL);
|
|
|
|
g_hash_table_foreach_remove (map->pid_map, foreach_remove, NULL);
|
|
g_hash_table_foreach_remove (map->uid_map, foreach_remove, NULL);
|
|
|
|
map->since = 0;
|
|
map->write_touched_only = FALSE;
|
|
}
|
|
|
|
void
|
|
e_pilot_map_destroy (EPilotMap *map)
|
|
{
|
|
g_return_if_fail (map != NULL);
|
|
|
|
g_hash_table_foreach_remove (map->pid_map, foreach_remove, NULL);
|
|
g_hash_table_foreach_remove (map->uid_map, foreach_remove, NULL);
|
|
|
|
g_hash_table_destroy (map->pid_map);
|
|
g_hash_table_destroy (map->uid_map);
|
|
g_free (map);
|
|
}
|