/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Authors: Jeffrey Stedfast <fejj@ximian.com> * * Copyright 2004 Ximian, Inc. (www.ximian.com) * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * 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. * */ #ifdef HAVE_CONFIG_H #include <config.h> #endif #include <string.h> #include <libxml/tree.h> #include <libxml/parser.h> #include <libxml/xmlmemory.h> #include <gconf/gconf-client.h> #include <libedataserver/e-uid.h> #include "e-signature.h" static void e_signature_class_init (ESignatureClass *klass); static void e_signature_init (ESignature *sig, ESignatureClass *klass); static void e_signature_finalize (GObject *object); static GObjectClass *parent_class = NULL; GType e_signature_get_type (void) { static GType type = 0; if (!type) { GTypeInfo type_info = { sizeof (ESignatureClass), NULL, NULL, (GClassInitFunc) e_signature_class_init, NULL, NULL, sizeof (ESignature), 0, (GInstanceInitFunc) e_signature_init, }; type = g_type_register_static (G_TYPE_OBJECT, "ESignature", &type_info, 0); } return type; } static void e_signature_class_init (ESignatureClass *klass) { GObjectClass *object_class = (GObjectClass *) klass; parent_class = g_type_class_ref (G_TYPE_OBJECT); /* virtual method override */ object_class->finalize = e_signature_finalize; } static void e_signature_init (ESignature *sig, ESignatureClass *klass) { ; } static void e_signature_finalize (GObject *object) { ESignature *sig = (ESignature *) object; g_free (sig->uid); g_free (sig->name); g_free (sig->filename); G_OBJECT_CLASS (parent_class)->finalize (object); } /** * e_signature_new: * * Returns a new signature which can be filled in and * added to an #ESignatureList. **/ ESignature * e_signature_new (void) { ESignature *signature; signature = g_object_new (E_TYPE_SIGNATURE, NULL); signature->uid = e_uid_new (); return signature; } /** * e_signature_new_from_xml: * @xml: an XML signature description * * Return value: a new #ESignature based on the data in @xml, or %NULL * if @xml could not be parsed as valid signature data. **/ ESignature * e_signature_new_from_xml (const char *xml) { ESignature *signature; signature = g_object_new (E_TYPE_SIGNATURE, NULL); if (!e_signature_set_from_xml (signature, xml)) { g_object_unref (signature); return NULL; } return signature; } static gboolean xml_set_bool (xmlNodePtr node, const char *name, gboolean *val) { gboolean bool; char *buf; if ((buf = (char *)xmlGetProp (node, (const unsigned char *)name))) { bool = (!strcmp (buf, "true") || !strcmp (buf, "yes")); xmlFree (buf); if (bool != *val) { *val = bool; return TRUE; } } return FALSE; } static gboolean xml_set_prop (xmlNodePtr node, const char *name, char **val) { char *buf, *new_val; buf = (char *)xmlGetProp (node, (const unsigned char *)name); new_val = g_strdup (buf); xmlFree (buf); /* We can use strcmp here whether the value is UTF8 or * not, since we only care if the bytes changed. */ if (!*val || strcmp (*val, new_val)) { g_free (*val); *val = new_val; return TRUE; } else { g_free (new_val); return FALSE; } } static gboolean xml_set_content (xmlNodePtr node, char **val) { char *buf, *new_val; buf = (char *)xmlNodeGetContent (node); new_val = g_strdup (buf); xmlFree (buf); /* We can use strcmp here whether the value is UTF8 or * not, since we only care if the bytes changed. */ if (!*val || strcmp (*val, new_val)) { g_free (*val); *val = new_val; return TRUE; } else { g_free (new_val); return FALSE; } } /** * e_signature_uid_from_xml: * @xml: an XML signature description * * Return value: the permanent UID of the signature described by @xml * (or %NULL if @xml could not be parsed or did not contain a uid). * The caller must free this string. **/ char * e_signature_uid_from_xml (const char *xml) { xmlNodePtr node; xmlDocPtr doc; char *uid = NULL; if (!(doc = xmlParseDoc ((unsigned char *) xml))) return NULL; node = doc->children; if (strcmp ((char *)node->name, "signature") != 0) { xmlFreeDoc (doc); return NULL; } xml_set_prop (node, "uid", &uid); xmlFreeDoc (doc); return uid; } /** * e_signature_set_from_xml: * @signature: an #ESignature * @xml: an XML signature description. * * Changes @signature to match @xml. * * Returns %TRUE if the signature was loaded or %FALSE otherwise. **/ gboolean e_signature_set_from_xml (ESignature *signature, const char *xml) { gboolean changed = FALSE; xmlNodePtr node, cur; xmlDocPtr doc; gboolean bool; char *buf; if (!(doc = xmlParseDoc ((unsigned char *) xml))) return FALSE; node = doc->children; if (strcmp ((char *)node->name, "signature") != 0) { xmlFreeDoc (doc); return FALSE; } if (!signature->uid) xml_set_prop (node, "uid", &signature->uid); changed |= xml_set_prop (node, "name", &signature->name); changed |= xml_set_bool (node, "auto", &signature->autogen); if (signature->autogen) { /* we're done */ g_free (signature->filename); signature->filename = NULL; signature->script = FALSE; signature->html = FALSE; xmlFreeDoc (doc); return changed; } buf = NULL; xml_set_prop (node, "format", &buf); if (buf && !strcmp (buf, "text/html")) bool = TRUE; else bool = FALSE; g_free (buf); if (signature->html != bool) { signature->html = bool; changed = TRUE; } cur = node->children; while (cur) { if (!strcmp ((char *)cur->name, "filename")) { changed |= xml_set_content (cur, &signature->filename); changed |= xml_set_bool (cur, "script", &signature->script); break; } else if (!strcmp ((char *)cur->name, "script")) { /* this is for handling 1.4 signature script definitions */ changed |= xml_set_content (cur, &signature->filename); if (!signature->script) { signature->script = TRUE; changed = TRUE; } break; } cur = cur->next; } xmlFreeDoc (doc); return changed; } /** * e_signature_to_xml: * @signature: an #ESignature * * Return value: an XML representation of @signature, which the caller * must free. **/ char * e_signature_to_xml (ESignature *signature) { xmlChar *xmlbuf; char *tmp; xmlNodePtr root, node; xmlDocPtr doc; int n; doc = xmlNewDoc ((const unsigned char *)"1.0"); root = xmlNewDocNode (doc, NULL, (const unsigned char *)"signature", NULL); xmlDocSetRootElement (doc, root); xmlSetProp (root, (const unsigned char *)"name", (unsigned char *)signature->name); xmlSetProp (root, (const unsigned char *)"uid", (unsigned char *)signature->uid); xmlSetProp (root, (const unsigned char *)"auto", (const unsigned char *)(signature->autogen ? "true" : "false")); if (!signature->autogen) { xmlSetProp (root, (const unsigned char *)"format", (const unsigned char *)(signature->html ? "text/html" : "text/plain")); if (signature->filename) { node = xmlNewTextChild (root, NULL, (const unsigned char *)"filename", (unsigned char *)signature->filename); if (signature->script) xmlSetProp (node, (const unsigned char *)"script", (const unsigned char *)"true"); } } else { /* this is to make Evolution-1.4 and older 1.5 versions happy */ xmlSetProp (root, (const unsigned char *)"format", (const unsigned char *)"text/html"); } xmlDocDumpMemory (doc, &xmlbuf, &n); xmlFreeDoc (doc); /* remap to glib memory */ tmp = g_malloc (n + 1); memcpy (tmp, xmlbuf, n); tmp[n] = '\0'; xmlFree (xmlbuf); return tmp; }