aboutsummaryrefslogtreecommitdiffstats
path: root/modules/backup-restore/evolution-backup-restore.c
diff options
context:
space:
mode:
authorMatthew Barnes <mbarnes@redhat.com>2012-03-26 06:19:01 +0800
committerMatthew Barnes <mbarnes@redhat.com>2012-06-03 11:00:41 +0800
commitcd3759369b675e754dbed5ba19894cdd87a63a88 (patch)
tree3eaca93bd8630675c57896da239f5ade32300e40 /modules/backup-restore/evolution-backup-restore.c
parent71f5369ebfe5ee1d06b0bd1936cca80abc58e60a (diff)
downloadgsoc2013-evolution-cd3759369b675e754dbed5ba19894cdd87a63a88.tar
gsoc2013-evolution-cd3759369b675e754dbed5ba19894cdd87a63a88.tar.gz
gsoc2013-evolution-cd3759369b675e754dbed5ba19894cdd87a63a88.tar.bz2
gsoc2013-evolution-cd3759369b675e754dbed5ba19894cdd87a63a88.tar.lz
gsoc2013-evolution-cd3759369b675e754dbed5ba19894cdd87a63a88.tar.xz
gsoc2013-evolution-cd3759369b675e754dbed5ba19894cdd87a63a88.tar.zst
gsoc2013-evolution-cd3759369b675e754dbed5ba19894cdd87a63a88.zip
Add 'backup-restore' module.
Replaces the 'backup-restore' plugin.
Diffstat (limited to 'modules/backup-restore/evolution-backup-restore.c')
-rw-r--r--modules/backup-restore/evolution-backup-restore.c494
1 files changed, 494 insertions, 0 deletions
diff --git a/modules/backup-restore/evolution-backup-restore.c b/modules/backup-restore/evolution-backup-restore.c
new file mode 100644
index 0000000000..75a070a353
--- /dev/null
+++ b/modules/backup-restore/evolution-backup-restore.c
@@ -0,0 +1,494 @@
+/*
+ * evolution-backup-restore.c
+ *
+ * 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 <http://www.gnu.org/licenses/>
+ *
+ */
+
+#include <config.h>
+#include <unistd.h>
+#include <sys/types.h>
+#ifdef HAVE_SYS_WAIT_H
+# include <sys/wait.h>
+#endif
+#include <stdlib.h>
+
+#include <gtk/gtk.h>
+#include <glib/gi18n.h>
+#include <glib/gstdio.h>
+
+#include <libebackend/e-extension.h>
+
+#include <mail/e-mail-config-assistant.h>
+#include <libevolution-utils/e-alert-dialog.h>
+#include <e-util/e-util.h>
+#include <e-util/e-dialog-utils.h>
+#include <shell/e-shell-utils.h>
+#include <shell/e-shell-window.h>
+
+#include "e-mail-config-restore-page.h"
+#include "e-mail-config-restore-ready-page.h"
+
+#ifdef G_OS_WIN32
+#ifdef localtime_r
+#undef localtime_r
+#endif
+/* The localtime() in Microsoft's C library *is* thread-safe */
+#define localtime_r(timep, result) \
+ (localtime (timep) ? memcpy ( \
+ (result), localtime (timep), sizeof (*(result))) : 0)
+#endif
+
+typedef EExtension EvolutionBackupRestoreAssistant;
+typedef EExtensionClass EvolutionBackupRestoreAssistantClass;
+
+typedef EExtension EvolutionBackupRestoreMenuItems;
+typedef EExtensionClass EvolutionBackupRestoreMenuItemsClass;
+
+/* Module Entry Points */
+void e_module_load (GTypeModule *type_module);
+void e_module_unload (GTypeModule *type_module);
+
+/* Forward Declarations */
+GType evolution_backup_restore_assistant_get_type (void);
+GType evolution_backup_restore_menu_items_get_type (void);
+
+static const gchar *ui =
+"<ui>"
+" <menubar name='main-menu'>"
+" <menu action='file-menu'>"
+" <placeholder name='file-actions'>"
+" <menuitem action='settings-backup'/>"
+" <menuitem action='settings-restore'/>"
+" </placeholder>"
+" </menu>"
+" </menubar>"
+"</ui>";
+
+G_DEFINE_DYNAMIC_TYPE (
+ EvolutionBackupRestoreAssistant,
+ evolution_backup_restore_assistant,
+ E_TYPE_EXTENSION)
+
+G_DEFINE_DYNAMIC_TYPE (
+ EvolutionBackupRestoreMenuItems,
+ evolution_backup_restore_menu_items,
+ E_TYPE_EXTENSION)
+
+enum {
+ BR_OK = 1 << 0,
+ BR_START = 1 << 1
+};
+
+static void
+backup (const gchar *filename,
+ gboolean restart)
+{
+ if (restart)
+ execl (
+ EVOLUTION_TOOLSDIR "/evolution-backup",
+ "evolution-backup",
+ "--gui",
+ "--backup",
+ "--restart",
+ filename,
+ NULL);
+ else
+ execl (
+ EVOLUTION_TOOLSDIR "/evolution-backup",
+ "evolution-backup",
+ "--gui",
+ "--backup",
+ filename,
+ NULL);
+}
+
+static void
+restore (const gchar *filename,
+ gboolean restart)
+{
+ if (restart)
+ execl (
+ EVOLUTION_TOOLSDIR "/evolution-backup",
+ "evolution-backup",
+ "--gui",
+ "--restore",
+ "--restart",
+ filename,
+ NULL);
+ else
+ execl (
+ EVOLUTION_TOOLSDIR "/evolution-backup",
+ "evolution-backup",
+ "--gui",
+ "--restore",
+ filename,
+ NULL);
+}
+
+static guint32
+dialog_prompt_user (GtkWindow *parent,
+ const gchar *string,
+ const gchar *tag,
+ ...)
+{
+ GtkWidget *dialog;
+ GtkWidget *check = NULL;
+ GtkWidget *container;
+ va_list ap;
+ gint button;
+ guint32 mask = 0;
+ EAlert *alert = NULL;
+
+ va_start (ap, tag);
+ alert = e_alert_new_valist (tag, ap);
+ va_end (ap);
+
+ dialog = e_alert_dialog_new (parent, alert);
+ g_object_unref (alert);
+
+ container = e_alert_dialog_get_content_area (E_ALERT_DIALOG (dialog));
+
+ check = gtk_check_button_new_with_mnemonic (string);
+ gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (check), TRUE);
+ gtk_box_pack_start (GTK_BOX (container), check, FALSE, FALSE, 0);
+ gtk_widget_show (check);
+
+ button = gtk_dialog_run (GTK_DIALOG (dialog));
+
+ if (button == GTK_RESPONSE_YES)
+ mask |= BR_OK;
+ if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (check)))
+ mask |= BR_START;
+
+ gtk_widget_destroy (dialog);
+
+ return mask;
+}
+
+static void
+set_local_only (GtkFileChooser *file_chooser)
+{
+ /* XXX Has to be a local file, since the backup utility
+ * takes a filename argument, not a URI. */
+ gtk_file_chooser_set_local_only (file_chooser, TRUE);
+}
+
+static gchar *
+suggest_file_name (void)
+{
+ time_t t;
+ struct tm tm;
+
+ t = time (NULL);
+ localtime_r (&t, &tm);
+
+ return g_strdup_printf (
+ "evolution-backup-%04d%02d%02d.tar.gz",
+ tm.tm_year + 1900, tm.tm_mon + 1, tm.tm_mday);
+}
+
+static void
+action_settings_backup_cb (GtkAction *action,
+ EShellWindow *shell_window)
+{
+ GFile *file;
+ GFile *parent;
+ GFileInfo *file_info;
+ const gchar *attribute;
+ GError *error = NULL;
+ gchar *suggest;
+
+ suggest = suggest_file_name ();
+
+ file = e_shell_run_save_dialog (
+ e_shell_window_get_shell (shell_window),
+ _("Select name of the Evolution backup file"),
+ suggest, "*.tar.gz", (GtkCallback)
+ set_local_only, NULL);
+
+ g_free (suggest);
+
+ if (file == NULL)
+ return;
+
+ /* Make sure the parent directory can be written to. */
+
+ parent = g_file_get_parent (file);
+ attribute = G_FILE_ATTRIBUTE_ACCESS_CAN_WRITE;
+
+ /* XXX The query operation blocks the main loop but we
+ * know it's a local file, so let it slide for now. */
+ file_info = g_file_query_info (
+ parent, attribute, G_FILE_QUERY_INFO_NONE, NULL, &error);
+
+ g_object_unref (parent);
+
+ if (error != NULL) {
+ g_warning ("%s", error->message);
+ g_error_free (error);
+ return;
+ }
+
+ if (g_file_info_get_attribute_boolean (file_info, attribute)) {
+ guint32 mask;
+ gchar *path;
+
+ mask = dialog_prompt_user (
+ GTK_WINDOW (shell_window),
+ _("_Restart Evolution after backup"),
+ "org.gnome.backup-restore:backup-confirm", NULL);
+ if (mask & BR_OK) {
+ path = g_file_get_path (file);
+ backup (path, (mask & BR_START) ? TRUE: FALSE);
+ g_free (path);
+ }
+ } else {
+ e_alert_run_dialog_for_args (
+ GTK_WINDOW (shell_window),
+ "org.gnome.backup-restore:insufficient-permissions",
+ NULL);
+ }
+
+ g_object_unref (file_info);
+ g_object_unref (file);
+}
+
+static void
+action_settings_restore_cb (GtkAction *action,
+ EShellWindow *shell_window)
+{
+ GFile *file;
+ gchar *path;
+
+ file = e_shell_run_open_dialog (
+ e_shell_window_get_shell (shell_window),
+ _("Select name of the Evolution backup file to restore"),
+ (GtkCallback) set_local_only, NULL);
+
+ if (file == NULL)
+ return;
+
+ path = g_file_get_path (file);
+
+ if (evolution_backup_restore_validate_backup_file (path)) {
+ guint32 mask;
+
+ mask = dialog_prompt_user (
+ GTK_WINDOW (shell_window),
+ _("_Restart Evolution after restore"),
+ "org.gnome.backup-restore:restore-confirm", NULL);
+ if (mask & BR_OK)
+ restore (path, mask & BR_START);
+ } else {
+ e_alert_run_dialog_for_args (
+ GTK_WINDOW (shell_window),
+ "org.gnome.backup-restore:invalid-backup", NULL);
+ }
+
+ g_object_unref (file);
+ g_free (path);
+}
+
+static GtkActionEntry entries[] = {
+
+ { "settings-backup",
+ NULL,
+ N_("_Back up Evolution Data..."),
+ NULL,
+ N_("Back up Evolution data and settings to an archive file"),
+ G_CALLBACK (action_settings_backup_cb) },
+
+ { "settings-restore",
+ NULL,
+ N_("R_estore Evolution Data..."),
+ NULL,
+ N_("Restore Evolution data and settings from an archive file"),
+ G_CALLBACK (action_settings_restore_cb) }
+};
+
+static gboolean
+evolution_backup_restore_filename_to_visible (GBinding *binding,
+ const GValue *source_value,
+ GValue *target_value,
+ gpointer unused)
+{
+ const gchar *filename;
+ gboolean visible;
+
+ filename = g_value_get_string (source_value);
+ visible = (filename != NULL && *filename != '\0');
+ g_value_set_boolean (target_value, visible);
+
+ return TRUE;
+}
+
+static void
+evolution_backup_restore_prepare_cb (GtkAssistant *assistant,
+ GtkWidget *page,
+ EMailConfigRestorePage *restore_page)
+{
+ const gchar *filename;
+
+ /* If we've landed on the EMailConfigRestoreReadyPage, that
+ * means the user has chosen a valid backup file to restore
+ * so start the "evolution-backup" tool immediately. */
+
+ filename = e_mail_config_restore_page_get_filename (restore_page);
+
+ if (E_IS_MAIL_CONFIG_RESTORE_READY_PAGE (page))
+ restore (filename, TRUE);
+}
+
+static void
+evolution_backup_restore_assistant_constructed (GObject *object)
+{
+ EExtension *extension;
+ EExtensible *extensible;
+ EMailConfigAssistant *assistant;
+ const gchar *type_name;
+
+ extension = E_EXTENSION (object);
+ extensible = e_extension_get_extensible (extension);
+
+ /* Chain up to parent's constructed() method. */
+ G_OBJECT_CLASS (evolution_backup_restore_assistant_parent_class)->
+ constructed (object);
+
+ assistant = E_MAIL_CONFIG_ASSISTANT (extensible);
+
+ /* XXX We only want to add the EMailConfigRestorePage to an
+ * EStartupAssistant instance, not a normal EMailConfigAssistant.
+ * But EStartupAssistant is defined in the "startup-wizard" module
+ * and we can't access its GType without knowing its type name, so
+ * just hard-code the type name. */
+ type_name = G_OBJECT_TYPE_NAME (assistant);
+ if (g_strcmp0 (type_name, "EStartupAssistant") == 0) {
+ EMailConfigPage *restore_page;
+ EMailConfigPage *ready_page;
+
+ restore_page = e_mail_config_restore_page_new ();
+ e_mail_config_assistant_add_page (assistant, restore_page);
+
+ ready_page = e_mail_config_restore_ready_page_new ();
+ e_mail_config_assistant_add_page (assistant, ready_page);
+
+ g_object_bind_property_full (
+ restore_page, "filename",
+ ready_page, "visible",
+ G_BINDING_SYNC_CREATE,
+ evolution_backup_restore_filename_to_visible,
+ NULL,
+ NULL, (GDestroyNotify) NULL);
+
+ g_signal_connect (
+ assistant, "prepare",
+ G_CALLBACK (evolution_backup_restore_prepare_cb),
+ restore_page);
+ }
+}
+
+static void
+evolution_backup_restore_assistant_class_init (EExtensionClass *class)
+{
+ GObjectClass *object_class;
+
+ object_class = G_OBJECT_CLASS (class);
+ object_class->constructed = evolution_backup_restore_assistant_constructed;
+
+ class->extensible_type = E_TYPE_MAIL_CONFIG_ASSISTANT;
+}
+
+static void
+evolution_backup_restore_assistant_class_finalize (EExtensionClass *class)
+{
+}
+
+static void
+evolution_backup_restore_assistant_init (EExtension *extension)
+{
+}
+
+static void
+evolution_backup_restore_menu_items_constructed (GObject *object)
+{
+ EExtension *extension;
+ EExtensible *extensible;
+ EShellWindow *shell_window;
+ GtkActionGroup *action_group;
+ GtkUIManager *ui_manager;
+ GError *error = NULL;
+
+ extension = E_EXTENSION (object);
+ extensible = e_extension_get_extensible (extension);
+
+ /* Chain up to parent's constructed() method. */
+ G_OBJECT_CLASS (evolution_backup_restore_menu_items_parent_class)->
+ constructed (object);
+
+ shell_window = E_SHELL_WINDOW (extensible);
+ action_group = e_shell_window_get_action_group (shell_window, "shell");
+
+ /* Add actions to the "shell" action group. */
+ gtk_action_group_add_actions (
+ action_group, entries,
+ G_N_ELEMENTS (entries), shell_window);
+
+ /* Because we are loading from a hard-coded string, there is
+ * no chance of I/O errors. Failure here implies a malformed
+ * UI definition. Full stop. */
+ ui_manager = e_shell_window_get_ui_manager (shell_window);
+ gtk_ui_manager_add_ui_from_string (ui_manager, ui, -1, &error);
+ if (error != NULL)
+ g_error ("%s", error->message);
+}
+
+static void
+evolution_backup_restore_menu_items_class_init (EExtensionClass *class)
+{
+ GObjectClass *object_class;
+ EExtensionClass *extension_class;
+
+ object_class = G_OBJECT_CLASS (class);
+ object_class->constructed = evolution_backup_restore_menu_items_constructed;
+
+ extension_class = E_EXTENSION_CLASS (class);
+ extension_class->extensible_type = E_TYPE_SHELL_WINDOW;
+}
+
+static void
+evolution_backup_restore_menu_items_class_finalize (EExtensionClass *class)
+{
+}
+
+static void
+evolution_backup_restore_menu_items_init (EExtension *extension)
+{
+}
+
+G_MODULE_EXPORT void
+e_module_load (GTypeModule *type_module)
+{
+ evolution_backup_restore_assistant_register_type (type_module);
+ evolution_backup_restore_menu_items_register_type (type_module);
+
+ e_mail_config_restore_page_type_register (type_module);
+ e_mail_config_restore_ready_page_type_register (type_module);
+}
+
+G_MODULE_EXPORT void
+e_module_unload (GTypeModule *type_module)
+{
+}
+