From 8c0bd86d5fdd6d87c3170e2a01423e7c7018a981 Mon Sep 17 00:00:00 2001 From: Matthew Barnes Date: Fri, 14 Nov 2008 03:56:01 +0000 Subject: Rearranged some of the addressbook code to try and eliminate some circular dependencies in our libraries. The circular dependency between the composer and the mail module is still causing me headaches. And it doesn't help that the addressbook and calendar also want to link to the composer. svn path=/branches/kill-bonobo/; revision=36782 --- addressbook/util/Makefile.am | 2 + addressbook/util/addressbook.c | 338 +++++++++++++++++++++++++++++++++++++++ addressbook/util/addressbook.h | 31 ++++ addressbook/util/eab-book-util.h | 2 - 4 files changed, 371 insertions(+), 2 deletions(-) create mode 100644 addressbook/util/addressbook.c create mode 100644 addressbook/util/addressbook.h (limited to 'addressbook/util') diff --git a/addressbook/util/Makefile.am b/addressbook/util/Makefile.am index ca7340017a..1a1fced7bd 100644 --- a/addressbook/util/Makefile.am +++ b/addressbook/util/Makefile.am @@ -13,6 +13,8 @@ INCLUDES = \ privsolib_LTLIBRARIES = libeabutil.la libeabutil_la_SOURCES = \ + addressbook.c \ + addressbook.h \ eab-book-util.c \ eab-book-util.h diff --git a/addressbook/util/addressbook.c b/addressbook/util/addressbook.c new file mode 100644 index 0000000000..c8285d8171 --- /dev/null +++ b/addressbook/util/addressbook.c @@ -0,0 +1,338 @@ +/* + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) version 3. + * + * 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with the program; if not, see + * + * + * Authors: + * Chris Lahey + * + * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) + * + */ + +#include + +#include + +#include +#include +#include +#include +#include + +#include "e-util/e-error.h" +#include "addressbook.h" + +#define d(x) + +static void addressbook_authenticate (EBook *book, gboolean previous_failure, + ESource *source, EBookCallback cb, gpointer closure); +static void auth_required_cb (EBook *book, gpointer data); + +typedef struct { + EBookCallback cb; + ESource *source; + gpointer closure; + guint cancelled : 1; +} LoadSourceData; + +static void +free_load_source_data (LoadSourceData *data) +{ + if (data->source) + g_object_unref (data->source); + g_free (data); +} + +/*this function removes of anything present after semicolon +in uri*/ + +static gchar* +remove_parameters_from_uri (const gchar *uri) +{ + char *euri_str; + EUri *euri; + + euri = e_uri_new (uri); + euri_str = e_uri_to_string (euri, FALSE); + e_uri_free (euri); + return euri_str; +} + +static void +load_source_auth_cb (EBook *book, EBookStatus status, gpointer closure) +{ + LoadSourceData *data = closure; + gboolean was_in = g_object_get_data (G_OBJECT (book), "authenticated") != NULL; + + g_object_set_data (G_OBJECT (book), "authenticated", NULL); + + if (data->cancelled) { + free_load_source_data (data); + return; + } + + if (status != E_BOOK_ERROR_OK) { + + /* the user clicked cancel in the password dialog */ + if (status == E_BOOK_ERROR_CANCELLED) { + + if (e_book_check_static_capability (book, "anon-access")) { + + GtkWidget *dialog; + + /* XXX "LDAP" has to be removed from the folowing message + so that it wil valid for other servers which provide + anonymous access*/ + + dialog = gtk_message_dialog_new (NULL, + 0, + GTK_MESSAGE_WARNING, + GTK_BUTTONS_OK, + "%s", _("Accessing LDAP Server anonymously")); + g_signal_connect (dialog, "response", G_CALLBACK(gtk_widget_destroy), NULL); + gtk_widget_show (dialog); + status = E_BOOK_ERROR_OK; + + goto done; + } + } else if (status == E_BOOK_ERROR_INVALID_SERVER_VERSION) { + e_error_run (NULL, "addressbook:server-version", NULL); + status = E_BOOK_ERROR_OK; + goto done; + } else if (status == E_BOOK_ERROR_UNSUPPORTED_AUTHENTICATION_METHOD) { + goto done; + } else { + if (status == E_BOOK_ERROR_AUTHENTICATION_FAILED) { + const gchar *uri = e_book_get_uri (book); + gchar *stripped_uri = remove_parameters_from_uri (uri); + const gchar *auth_domain = e_source_get_property (data->source, "auth-domain"); + const gchar *component_name; + + component_name = auth_domain ? auth_domain : "Addressbook"; + + e_passwords_forget_password (component_name, stripped_uri); + + g_free (stripped_uri); + } else if (was_in) { + /* We already tried to authenticate to the server, and it failed with + other reason than with E_BOOK_ERROR_AUTHENTICATION_FAILED, thus stop + poking with the server and report error to the user. */ + goto done; + } + + g_object_set_data (G_OBJECT (book), "authenticated", GINT_TO_POINTER (1)); + addressbook_authenticate (book, TRUE, data->source, load_source_auth_cb, closure); + return; + } + } + +done: + if (data->cb) + data->cb (book, status, data->closure); + + free_load_source_data (data); +} + +static gboolean +get_remember_password (ESource *source) +{ + const gchar *value; + + value = e_source_get_property (source, "remember_password"); + if (value && !g_ascii_strcasecmp (value, "true")) + return TRUE; + + return FALSE; +} + +static void +set_remember_password (ESource *source, gboolean value) +{ + e_source_set_property (source, "remember_password", + value ? "true" : "false"); +} + +static void +addressbook_authenticate (EBook *book, gboolean previous_failure, ESource *source, + EBookCallback cb, gpointer closure) +{ + const char *password = NULL; + char *pass_dup = NULL; + const gchar *auth; + const gchar *user; + gchar *uri = remove_parameters_from_uri(e_book_get_uri (book)); + const gchar *auth_domain = e_source_get_property (source, "auth-domain"); + const gchar *component_name; + + component_name = auth_domain ? auth_domain : "Addressbook"; + + password = e_passwords_get_password (component_name, uri); + + auth = e_source_get_property (source, "auth"); + + if (auth && !strcmp ("ldap/simple-binddn", auth)) { + user = e_source_get_property (source, "binddn"); + } + else if (auth && !strcmp ("plain/password", auth)) { + user = e_source_get_property (source, "user"); + if (!user) { + user = e_source_get_property (source, "username"); + } + } + else { + user = e_source_get_property (source, "email_addr"); + } + if (!user) + user = ""; + + if (!password) { + char *prompt; + char *password_prompt; + gboolean remember; + char *failed_auth; + guint32 flags = E_PASSWORDS_REMEMBER_FOREVER|E_PASSWORDS_SECRET|E_PASSWORDS_ONLINE; + + if (previous_failure) { + failed_auth = _("Failed to authenticate.\n"); + flags |= E_PASSWORDS_REPROMPT; + } + else { + failed_auth = ""; + } + + password_prompt = g_strdup_printf (_("Enter password for %s (user %s)"), + e_source_peek_name (source), user); + + prompt = g_strconcat (failed_auth, password_prompt, NULL); + g_free (password_prompt); + + remember = get_remember_password (source); + pass_dup = e_passwords_ask_password ( + _("Enter password"), component_name, + uri, prompt, flags, &remember, NULL); + if (remember != get_remember_password (source)) + set_remember_password (source, remember); + + g_free (prompt); + } + + if (password || pass_dup) { + e_book_async_authenticate_user (book, user, password ? password : pass_dup, + e_source_get_property (source, "auth"), + cb, closure); + g_free (pass_dup); + } + else { + /* they hit cancel */ + cb (book, E_BOOK_ERROR_CANCELLED, closure); + } + + g_free (uri); +} + + + +static void +auth_required_cb (EBook *book, gpointer data) +{ + LoadSourceData *load_source_data = g_new0(LoadSourceData, 1); + + load_source_data->source = g_object_ref (g_object_ref (e_book_get_source (book))); + load_source_data->cancelled = FALSE; + addressbook_authenticate (book, FALSE, load_source_data->source, + load_source_auth_cb, load_source_data); + + + +} +static void +load_source_cb (EBook *book, EBookStatus status, gpointer closure) +{ + LoadSourceData *load_source_data = closure; + + if (load_source_data->cancelled) { + free_load_source_data (load_source_data); + return; + } + + if (status == E_BOOK_ERROR_OK && book != NULL) { + const gchar *auth; + + auth = e_source_get_property (load_source_data->source, "auth"); + if (auth && strcmp (auth, "none")) { + g_signal_connect (book, "auth_required", G_CALLBACK(auth_required_cb), NULL); + + if (e_book_is_online (book)) { + addressbook_authenticate (book, FALSE, load_source_data->source, + load_source_auth_cb, closure); + return; + } + } + } + load_source_data->cb (book, status, load_source_data->closure); + free_load_source_data (load_source_data); +} + +guint +addressbook_load (EBook *book, + EBookCallback cb, gpointer closure) +{ + LoadSourceData *load_source_data = g_new0 (LoadSourceData, 1); + + load_source_data->cb = cb; + load_source_data->closure = closure; + load_source_data->source = g_object_ref (g_object_ref (e_book_get_source (book))); + load_source_data->cancelled = FALSE; + + e_book_async_open (book, FALSE, load_source_cb, load_source_data); + + return GPOINTER_TO_UINT (load_source_data); +} + +void +addressbook_load_cancel (guint id) +{ + LoadSourceData *load_source_data = GUINT_TO_POINTER (id); + + load_source_data->cancelled = TRUE; +} + +static void +default_book_cb (EBook *book, EBookStatus status, gpointer closure) +{ + LoadSourceData *load_source_data = closure; + + if (status == E_BOOK_ERROR_OK) + load_source_data->source = g_object_ref (e_book_get_source (book)); + + load_source_cb (book, status, closure); +} + +void +addressbook_load_default_book (EBookCallback cb, gpointer closure) +{ + LoadSourceData *load_source_data = g_new (LoadSourceData, 1); + EBook *book; + + load_source_data->cb = cb; + load_source_data->source = NULL; + load_source_data->closure = closure; + load_source_data->cancelled = FALSE; + + book = e_book_new_default_addressbook (NULL); + if (!book) + load_source_cb (NULL, E_BOOK_ERROR_OTHER_ERROR, load_source_data); /* XXX we should just use a GError and it's error code here */ + else + e_book_async_open (book, FALSE, default_book_cb, load_source_data); +} diff --git a/addressbook/util/addressbook.h b/addressbook/util/addressbook.h new file mode 100644 index 0000000000..2e25448717 --- /dev/null +++ b/addressbook/util/addressbook.h @@ -0,0 +1,31 @@ +/* + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public + * License as published by the Free Software Foundation; either + * version 2 of the License, or (at your option) version 3. + * + * 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with the program; if not, see + * + * + * Authors: + * + * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) + * + */ + +#ifndef __ADDRESSBOOK_H__ +#define __ADDRESSBOOK_H__ + +#include + +guint addressbook_load (EBook *book, EBookCallback cb, gpointer closure); +void addressbook_load_cancel (guint id); +void addressbook_load_default_book (EBookCallback open_response, gpointer closure); + +#endif /* __ADDRESSBOOK_H__ */ diff --git a/addressbook/util/eab-book-util.h b/addressbook/util/eab-book-util.h index 66c3a6c23e..2051c77ad0 100644 --- a/addressbook/util/eab-book-util.h +++ b/addressbook/util/eab-book-util.h @@ -24,8 +24,6 @@ #ifndef __EAB_UTIL_H__ #define __EAB_UTIL_H__ -#include -#include #include #include "e-util/e-config-listener.h" -- cgit