aboutsummaryrefslogtreecommitdiffstats
path: root/plugins/backup-restore
diff options
context:
space:
mode:
Diffstat (limited to 'plugins/backup-restore')
-rw-r--r--plugins/backup-restore/backup.c295
1 files changed, 255 insertions, 40 deletions
diff --git a/plugins/backup-restore/backup.c b/plugins/backup-restore/backup.c
index dcfe20c5d9..03ff5a4d97 100644
--- a/plugins/backup-restore/backup.c
+++ b/plugins/backup-restore/backup.c
@@ -27,9 +27,13 @@
#include <stdlib.h>
#include <string.h>
+#include <glib.h>
#include <glib/gi18n.h>
+#include <glib/gstdio.h>
#include <gtk/gtk.h>
+#include <libedataserver/e-data-server-util.h>
+
#ifdef G_OS_WIN32
#ifdef DATADIR
#undef DATADIR
@@ -50,10 +54,9 @@
#define EVOUSERDATADIR_MAGIC "#EVO_USERDATADIR#"
#define EVOLUTION "evolution"
-#define EVOLUTION_DIR "$HOME/.evolution/"
-#define EVOLUTION_DIR_BACKUP "$HOME/.evolution-old/"
+#define EVOLUTION_DIR "$DATADIR/"
+#define EVOLUTION_DIR_FILE EVOLUTION ".dir"
#define GCONF_DUMP_FILE "backup-restore-gconf.xml"
-#define GCONF_DUMP_PATH EVOLUTION_DIR GCONF_DUMP_FILE
#define GCONF_DIR "/apps/evolution"
#define ARCHIVE_NAME "evolution-backup.tar.gz"
@@ -93,7 +96,7 @@ static GOptionEntry options[] = {
#define print_and_run(x) G_STMT_START { g_message ("%s", x); system (x); } G_STMT_END
#define CANCEL(x) if (x) return;
-static gboolean check (const gchar *filename);
+static gboolean check (const gchar *filename, gboolean *is_new_format);
static GString *
replace_string (const gchar *text, const gchar *find, const gchar *replace)
@@ -111,7 +114,7 @@ replace_string (const gchar *text, const gchar *find, const gchar *replace)
p = text;
while (next = strstr (p, find), next) {
- if (p + 1 < next)
+ if (p < next)
g_string_append_len (str, p, next - p);
if (replace && *replace)
@@ -125,6 +128,64 @@ replace_string (const gchar *text, const gchar *find, const gchar *replace)
return str;
}
+static const gchar *
+strip_home_dir (const gchar *dir)
+{
+ const gchar *home_dir, *res;
+
+ g_return_val_if_fail (dir != NULL, NULL);
+
+ home_dir = g_get_home_dir ();
+ g_return_val_if_fail (home_dir != NULL, dir);
+ g_return_val_if_fail (*home_dir != '\0', dir);
+
+ res = dir;
+ if (g_str_has_prefix (res, home_dir))
+ res += strlen (home_dir);
+
+ if (*res == G_DIR_SEPARATOR)
+ res++;
+
+ return res;
+}
+
+static GString *
+replace_variables (const gchar *str)
+{
+ GString *res = NULL, *use;
+ const gchar *strip_datadir, *strip_configdir;
+
+ g_return_val_if_fail (str != NULL, NULL);
+
+ strip_datadir = strip_home_dir (e_get_user_data_dir ());
+ strip_configdir = strip_home_dir (e_get_user_config_dir ());
+
+ #define repl(_find, _replace) \
+ use = replace_string (res ? res->str : str, _find, _replace); \
+ g_return_val_if_fail (use != NULL, NULL); \
+ if (res) \
+ g_string_free (res, TRUE); \
+ res = use;
+
+ repl ("$HOME", g_get_home_dir ());
+ repl ("$TMP", g_get_tmp_dir ());
+ repl ("$DATADIR", e_get_user_data_dir ());
+ repl ("$CONFIGDIR", e_get_user_config_dir ());
+ repl ("$STRIPDATADIR", strip_datadir);
+ repl ("$STRIPCONFIGDIR", strip_configdir);
+
+ #undef repl
+
+ g_return_val_if_fail (res != NULL, NULL);
+
+ /* remove trailing dir separator */
+ while (res->len > 0 && res->str[res->len - 1] == G_DIR_SEPARATOR) {
+ g_string_truncate (res, res->len - 1);
+ }
+
+ return res;
+}
+
static void
replace_in_file (const gchar *filename, const gchar *find, const gchar *replace)
{
@@ -137,11 +198,11 @@ replace_in_file (const gchar *filename, const gchar *find, const gchar *replace)
g_return_if_fail (*find);
g_return_if_fail (replace != NULL);
- if (strstr (filename, "$HOME")) {
- filenamestr = replace_string (filename, "$HOME", g_get_home_dir ());
+ if (strstr (filename, "$")) {
+ filenamestr = replace_variables (filename);
if (!filenamestr) {
- g_warning ("%s: Replace string on $HOME failed!", G_STRFUNC);
+ g_warning ("%s: Replace variables in '%s' failed!", G_STRFUNC, filename);
return;
}
@@ -178,9 +239,9 @@ run_cmd (const gchar *cmd)
if (!cmd)
return;
- if (strstr (cmd, "$HOME") != NULL) {
+ if (strstr (cmd, "$") != NULL) {
/* read the doc for g_get_home_dir to know why replacing it here */
- GString *str = replace_string (cmd, "$HOME", g_get_home_dir ());
+ GString *str = replace_variables (cmd);
if (str) {
print_and_run (str->str);
@@ -191,6 +252,32 @@ run_cmd (const gchar *cmd)
}
static void
+write_dir_file (void)
+{
+ GString *content, *filename;
+ GError *error = NULL;
+
+ filename = replace_variables ("$HOME/" EVOLUTION_DIR_FILE);
+ g_return_if_fail (filename != NULL);
+
+ content = replace_variables (
+ "[dirs]\n"
+ "data=$STRIPDATADIR\n"
+ "config=$STRIPCONFIGDIR\n");
+ g_return_if_fail (content != NULL);
+
+ g_file_set_contents (filename->str, content->str, content->len, &error);
+
+ if (error) {
+ g_warning ("Failed to write file '%s': %s\n", filename->str, error->message);
+ g_error_free (error);
+ }
+
+ g_string_free (filename, TRUE);
+ g_string_free (content, TRUE);
+}
+
+static void
backup (const gchar *filename)
{
gchar *command;
@@ -204,13 +291,15 @@ backup (const gchar *filename)
/* FIXME Will the versioned setting always work? */
run_cmd (EVOLUTION " --quit");
- run_cmd ("rm $HOME/.evolution/.running");
+ run_cmd ("rm $DATADIR/.running");
CANCEL (complete);
txt = _("Backing Evolution accounts and settings");
- run_cmd ("gconftool-2 --dump " GCONF_DIR " > " GCONF_DUMP_PATH);
+ run_cmd ("gconftool-2 --dump " GCONF_DIR " > " EVOLUTION_DIR GCONF_DUMP_FILE);
- replace_in_file (GCONF_DUMP_PATH, e_get_user_data_dir (), EVOUSERDATADIR_MAGIC);
+ replace_in_file (EVOLUTION_DIR GCONF_DUMP_FILE, e_get_user_data_dir (), EVOUSERDATADIR_MAGIC);
+
+ write_dir_file ();
CANCEL (complete);
txt = _("Backing Evolution data (Mails, Contacts, Calendar, Tasks, Memos)");
@@ -219,11 +308,13 @@ backup (const gchar *filename)
/* FIXME compression type?" */
/* FIXME date/time stamp?" */
/* FIXME backup location?" */
- command = g_strdup_printf ("cd $HOME && tar chf - .evolution .camel_certs | gzip > %s", quotedfname);
+ command = g_strdup_printf ("cd $HOME && tar chf - $STRIPDATADIR $STRIPCONFIGDIR .camel_certs " EVOLUTION_DIR_FILE " | gzip > %s", quotedfname);
run_cmd (command);
g_free (command);
g_free (quotedfname);
+ run_cmd ("rm $HOME/" EVOLUTION_DIR_FILE);
+
txt = _("Backup complete");
if (restart_arg) {
@@ -238,24 +329,68 @@ backup (const gchar *filename)
}
static void
+extract_backup_dirs (const gchar *filename, gchar **data_dir, gchar **config_dir)
+{
+ GKeyFile *key_file;
+ GError *error = NULL;
+
+ g_return_if_fail (filename != NULL);
+ g_return_if_fail (data_dir != NULL);
+ g_return_if_fail (config_dir != NULL);
+
+ key_file = g_key_file_new ();
+ g_key_file_load_from_file (key_file, filename, G_KEY_FILE_NONE, &error);
+
+ if (error) {
+ g_warning ("Failed to read '%s': %s", filename, error->message);
+ g_error_free (error);
+ } else {
+ gchar *tmp;
+
+ tmp = g_key_file_get_value (key_file, "dirs", "data", NULL);
+ if (tmp)
+ *data_dir = g_shell_quote (tmp);
+ g_free (tmp);
+
+ tmp = g_key_file_get_value (key_file, "dirs", "config", NULL);
+ if (tmp)
+ *config_dir = g_shell_quote (tmp);
+ g_free (tmp);
+ }
+
+ g_key_file_free (key_file);
+}
+
+static gint
+get_dir_level (const gchar *dir)
+{
+ gint res = 0, i;
+
+ g_return_val_if_fail (dir != NULL, -1);
+
+ for (i = 0; dir[i]; i++) {
+ if (dir[i] == '/' || dir[i] == '\\')
+ res++;
+ }
+
+ if (i > 0)
+ res++;
+
+ return res;
+}
+
+static void
restore (const gchar *filename)
{
gchar *command;
gchar *quotedfname;
+ gboolean is_new_format = FALSE;
g_return_if_fail (filename && *filename);
- if (!check (filename)) {
+ if (!check (filename, &is_new_format)) {
g_message ("Cannot restore from an incorrect archive '%s'.", filename);
-
- if (restart_arg) {
- CANCEL (complete);
- txt = _("Restarting Evolution");
- complete=TRUE;
- run_cmd (EVOLUTION);
- }
-
- return;
+ goto end;
}
quotedfname = g_shell_quote (filename);
@@ -267,51 +402,117 @@ restore (const gchar *filename)
CANCEL (complete);
txt = _("Back up current Evolution data");
- run_cmd ("mv " EVOLUTION_DIR " " EVOLUTION_DIR_BACKUP);
- run_cmd ("mv $HOME/.camel_certs ~/.camel_certs_old");
+ run_cmd ("mv $DATADIR $DATADIR_old");
+ run_cmd ("mv $CONFIGDIR $CONFIGDIR_old");
+ run_cmd ("mv $HOME/.camel_certs $HOME/.camel_certs_old");
CANCEL (complete);
txt = _("Extracting files from backup");
- command = g_strdup_printf ("cd $HOME && gzip -cd %s| tar xf -", quotedfname);
- run_cmd (command);
- g_free (command);
+
+ if (is_new_format) {
+ GString *dir_fn;
+ gchar *data_dir = NULL, *config_dir = NULL;
+
+ command = g_strdup_printf ("cd $TMP && tar xzf %s " EVOLUTION_DIR_FILE, quotedfname);
+ run_cmd (command);
+ g_free (command);
+
+ dir_fn = replace_variables ("$TMP" G_DIR_SEPARATOR_S EVOLUTION_DIR_FILE);
+ if (!dir_fn) {
+ g_warning ("Failed to create evolution's dir filename");
+ goto end;
+ }
+
+ /* data_dir and config_dir are quoted inside extract_backup_dirs */
+ extract_backup_dirs (dir_fn->str, &data_dir, &config_dir);
+
+ g_unlink (dir_fn->str);
+ g_string_free (dir_fn, TRUE);
+
+ if (!data_dir || !config_dir) {
+ g_warning ("Failed to get old data_dir (%p)/config_dir (%p)", data_dir, config_dir);
+ g_free (data_dir);
+ g_free (config_dir);
+ goto end;
+ }
+
+ g_mkdir_with_parents (e_get_user_data_dir (), 0700);
+ g_mkdir_with_parents (e_get_user_config_dir (), 0700);
+
+ command = g_strdup_printf ("cd $DATADIR && tar xzf %s %s --strip-components=%d", quotedfname, data_dir, get_dir_level (data_dir));
+ run_cmd (command);
+ g_free (command);
+
+ command = g_strdup_printf ("cd $CONFIGDIR && tar xzf %s %s --strip-components=%d", quotedfname, config_dir, get_dir_level (config_dir));
+ run_cmd (command);
+ g_free (command);
+
+ command = g_strdup_printf ("cd $HOME && tar xzf %s .camel_certs", quotedfname);
+ run_cmd (command);
+ g_free (command);
+
+ g_free (data_dir);
+ g_free (config_dir);
+ } else {
+ run_cmd ("mv $HOME/.evolution $HOME/.evolution_old");
+
+ command = g_strdup_printf ("cd $HOME && gzip -cd %s | tar xf -", quotedfname);
+ run_cmd (command);
+ g_free (command);
+ }
+
g_free (quotedfname);
CANCEL (complete);
txt = _("Loading Evolution settings");
- replace_in_file (GCONF_DUMP_PATH, EVOUSERDATADIR_MAGIC, e_get_user_data_dir ());
-
- run_cmd ("gconftool-2 --load " GCONF_DUMP_PATH);
+ if (is_new_format) {
+ /* new format has it in DATADIR... */
+ replace_in_file (EVOLUTION_DIR GCONF_DUMP_FILE, EVOUSERDATADIR_MAGIC, e_get_user_data_dir ());
+ run_cmd ("gconftool-2 --load " EVOLUTION_DIR GCONF_DUMP_FILE);
+ run_cmd ("rm " EVOLUTION_DIR GCONF_DUMP_FILE);
+ } else {
+ /* ... old format in ~/.evolution */
+ replace_in_file ("$HOME/.evolution/" GCONF_DUMP_FILE, EVOUSERDATADIR_MAGIC, e_get_user_data_dir ());
+ run_cmd ("gconftool-2 --load " "$HOME/.evolution/" GCONF_DUMP_FILE);
+ run_cmd ("rm " "$HOME/.evolution/" GCONF_DUMP_FILE);
+ }
CANCEL (complete);
txt = _("Removing temporary backup files");
- run_cmd ("rm -rf " GCONF_DUMP_PATH);
- run_cmd ("rm -rf " EVOLUTION_DIR_BACKUP);
+ run_cmd ("rm -rf $DATADIR_old");
+ run_cmd ("rm -rf $CONFIGDIR_old");
run_cmd ("rm -rf $HOME/.camel_certs_old");
- run_cmd ("rm $HOME/.evolution/.running");
+ run_cmd ("rm $DATADIR/.running");
+
+ if (!is_new_format)
+ run_cmd ("rm -rf $HOME/.evolution_old");
CANCEL (complete);
txt = _("Ensuring local sources");
+ end:
if (restart_arg) {
CANCEL (complete);
txt = _("Restarting Evolution");
complete=TRUE;
run_cmd (EVOLUTION);
}
-
}
static gboolean
-check (const gchar *filename)
+check (const gchar *filename, gboolean *is_new_format)
{
gchar *command;
gchar *quotedfname;
+ gboolean is_new = TRUE;
g_return_val_if_fail (filename && *filename, FALSE);
quotedfname = g_shell_quote (filename);
+ if (is_new_format)
+ *is_new_format = FALSE;
+
command = g_strdup_printf ("tar ztf %s 1>/dev/null", quotedfname);
result = system (command);
g_free (command);
@@ -322,16 +523,30 @@ check (const gchar *filename)
return FALSE;
}
- command = g_strdup_printf ("tar ztf %s | grep -e \"^\\.evolution/$\"", quotedfname);
+ command = g_strdup_printf ("tar ztf %s | grep -e \"%s$\"", quotedfname, EVOLUTION_DIR_FILE);
result = system (command);
g_free (command);
+ if (result) {
+ command = g_strdup_printf ("tar ztf %s | grep -e \"^\\.evolution/$\"", quotedfname);
+ result = system (command);
+ g_free (command);
+ is_new = FALSE;
+ }
+
g_message ("Second result %d", result);
if (result) {
g_free (quotedfname);
return FALSE;
}
+ if (is_new) {
+ if (is_new_format)
+ *is_new_format = TRUE;
+ g_free (quotedfname);
+ return TRUE;
+ }
+
command = g_strdup_printf ("tar ztf %s | grep -e \"^\\.evolution/%s$\"", quotedfname, GCONF_DUMP_FILE);
result = system (command);
g_free (command);
@@ -363,7 +578,7 @@ thread_start (gpointer data)
else if (restore_op)
restore (res_file);
else if (check_op)
- check (chk_file);
+ check (chk_file, NULL);
complete = TRUE;
@@ -582,7 +797,7 @@ main (gint argc, gchar **argv)
gtk_widget_show_all (progress_dialog);
} else if (check_op) {
/* For sanity we don't need gui */
- check (chk_file);
+ check (chk_file, NULL);
exit (result == 0 ? 0 : 1);
}