aboutsummaryrefslogtreecommitdiffstats
path: root/plugins/bbdb/gaimbuddies.c
diff options
context:
space:
mode:
Diffstat (limited to 'plugins/bbdb/gaimbuddies.c')
-rw-r--r--plugins/bbdb/gaimbuddies.c459
1 files changed, 459 insertions, 0 deletions
diff --git a/plugins/bbdb/gaimbuddies.c b/plugins/bbdb/gaimbuddies.c
new file mode 100644
index 0000000000..a9bd81bc9b
--- /dev/null
+++ b/plugins/bbdb/gaimbuddies.c
@@ -0,0 +1,459 @@
+/*
+ * Routines to copy information from a Gaim buddy list into an
+ * Evolution addressbook.
+ *
+ * I currently copy IM account names and buddy icons, provided you
+ * don't already have a buddy icon defined for a person.
+ *
+ * This works today (25 October 2004), but is pretty sure to break
+ * later on as the Gaim buddylist file format shifts.
+ *
+ * Nat Friedman <nat@novell.com>
+ *
+ * Copyright 2004 Novell, Inc.
+ *
+ * Permission is hereby granted, free of charge, to any person
+ * obtaining a copy of this software and associated documentation
+ * files (the "Software"), to deal in the Software without
+ * restriction, including without limitation the rights to use, copy,
+ * modify, merge, publish, distribute, sublicense, and/or sell copies
+ * of the Software, and to permit persons to whom the Software is
+ * furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice shall be
+ * included in all copies or substantial portions of the Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+ * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+ * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+ * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
+ * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
+ * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
+ * DEALINGS IN THE SOFTWARE.
+ */
+
+#include <glib.h>
+#include <libxml/tree.h>
+#include <libxml/parser.h>
+#include <libxml/xmlmemory.h>
+#include <gal/util/e-xml-utils.h>
+
+#include <gtk/gtk.h>
+#include <libgnome/gnome-i18n.h>
+#include <string.h>
+
+#include <libebook/e-book.h>
+#include <libedataserverui/e-source-option-menu.h>
+
+#include <sys/time.h>
+#include <sys/stat.h>
+
+#include <e-util/e-config.h>
+
+#include "bbdb.h"
+
+typedef struct {
+ char *account_name;
+ char *proto;
+ char *alias;
+ char *icon;
+} GaimBuddy;
+
+/* Defined in bbdb.c */
+EBook *bbdb_open_addressbook (void);
+
+/* Forward declarations for this file. */
+void bbdb_sync_buddy_list (void);
+static EBookQuery *e_book_query_im_field_contains (const char *im);
+static void bbdb_merge_buddy_to_contact (EBook *book, GaimBuddy *b, EContact *c);
+static GList *bbdb_get_gaim_buddy_list (void);
+static char *get_node_text (xmlNodePtr node);
+static char *get_buddy_icon_from_setting (xmlNodePtr setting);
+static char *get_node_text (xmlNodePtr node);
+static void free_contact_list (GList *contacts);
+static void free_buddy_list (GList *blist);
+static void parse_buddy_group (xmlNodePtr group, GList **buddies);
+static EContactField proto_to_contact_field (const char *proto);
+
+void
+bbdb_sync_buddy_list_check (void)
+{
+ GConfClient *gconf;
+ struct stat statbuf;
+ time_t last_sync;
+ char *blist_path;
+ char *last_sync_str;
+
+ gconf = gconf_client_get_default ();
+
+ if (! gconf_client_get_bool (gconf, GCONF_KEY_ENABLE_GAIM, NULL)) {
+ g_object_unref (G_OBJECT (gconf));
+ return;
+ }
+
+ blist_path = g_build_path ("/", getenv ("HOME"), ".gaim/blist.xml", NULL);
+ if (stat (blist_path, &statbuf) < 0) {
+ g_object_unref (G_OBJECT (gconf));
+ return;
+ }
+
+ /* Reprocess the buddy list if it's been updated. */
+ last_sync_str = gconf_client_get_string (gconf, GCONF_KEY_GAIM_LAST_SYNC, NULL);
+ if (last_sync_str == NULL || ! strcmp (last_sync_str, ""))
+ last_sync = (time_t) 0;
+ else
+ last_sync = (time_t) g_ascii_strtoull (last_sync_str, NULL, 10);
+
+ g_free (last_sync_str);
+ g_object_unref (G_OBJECT (gconf));
+
+ printf ("bbdb: Last sync: %ld\n", last_sync);
+ printf ("bbdb: Modified: %ld\n", statbuf.st_mtime);
+
+ if (statbuf.st_mtime > last_sync) {
+ fprintf (stderr, "bbdb: Buddy list dirty!\n");
+
+ bbdb_sync_buddy_list ();
+ }
+}
+
+void
+bbdb_sync_buddy_list (void)
+{
+ GList *blist, *l;
+ EBook *book = NULL;
+
+ /* Get the Gaim buddy list */
+ blist = bbdb_get_gaim_buddy_list ();
+ if (blist == NULL)
+ return;
+
+ /* Open the addressbook */
+ book = bbdb_open_addressbook ();
+ if (book == NULL) {
+ free_buddy_list (blist);
+ return;
+ }
+
+ /* Walk the buddy list */
+ for (l = blist; l != NULL; l = l->next) {
+ GaimBuddy *b = l->data;
+ EBookQuery *query;
+ GList *contacts;
+ GError *error;
+ EContact *c;
+
+ if (b->alias == NULL || strlen (b->alias) == 0)
+ continue;
+
+ /* Check to see if the buddy is already in the addressbook */
+ query = e_book_query_im_field_contains (b->account_name);
+ e_book_get_contacts (book, query, &contacts, NULL);
+ e_book_query_unref (query);
+ if (contacts != NULL) {
+ free_contact_list (contacts);
+ continue;
+ }
+ free_contact_list (contacts);
+
+ /* Look for an exact match full name == buddy alias */
+ query = e_book_query_field_test (E_CONTACT_FULL_NAME, E_BOOK_QUERY_IS, b->alias);
+ e_book_get_contacts (book, query, &contacts, NULL);
+ e_book_query_unref (query);
+ if (contacts != NULL) {
+
+ /* FIXME: If there's more than one contact with this
+ name, just give up; we're not smart enough for
+ this. */
+ if (contacts->next != NULL)
+ continue;
+
+ c = E_CONTACT (contacts->data);
+
+ bbdb_merge_buddy_to_contact (book, b, c);
+
+ /* Write it out to the addressbook */
+ if (! e_book_commit_contact (book, c, &error)) {
+ g_warning ("bbdb: Could not modify contact: %s\n", error->message);
+ g_error_free (error);
+ }
+
+ continue;
+ }
+
+ /* Otherwise, create a new contact. */
+ c = e_contact_new ();
+ e_contact_set (c, E_CONTACT_FULL_NAME, (gpointer) b->alias);
+ bbdb_merge_buddy_to_contact (book, b, c);
+ if (! e_book_add_contact (book, c, &error)) {
+ g_warning ("bbdb: Failed to add new contact: %s\n", error->message);
+ g_error_free (error);
+ return;
+ }
+ g_object_unref (G_OBJECT (c));
+ }
+
+
+ /* Update the last-sync'd time */
+ {
+ GConfClient *gconf;
+ time_t last_sync;
+ char *last_sync_str;
+
+ gconf = gconf_client_get_default ();
+
+ time (&last_sync);
+ last_sync_str = g_strdup_printf ("%ld", (glong) last_sync);
+ printf ("Str: %s\n", last_sync_str);
+ gconf_client_set_string (gconf, GCONF_KEY_GAIM_LAST_SYNC, last_sync_str, NULL);
+ g_free (last_sync_str);
+
+ g_object_unref (G_OBJECT (gconf));
+ }
+}
+
+static EBookQuery *
+e_book_query_im_field_contains (const char *im)
+{
+ char *query_string;
+ EBookQuery *query;
+
+ query_string = g_strdup_printf (
+ "(or "
+ "(is \"im_aim\" \"%s\") "
+ "(is \"im_yahoo\" \"%s\") "
+ "(is \"im_msn\" \"%s\") "
+ "(is \"im_icq\" \"%s\") "
+ "(is \"im_jabber\" \"%s\") "
+ "(is \"im_groupwise\" \"%s\")"
+ ")",
+ im, im, im, im, im, im);
+
+ query = e_book_query_from_string (query_string);
+
+ g_free (query_string);
+
+ return query;
+}
+
+static void
+bbdb_merge_buddy_to_contact (EBook *book, GaimBuddy *b, EContact *c)
+{
+ EContactField field;
+ GList *ims, *l;
+
+ EContactPhoto *photo = NULL;
+
+ GError *error = NULL;
+
+ /* Set the IM account */
+ field = proto_to_contact_field (b->proto);
+ ims = e_contact_get (c, field);
+ ims = g_list_append (ims, (gpointer) b->account_name);
+ e_contact_set (c, field, (gpointer) ims);
+
+ /* Set the photo if it's not set */
+ if (b->icon != NULL) {
+ photo = e_contact_get (c, E_CONTACT_PHOTO);
+ if (photo == NULL) {
+
+ photo = g_new0 (EContactPhoto, 1);
+
+ if (! g_file_get_contents (b->icon, &photo->data, &photo->length, &error)) {
+ g_warning ("bbdb: Could not read buddy icon: %s\n", error->message);
+ g_error_free (error);
+ return;
+ }
+
+ e_contact_set (c, E_CONTACT_PHOTO, (gpointer) photo);
+ }
+ }
+
+ /* Clean up */
+ if (photo != NULL) {
+ g_free (photo->data);
+ g_free (photo);
+ }
+
+ for (l = ims; l != NULL; l = l->next)
+ g_free ((char *) l->data);
+ g_list_free (ims);
+}
+
+static EContactField
+proto_to_contact_field (const char *proto)
+{
+ if (! strcmp (proto, "prpl-oscar"))
+ return E_CONTACT_IM_AIM;
+ if (! strcmp (proto, "prpl-novell"))
+ return E_CONTACT_IM_GROUPWISE;
+ if (! strcmp (proto, "prpl-msn"))
+ return E_CONTACT_IM_MSN;
+ if (! strcmp (proto, "prpl-icq"))
+ return E_CONTACT_IM_ICQ;
+ if (! strcmp (proto, "prpl-yahoo"))
+ return E_CONTACT_IM_YAHOO;
+ if (! strcmp (proto, "prpl-jabber"))
+ return E_CONTACT_IM_JABBER;
+
+ return E_CONTACT_IM_AIM;
+}
+
+static void
+free_contact_list (GList *contacts)
+{
+ GList *l;
+
+ for (l = contacts; l != NULL; l = l->next)
+ g_object_unref (G_OBJECT (l->data));
+ g_list_free (contacts);
+}
+
+static GList *
+bbdb_get_gaim_buddy_list (void)
+{
+ char *blist_path;
+ xmlDocPtr buddy_xml;
+ xmlNodePtr root, child, blist;
+ GList *buddies = NULL;
+
+ blist_path = g_build_path ("/", getenv ("HOME"), ".gaim/blist.xml", NULL);
+
+ buddy_xml = xmlParseFile (blist_path);
+ g_free (blist_path);
+ if (! buddy_xml) {
+ fprintf (stderr, "bbdb: Could not open Gaim buddy list.\n");
+ return NULL;
+ }
+
+ root = xmlDocGetRootElement (buddy_xml);
+ if (strcmp (root->name, "gaim")) {
+ fprintf (stderr, "bbdb: Could not parse Gaim buddy list.\n");
+ xmlFreeDoc (buddy_xml);
+ return NULL;
+ }
+
+ blist = NULL;
+ for (child = root->children; child != NULL; child = child->next) {
+ if (! strcmp (child->name, "blist")) {
+ blist = child;
+ break;
+ }
+ }
+ if (blist == NULL) {
+ fprintf (stderr, "bbdb: Could not find 'blist' element in Gaim buddy list.\n");
+ xmlFreeDoc (buddy_xml);
+ return NULL;
+ }
+
+ for (child = blist->children; child != NULL; child = child->next) {
+ if (! strcmp (child->name, "group"))
+ parse_buddy_group (child, &buddies);
+ }
+
+ xmlFreeDoc (buddy_xml);
+
+ return buddies;
+}
+
+static void
+free_buddy_list (GList *blist)
+{
+ GList *l;
+
+ for (l = blist; l != NULL; l = l->next) {
+ GaimBuddy *gb = l->data;
+
+ g_free (gb->icon);
+ g_free (gb->alias);
+ g_free (gb->account_name);
+ g_free (gb->proto);
+ g_free (gb);
+ }
+
+ g_list_free (l);
+}
+
+static char *
+get_node_text (xmlNodePtr node)
+{
+ if (node->children == NULL || node->children->content == NULL ||
+ strcmp (node->children->name, "text"))
+ return NULL;
+
+ return g_strdup (node->children->content);
+}
+
+static char *
+get_buddy_icon_from_setting (xmlNodePtr setting)
+{
+ char *icon = NULL;
+
+ icon = get_node_text (setting);
+ if (icon [0] != '/') {
+ char *path;
+
+ path = g_build_path ("/", getenv ("HOME"), ".gaim/icons", icon, NULL);
+ g_free (icon);
+ icon = path;
+ }
+
+
+ return icon;
+}
+
+static void
+parse_contact (xmlNodePtr contact, GList **buddies)
+{
+ xmlNodePtr child;
+ xmlNodePtr buddy = NULL;
+ GaimBuddy *gb;
+
+ for (child = contact->children; child != NULL; child = child->next) {
+ if (! strcmp (child->name, "buddy")) {
+ buddy = child;
+ break;
+ }
+ }
+
+ if (buddy == NULL) {
+ fprintf (stderr, "bbdb: Could not find buddy in contact. Malformed Gaim buddy list file.\n");
+ return;
+ }
+
+ gb = g_new0 (GaimBuddy, 1);
+
+ gb->proto = e_xml_get_string_prop_by_name (buddy, "proto");
+
+ for (child = buddy->children; child != NULL; child = child->next) {
+ if (! strcmp (child->name, "setting")) {
+ char *setting_type;
+ setting_type = e_xml_get_string_prop_by_name (child, "name");
+
+ if (! strcmp (setting_type, "buddy_icon"))
+ gb->icon = get_buddy_icon_from_setting (child);
+
+ g_free (setting_type);
+ } else if (! strcmp (child->name, "name"))
+ gb->account_name = get_node_text (child);
+ else if (! strcmp (child->name, "alias"))
+ gb->alias = get_node_text (child);
+
+ }
+
+ *buddies = g_list_prepend (*buddies, gb);
+}
+
+static void
+parse_buddy_group (xmlNodePtr group, GList **buddies)
+{
+ xmlNodePtr child;
+
+ for (child = group->children; child != NULL; child = child->next) {
+ if (strcmp (child->name, "contact"))
+ continue;
+
+ parse_contact (child, buddies);
+ }
+}