/* * * 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 * * * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) * */ #ifdef HAVE_CONFIG_H #include #endif #include #include #ifdef HAVE_SYS_WAIT_H # include #endif #include #include #include #include #include "mail/em-config.h" #include "mail/em-account-editor.h" #include "e-util/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" #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 gboolean e_plugin_ui_init (GtkUIManager *ui_manager, EShellWindow *shell_window); GtkWidget * backup_restore_page (EPlugin *ep, EConfigHookItemFactoryData *hook_data); void backup_restore_commit (EPlugin *ep, EMConfigTargetAccount *target); void backup_restore_abort (EPlugin *ep, EMConfigTargetAccount *target); typedef enum _br_flags { BR_OK = 1 << 0, BR_START = 1 << 1 }br_flags; gint e_plugin_lib_enable (EPlugin *ep, gint enable); gint e_plugin_lib_enable (EPlugin *ep, gint enable) { return 0; } static void backup (const gchar *filename, gboolean restart) { if (restart) execl (EVOLUTION_TOOLSDIR "/evolution-backup", "evolution-backup", "--gui", "--backup", "--restart", filename, (gchar *)NULL); else execl (EVOLUTION_TOOLSDIR "/evolution-backup", "evolution-backup", "--gui", "--backup", filename, (gchar *)NULL); } static void restore (const gchar *filename, gboolean restart) { if (restart) execl (EVOLUTION_TOOLSDIR "/evolution-backup", "evolution-backup", "--gui", "--restore", "--restart", filename, (gchar *)NULL); else execl (EVOLUTION_TOOLSDIR "/evolution-backup", "evolution-backup", "--gui", "--restore", filename, (gchar *)NULL); } static gboolean sanity_check (const gchar *filename) { gchar *command; gint result; gchar *quotedfname, *toolfname; quotedfname = g_shell_quote (filename); toolfname = g_build_filename (EVOLUTION_TOOLSDIR, "evolution-backup", NULL); command = g_strdup_printf("%s --check %s", toolfname, quotedfname); result = system (command); g_free (command); g_free (quotedfname); g_free (toolfname); #ifdef HAVE_SYS_WAIT_H g_message ("Sanity check result %d:%d %d", WIFEXITED (result), WEXITSTATUS (result), result); return WIFEXITED (result) && (WEXITSTATUS (result) == 0); #else return result; #endif } 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); /* We should hardcode this to true */ 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 (sanity_check (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 void check_toggled (GtkToggleButton *button, GtkAssistant *assistant) { GtkWidget *box = g_object_get_data ((GObject *)button, "box"); gboolean state = gtk_toggle_button_get_active ((GtkToggleButton *) button); gtk_widget_set_sensitive (box, state); g_object_set_data ((GObject *)assistant, "restore", GINT_TO_POINTER (state?1:0)); e_config_target_changed ((EConfig *) g_object_get_data ((GObject *)assistant, "restore-config"), E_CONFIG_TARGET_CHANGED_STATE); } static void file_changed (GtkFileChooser *chooser, GtkAssistant *assistant) { gchar *file = NULL, *prevfile = NULL; file = gtk_file_chooser_get_filename (chooser); prevfile = g_object_get_data ((GObject *)assistant, "restore-file"); g_object_set_data ((GObject *)assistant, "restore-file", file); g_free (prevfile); e_config_target_changed ((EConfig *) g_object_get_data ((GObject *)assistant, "restore-config"), E_CONFIG_TARGET_CHANGED_STATE); } static gboolean backup_restore_check (EConfig *ec, const gchar *pageid, gpointer data) { GtkAssistant *assistant = data; gint do_restore; gchar *file; g_return_val_if_fail (data != NULL, FALSE); g_return_val_if_fail (GTK_IS_ASSISTANT (data), FALSE); do_restore = GPOINTER_TO_INT (g_object_get_data ((GObject *)assistant, "restore")); file = g_object_get_data ((GObject *)assistant, "restore-file"); e_config_set_page_is_finish (ec, "0.startup_page.10.backup_restore", do_restore); return !do_restore || file; } GtkWidget * backup_restore_page (EPlugin *ep, EConfigHookItemFactoryData *hook_data) { GtkWidget *page, *hbox, *label, *cbox, *button; GtkAssistant *assistant = GTK_ASSISTANT (hook_data->parent); page = gtk_vbox_new (FALSE, 6); gtk_container_set_border_width (GTK_CONTAINER (page), 12); hbox = gtk_hbox_new (FALSE, 6); label = gtk_label_new (_("You can restore Evolution from your backup. It can restore all the Mails, Calendars, Tasks, Memos, Contacts. It also restores all your personal settings, mail filters etc.")); gtk_label_set_line_wrap ((GtkLabel *) label, TRUE); gtk_label_set_single_line_mode ((GtkLabel *) label, FALSE); gtk_box_pack_start ((GtkBox *) hbox, label, FALSE, FALSE, 6); gtk_box_pack_start ((GtkBox *) page, hbox, FALSE, FALSE, 0); hbox = gtk_hbox_new (FALSE, 6); cbox = gtk_check_button_new_with_mnemonic (_("_Restore Evolution from the backup file")); g_signal_connect (cbox, "toggled", G_CALLBACK (check_toggled), assistant); gtk_box_pack_start ((GtkBox *) hbox, cbox, FALSE, FALSE, 6); gtk_box_pack_start ((GtkBox *) page, hbox, FALSE, FALSE, 0); hbox = gtk_hbox_new (FALSE, 6); g_object_set_data ((GObject *)cbox, "box", hbox); label = gtk_label_new (_("Please select an Evolution Archive to restore:")); gtk_box_pack_start ((GtkBox *) hbox, label, FALSE, FALSE, 12); button = gtk_file_chooser_button_new (_("Choose a file to restore"), GTK_FILE_CHOOSER_ACTION_OPEN); g_signal_connect (button, "selection-changed", G_CALLBACK (file_changed), assistant); gtk_file_chooser_button_set_width_chars ((GtkFileChooserButton *) button, 20); gtk_box_pack_start ((GtkBox *) hbox, button, FALSE, FALSE, 0); gtk_box_pack_start ((GtkBox *) page, hbox, FALSE, FALSE, 0); gtk_widget_set_sensitive (hbox, FALSE); gtk_assistant_append_page (assistant, page); gtk_assistant_set_page_title (assistant, page, _("Restore from backup")); gtk_widget_show_all (page); g_object_set_data ((GObject *)assistant, "restore", GINT_TO_POINTER (FALSE)); g_object_set_data ((GObject *)assistant, "restore-config", hook_data->config); e_config_add_page_check (hook_data->config, "0.startup_page.10.backup_restore", backup_restore_check, assistant); return GTK_WIDGET (page); } void backup_restore_commit (EPlugin *ep, EMConfigTargetAccount *target) { GtkWidget *assistant = target->target.config->widget; gboolean state = GPOINTER_TO_INT (g_object_get_data ((GObject *)assistant, "restore")) ? TRUE : FALSE; gchar *file = g_object_get_data ((GObject *)assistant, "restore-file"); if (state) { if (!file || !sanity_check (file)) { e_alert_run_dialog_for_args ((GtkWindow *) assistant, "org.gnome.backup-restore:invalid-backup", NULL); } else { restore (file, TRUE); } } } void backup_restore_abort (EPlugin *ep, EMConfigTargetAccount *target) { /* Nothing really */ } 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) } }; gboolean e_plugin_ui_init (GtkUIManager *ui_manager, EShellWindow *shell_window) { GtkActionGroup *action_group; 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); return TRUE; }