aboutsummaryrefslogtreecommitdiffstats
path: root/shell/e-shell-offline-handler.c
diff options
context:
space:
mode:
Diffstat (limited to 'shell/e-shell-offline-handler.c')
-rw-r--r--shell/e-shell-offline-handler.c646
1 files changed, 646 insertions, 0 deletions
diff --git a/shell/e-shell-offline-handler.c b/shell/e-shell-offline-handler.c
new file mode 100644
index 0000000000..1967041a7e
--- /dev/null
+++ b/shell/e-shell-offline-handler.c
@@ -0,0 +1,646 @@
+/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
+/* e-shell-offline-handler.c
+ *
+ * Copyright (C) 2001 Ximian, Inc.
+ *
+ * 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., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ *
+ * Author: Ettore Perazzoli <ettore@ximian.com>
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <gtk/gtktypeutils.h>
+#include <gtk/gtksignal.h>
+#include <gtk/gtkwidget.h>
+
+#include <gal/util/e-util.h>
+
+#include <bonobo/bonobo-main.h>
+
+#include "e-shell-offline-handler.h"
+
+
+#define PARENT_TYPE GTK_TYPE_OBJECT
+static GtkObjectClass *parent_class = NULL;
+
+
+/* Private part. */
+
+struct _OfflineProgressListenerServant {
+ POA_GNOME_Evolution_OfflineProgressListener servant;
+ EShellOfflineHandler *offline_handler;
+ char *component_id;
+};
+typedef struct _OfflineProgressListenerServant OfflineProgressListenerServant;
+
+struct _ComponentInfo {
+ /* Component ID. */
+ char *id;
+
+ /* The `Evolution::Offline' interface for this component (cached just
+ to avoid going through the EComponentRegistry all the time). */
+ GNOME_Evolution_Offline offline_interface;
+
+ /* The interface and servant for the
+ `Evolution::OfflineProgressListener' we have to implement to get
+ notifications about progress of the off-line process. */
+ GNOME_Evolution_OfflineProgressListener progress_listener_interface;
+ OfflineProgressListenerServant *progress_listener_servant;
+
+ /* The current active connections for this component. This is updated
+ by the component itself through the `::ProgressListener' interface;
+ when the count reaches zero, the off-line process is considered to
+ be complete. */
+ GNOME_Evolution_ConnectionList *active_connection_list;
+};
+typedef struct _ComponentInfo ComponentInfo;
+
+struct _EShellOfflineHandlerPrivate {
+ EComponentRegistry *component_registry;
+
+ EShellView *parent_shell_view;
+
+ int num_total_connections;
+ GHashTable *id_to_component_info;
+
+ gboolean procedure_in_progress : 1;
+};
+
+
+/* Signals. */
+
+enum {
+ OFFLINE_PROCEDURE_STARTED,
+ OFFLINE_PROCEDURE_FINISHED,
+ LAST_SIGNAL
+};
+
+static guint signals[LAST_SIGNAL] = { 0 };
+
+
+/* Implementation for the OfflineProgressListener interface. */
+
+static PortableServer_ServantBase__epv OfflineProgressListener_base_epv;
+static POA_GNOME_Evolution_OfflineProgressListener__epv OfflineProgressListener_epv;
+static POA_GNOME_Evolution_OfflineProgressListener__vepv OfflineProgressListener_vepv;
+
+static OfflineProgressListenerServant *
+progress_listener_servant_new (EShellOfflineHandler *offline_handler,
+ const char *id)
+{
+ OfflineProgressListenerServant *servant;
+
+ servant = g_new0 (OfflineProgressListenerServant, 1);
+
+ servant->servant.vepv = &OfflineProgressListener_vepv;
+ servant->offline_handler = offline_handler;
+ servant->component_id = g_strdup (id);
+
+ return servant;
+}
+
+static void
+progress_listener_servant_free (OfflineProgressListenerServant *servant)
+{
+ g_free (servant->component_id);
+ g_free (servant);
+}
+
+static GNOME_Evolution_ConnectionList *
+duplicate_connection_list (const GNOME_Evolution_ConnectionList *source)
+{
+ GNOME_Evolution_ConnectionList *copy;
+ int i;
+
+ copy = GNOME_Evolution_ConnectionList__alloc ();
+
+ copy->_length = source->_length;
+ copy->_maximum = source->_length;
+
+ copy->_buffer = CORBA_sequence_GNOME_Evolution_Connection_allocbuf (copy->_maximum);
+
+ for (i = 0; i < source->_length; i++) {
+ copy->_buffer[i].hostName = CORBA_string_dup (source->_buffer[i].hostName);
+ copy->_buffer[i].portNumber = source->_buffer[i].portNumber;
+ }
+
+ return copy;
+}
+
+static void
+impl_OfflineProgressListener_updateProgress (PortableServer_Servant servant,
+ const GNOME_Evolution_ConnectionList *current_active_connections,
+ CORBA_Environment *ev)
+{
+ EShellOfflineHandler *offline_handler;
+ EShellOfflineHandlerPrivate *priv;
+ ComponentInfo *component_info;
+ int connection_delta;
+ const char *component_id;
+
+ component_id = ((OfflineProgressListenerServant *) servant)->component_id;
+
+ offline_handler = ((OfflineProgressListenerServant *) servant)->offline_handler;
+ priv = offline_handler->priv;
+
+ component_info = g_hash_table_lookup (priv->id_to_component_info, component_id);
+ g_assert (component_info != NULL);
+
+ connection_delta = component_info->active_connection_list->_length - current_active_connections->_length;
+ if (connection_delta < 0) {
+ /* FIXME: Should raise an exception or something? */
+ g_warning ("Weird, buggy component increased number of connection when going off-line -- %s",
+ component_id);
+ }
+
+ g_assert (priv->num_total_connections >= connection_delta);
+ priv->num_total_connections -= connection_delta;
+
+ CORBA_free (component_info->active_connection_list);
+ component_info->active_connection_list = duplicate_connection_list (current_active_connections);
+
+ if (priv->num_total_connections == 0)
+ gtk_signal_emit (GTK_OBJECT (offline_handler), signals[OFFLINE_PROCEDURE_FINISHED], TRUE);
+
+ /* TODO: update the dialog. */
+}
+
+static gboolean
+create_progress_listener (EShellOfflineHandler *offline_handler,
+ const char *component_id,
+ GNOME_Evolution_OfflineProgressListener *objref_return,
+ OfflineProgressListenerServant **servant_return)
+{
+ OfflineProgressListenerServant *servant;
+ CORBA_Environment ev;
+
+ *servant_return = NULL;
+ *objref_return = CORBA_OBJECT_NIL;
+
+ OfflineProgressListener_base_epv._private = NULL;
+ OfflineProgressListener_base_epv.finalize = NULL;
+ OfflineProgressListener_base_epv.default_POA = NULL;
+
+ OfflineProgressListener_epv.updateProgress = impl_OfflineProgressListener_updateProgress;
+
+ OfflineProgressListener_vepv._base_epv = &OfflineProgressListener_base_epv;
+ OfflineProgressListener_vepv.GNOME_Evolution_OfflineProgressListener_epv = &OfflineProgressListener_epv;
+
+ servant = progress_listener_servant_new (offline_handler, component_id);
+
+ CORBA_exception_init (&ev);
+
+ POA_GNOME_Evolution_OfflineProgressListener__init ((PortableServer_Servant) servant, &ev);
+ if (ev._major != CORBA_NO_EXCEPTION) {
+ g_warning ("Cannot initialize GNOME::Evolution::Offline::ProgressListener");
+ progress_listener_servant_free (servant);
+ CORBA_exception_free (&ev);
+ return FALSE;
+ }
+
+ CORBA_free (PortableServer_POA_activate_object (bonobo_poa (), servant, &ev));
+
+ if (ev._major != CORBA_NO_EXCEPTION) {
+ g_warning ("Cannot activate GNOME::Evolution::Offline::ProgressListener");
+ progress_listener_servant_free (servant);
+ CORBA_exception_free (&ev);
+ return FALSE;
+ }
+
+ *servant_return = servant;
+ *objref_return = PortableServer_POA_servant_to_reference (bonobo_poa (), servant, &ev);
+
+ CORBA_exception_free (&ev);
+
+ return TRUE;
+}
+
+
+/* ComponentInfo handling. */
+
+static ComponentInfo *
+component_info_new (const char *id,
+ const GNOME_Evolution_Offline offline_interface,
+ GNOME_Evolution_OfflineProgressListener progress_listener_interface,
+ OfflineProgressListenerServant *progress_listener_servant,
+ GNOME_Evolution_ConnectionList *active_connection_list)
+{
+ ComponentInfo *new;
+ CORBA_Environment ev;
+
+ CORBA_exception_init (&ev);
+
+ new = g_new (ComponentInfo, 1);
+ new->id = g_strdup (id);
+ new->offline_interface = CORBA_Object_duplicate (offline_interface, &ev);
+ new->progress_listener_interface = progress_listener_interface;
+ new->progress_listener_servant = progress_listener_servant;
+ new->active_connection_list = active_connection_list;
+
+ CORBA_exception_free (&ev);
+
+ return new;
+}
+
+static void
+component_info_free (ComponentInfo *component_info)
+{
+ CORBA_Environment ev;
+
+ CORBA_exception_init (&ev);
+
+ g_free (component_info->id);
+
+ CORBA_Object_release (component_info->offline_interface, &ev);
+
+ POA_GNOME_Evolution_OfflineProgressListener__fini
+ ((POA_GNOME_Evolution_OfflineProgressListener *) component_info->progress_listener_servant, &ev);
+ progress_listener_servant_free (component_info->progress_listener_servant);
+
+ CORBA_Object_release (component_info->progress_listener_interface, &ev);
+
+ CORBA_free (component_info->active_connection_list);
+
+ g_free (component_info);
+
+ CORBA_exception_free (&ev);
+}
+
+
+/* Utility functions. */
+
+static void
+hash_foreach_free_component_info (void *key,
+ void *value,
+ void *user_data)
+{
+ ComponentInfo *component_info;
+
+ component_info = (ComponentInfo *) value;
+ component_info_free (component_info);
+}
+
+
+/* Cancelling the off-line procedure. */
+
+static void
+cancel_offline (EShellOfflineHandler *offline_handler)
+{
+ EShellOfflineHandlerPrivate *priv;
+ GList *component_ids;
+ GList *p;
+
+ priv = offline_handler->priv;
+
+ component_ids = e_component_registry_get_id_list (priv->component_registry);
+
+ for (p = component_ids; p != NULL; p = p->next) {
+ EvolutionShellComponentClient *shell_component_client;
+ GNOME_Evolution_Offline offline_interface;
+ CORBA_Environment ev;
+ const char *id;
+
+ id = (const char *) p->data;
+ shell_component_client = e_component_registry_get_component_by_id (priv->component_registry, id);
+
+ offline_interface = evolution_shell_component_client_get_offline_interface (shell_component_client);
+ if (offline_interface == CORBA_OBJECT_NIL)
+ continue;
+
+ CORBA_exception_init (&ev);
+
+ GNOME_Evolution_Offline_goOnline (offline_interface, &ev);
+ if (ev._major != CORBA_NO_EXCEPTION)
+ g_warning ("Error putting component `%s' on-line.", id);
+
+ CORBA_exception_free (&ev);
+ }
+
+ e_free_string_list (component_ids);
+
+ priv->num_total_connections = 0;
+}
+
+
+/* Preparing the off-line procedure. */
+
+static gboolean
+prepare_for_offline (EShellOfflineHandler *offline_handler)
+{
+ EShellOfflineHandlerPrivate *priv;
+ GList *component_ids;
+ GList *p;
+ gboolean error;
+
+ priv = offline_handler->priv;
+
+ component_ids = e_component_registry_get_id_list (priv->component_registry);
+
+ error = FALSE;
+ for (p = component_ids; p != NULL; p = p->next) {
+ EvolutionShellComponentClient *shell_component_client;
+ GNOME_Evolution_Offline offline_interface;
+ GNOME_Evolution_OfflineProgressListener progress_listener_interface;
+ GNOME_Evolution_ConnectionList *active_connection_list;
+ OfflineProgressListenerServant *progress_listener_servant;
+ ComponentInfo *component_info;
+ CORBA_Environment ev;
+ const char *id;
+
+ id = (const char *) p->data;
+ shell_component_client = e_component_registry_get_component_by_id (priv->component_registry, id);
+ offline_interface = evolution_shell_component_client_get_offline_interface (shell_component_client);
+ if (offline_interface == CORBA_OBJECT_NIL)
+ continue;
+
+ if (! create_progress_listener (offline_handler, id,
+ &progress_listener_interface,
+ &progress_listener_servant)) {
+ g_warning ("Cannot create the Evolution::OfflineProgressListener interface for `%s'", id);
+ continue;
+ }
+
+ CORBA_exception_init (&ev);
+
+ GNOME_Evolution_Offline_prepareForOffline (offline_interface, &active_connection_list, &ev);
+ if (ev._major != CORBA_NO_EXCEPTION) {
+ g_warning ("Cannot prepare component component to go offline -- %s", id);
+
+ POA_GNOME_Evolution_OfflineProgressListener__fini
+ ((POA_GNOME_Evolution_OfflineProgressListener *) progress_listener_servant, &ev);
+ progress_listener_servant_free (progress_listener_servant);
+
+ CORBA_Object_release (progress_listener_interface, &ev);
+
+ CORBA_exception_free (&ev);
+
+ error = TRUE;
+ break;
+ }
+
+ CORBA_exception_free (&ev);
+
+ priv->num_total_connections += active_connection_list->_length;
+
+ component_info = component_info_new (id,
+ offline_interface,
+ progress_listener_interface,
+ progress_listener_servant,
+ active_connection_list);
+
+ g_assert (g_hash_table_lookup (priv->id_to_component_info, component_info->id) == NULL);
+ g_hash_table_insert (priv->id_to_component_info, component_info->id, component_info);
+ }
+
+ /* If an error occurred while preparing, just put all the components
+ on-line again. */
+ if (error)
+ cancel_offline (offline_handler);
+
+ e_free_string_list (component_ids);
+
+ return ! error;
+}
+
+
+/* Finalizing the off-line procedure. */
+
+static void
+finalize_offline_hash_foreach (void *key,
+ void *value,
+ void *user_data)
+{
+ EShellOfflineHandler *offline_handler;
+ EShellOfflineHandlerPrivate *priv;
+ ComponentInfo *component_info;
+ CORBA_Environment ev;
+
+ offline_handler = E_SHELL_OFFLINE_HANDLER (user_data);
+ priv = offline_handler->priv;
+
+ component_info = (ComponentInfo *) value;
+
+ CORBA_exception_init (&ev);
+
+ g_print ("Putting component off-line -- %s", component_info->id);
+
+ GNOME_Evolution_Offline_goOffline (component_info->offline_interface,
+ component_info->progress_listener_interface,
+ &ev);
+
+ if (ev._major != CORBA_NO_EXCEPTION) {
+ /* FIXME: Should detect an error and put all the components
+ on-line again. */
+ g_warning ("Error putting component off-line -- %s", component_info->id);
+ }
+
+ CORBA_exception_free (&ev);
+}
+
+static void
+finalize_offline (EShellOfflineHandler *offline_handler)
+{
+ EShellOfflineHandlerPrivate *priv;
+
+ priv = offline_handler->priv;
+
+ g_hash_table_foreach (priv->id_to_component_info, finalize_offline_hash_foreach, offline_handler);
+
+ if (priv->num_total_connections == 0) {
+ /* Nothing else to do, we are all set. */
+ gtk_signal_emit (GTK_OBJECT (offline_handler), signals[OFFLINE_PROCEDURE_FINISHED], TRUE);
+ }
+}
+
+
+/* The confirmation dialog. */
+
+static void
+pop_up_confirmation_dialog (EShellOfflineHandler *offline_handler)
+{
+ EShellOfflineHandlerPrivate *priv;
+
+ priv = offline_handler->priv;
+
+ g_warning ("Should pop up dialog here");
+
+ finalize_offline (offline_handler);
+}
+
+
+/* GtkObject methods. */
+
+static void
+impl_destroy (GtkObject *object)
+{
+ EShellOfflineHandler *offline_handler;
+ EShellOfflineHandlerPrivate *priv;
+
+ offline_handler = E_SHELL_OFFLINE_HANDLER (object);
+ priv = offline_handler->priv;
+
+ if (priv->component_registry != NULL)
+ gtk_object_unref (GTK_OBJECT (priv->component_registry));
+
+ g_hash_table_foreach (priv->id_to_component_info, hash_foreach_free_component_info, NULL);
+ g_hash_table_destroy (priv->id_to_component_info);
+
+ if (GTK_OBJECT_CLASS (parent_class)->destroy != NULL)
+ (* GTK_OBJECT_CLASS (parent_class)->destroy) (object);
+}
+
+
+/* GTK type handling. */
+
+static void
+class_init (EShellOfflineHandlerClass *klass)
+{
+ GtkObjectClass *object_class;
+
+ object_class = GTK_OBJECT_CLASS (klass);
+ object_class->destroy = impl_destroy;
+
+ parent_class = gtk_type_class (gtk_object_get_type ());
+
+ signals[OFFLINE_PROCEDURE_STARTED]
+ = gtk_signal_new ("offline_procedure_started",
+ GTK_RUN_LAST,
+ object_class->type,
+ GTK_SIGNAL_OFFSET (EShellOfflineHandlerClass, offline_procedure_started),
+ gtk_marshal_NONE__NONE,
+ GTK_TYPE_NONE, 0);
+
+ signals[OFFLINE_PROCEDURE_FINISHED]
+ = gtk_signal_new ("offline_procedure_finished",
+ GTK_RUN_LAST,
+ object_class->type,
+ GTK_SIGNAL_OFFSET (EShellOfflineHandlerClass, offline_procedure_finished),
+ gtk_marshal_NONE__BOOL,
+ GTK_TYPE_NONE, 1,
+ GTK_TYPE_BOOL);
+
+ gtk_object_class_add_signals (object_class, signals, LAST_SIGNAL);
+}
+
+
+static void
+init (EShellOfflineHandler *shell_offline_handler)
+{
+ EShellOfflineHandlerPrivate *priv;
+
+ priv = g_new (EShellOfflineHandlerPrivate, 1);
+
+ priv->component_registry = NULL;
+ priv->parent_shell_view = NULL;
+ priv->procedure_in_progress = TRUE;
+
+ priv->num_total_connections = 0;
+ priv->id_to_component_info = g_hash_table_new (g_str_hash, g_str_equal);
+
+ shell_offline_handler->priv = priv;
+}
+
+
+/**
+ * e_shell_offline_handler_construct:
+ * @offline_handler: A pointer to an EShellOfflineHandler to construct.
+ * @component_registry: The registry for the components that we want to put
+ * off-line.
+ *
+ * Construct the @offline_handler.
+ **/
+void
+e_shell_offline_handler_construct (EShellOfflineHandler *offline_handler,
+ EComponentRegistry *component_registry)
+{
+ EShellOfflineHandlerPrivate *priv;
+
+ g_return_if_fail (offline_handler != NULL);
+ g_return_if_fail (E_IS_SHELL_OFFLINE_HANDLER (offline_handler));
+ g_return_if_fail (component_registry != NULL);
+ g_return_if_fail (E_IS_COMPONENT_REGISTRY (component_registry));
+
+ g_assert (priv->component_registry == NULL);
+
+ gtk_object_ref (GTK_OBJECT (component_registry));
+ priv->component_registry = component_registry;
+}
+
+/**
+ * e_shell_offline_handler_new:
+ * @component_registry: The registry for the components that we want to put
+ * off-line.
+ *
+ * Create a new offline handler.
+ *
+ * Return value: A pointer to the newly created EShellOfflineHandler object.
+ **/
+EShellOfflineHandler *
+e_shell_offline_handler_new (EComponentRegistry *component_registry)
+{
+ EShellOfflineHandler *offline_handler;
+
+ g_return_val_if_fail (component_registry != NULL, NULL);
+ g_return_val_if_fail (E_IS_COMPONENT_REGISTRY (component_registry), NULL);
+
+ offline_handler = (EShellOfflineHandler *) gtk_type_new (e_shell_offline_handler_get_type ());
+ e_shell_offline_handler_construct (offline_handler, component_registry);
+
+ return offline_handler;
+}
+
+
+/**
+ * e_shell_offline_handler_put_components_offline:
+ * @offline_handler: A pointer to an EShellOfflineHandler object.
+ *
+ * Put the components offline.
+ **/
+void
+e_shell_offline_handler_put_components_offline (EShellOfflineHandler *offline_handler,
+ EShellView *parent_shell_view)
+{
+ EShellOfflineHandlerPrivate *priv;
+
+ g_return_if_fail (offline_handler != NULL);
+ g_return_if_fail (E_IS_SHELL_OFFLINE_HANDLER (offline_handler));
+ g_return_if_fail (parent_shell_view == NULL || E_IS_SHELL_VIEW (parent_shell_view));
+
+ priv = offline_handler->priv;
+
+ priv->procedure_in_progress = TRUE;
+ priv->parent_shell_view = parent_shell_view;
+
+ gtk_signal_emit (GTK_OBJECT (offline_handler), signals[OFFLINE_PROCEDURE_STARTED]);
+
+ if (! prepare_for_offline (offline_handler)) {
+ /* FIXME: Maybe do something smarter here. */
+ g_warning ("Couldn't put components off-line for some internal error");
+ gtk_signal_emit (GTK_OBJECT (offline_handler), signals[OFFLINE_PROCEDURE_FINISHED], FALSE);
+ return;
+ }
+
+ if (priv->num_total_connections == 0 && priv->parent_shell_view != NULL)
+ pop_up_confirmation_dialog (offline_handler);
+ else
+ finalize_offline (offline_handler);
+}
+
+
+E_MAKE_TYPE (e_shell_offline_handler, "EShellOfflineHandler", EShellOfflineHandler, class_init, init, PARENT_TYPE)