/* -*- 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., 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_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, (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) { 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 ((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; } 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); }