X-Git-Url: https://git.sur5r.net/?a=blobdiff_plain;f=bacula%2Fsrc%2Fdird%2Fua_restore.c;h=500c0c1aa7fe8ae28095840145a72ee52eb7938b;hb=6e637fce30a1fdbc2da43552f513f529db4d4e87;hp=b3ec5f420668fbfc6952c708f14b7aab94407911;hpb=45e39de44045f8ad6fc920792c13328dbe2b915b;p=bacula%2Fbacula diff --git a/bacula/src/dird/ua_restore.c b/bacula/src/dird/ua_restore.c index b3ec5f4206..500c0c1aa7 100644 --- a/bacula/src/dird/ua_restore.c +++ b/bacula/src/dird/ua_restore.c @@ -1,14 +1,14 @@ /* Bacula® - The Network Backup Solution - Copyright (C) 2002-2007 Free Software Foundation Europe e.V. + Copyright (C) 2002-2008 Free Software Foundation Europe e.V. The main author of Bacula is Kern Sibbald, with contributions from many others, a complete list can be found in the file AUTHORS. This program is Free Software; you can redistribute it and/or modify it under the terms of version two of the GNU General Public - License as published by the Free Software Foundation plus additions - that are listed in the file LICENSE. + License as published by the Free Software Foundation and included + in the file LICENSE. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of @@ -44,7 +44,6 @@ #include "bacula.h" #include "dird.h" - /* Imported functions */ extern void print_bsr(UAContext *ua, RBSR *bsr); @@ -59,7 +58,7 @@ static void free_name_list(NAME_LIST *name_list); static bool select_backups_before_date(UAContext *ua, RESTORE_CTX *rx, char *date); static bool build_directory_tree(UAContext *ua, RESTORE_CTX *rx); static void free_rx(RESTORE_CTX *rx); -static void split_path_and_filename(RESTORE_CTX *rx, char *fname); +static void split_path_and_filename(UAContext *ua, RESTORE_CTX *rx, char *fname); static int jobid_fileindex_handler(void *ctx, int num_fields, char **row); static bool insert_file_into_findex_list(UAContext *ua, RESTORE_CTX *rx, char *file, char *date); @@ -67,6 +66,7 @@ static bool insert_dir_into_findex_list(UAContext *ua, RESTORE_CTX *rx, char *di char *date); static void insert_one_file_or_dir(UAContext *ua, RESTORE_CTX *rx, char *date, bool dir); static int get_client_name(UAContext *ua, RESTORE_CTX *rx); +static int get_restore_client_name(UAContext *ua, RESTORE_CTX &rx); static int get_date(UAContext *ua, char *date, int date_len); static int restore_count_handler(void *ctx, int num_fields, char **row); static bool insert_table_into_findex_list(UAContext *ua, RESTORE_CTX *rx, char *table); @@ -83,6 +83,8 @@ int restore_cmd(UAContext *ua, const char *cmd) JCR *jcr = ua->jcr; char *escaped_bsr_name = NULL; char *escaped_where_name = NULL; + char *strip_prefix, *add_prefix, *add_suffix, *regexp; + strip_prefix = add_prefix = add_suffix = regexp = NULL; memset(&rx, 0, sizeof(rx)); rx.path = get_pool_memory(PM_FNAME); @@ -94,6 +96,46 @@ int restore_cmd(UAContext *ua, const char *cmd) i = find_arg_with_value(ua, "where"); if (i >= 0) { rx.where = ua->argv[i]; + } + + i = find_arg_with_value(ua, "strip_prefix"); + if (i >= 0) { + strip_prefix = ua->argv[i]; + } + + i = find_arg_with_value(ua, "add_prefix"); + if (i >= 0) { + add_prefix = ua->argv[i]; + } + + i = find_arg_with_value(ua, "add_suffix"); + if (i >= 0) { + add_suffix = ua->argv[i]; + } + + i = find_arg_with_value(ua, "regexwhere"); + if (i >= 0) { + rx.RegexWhere = ua->argv[i]; + } + + if (strip_prefix || add_suffix || add_prefix) { + int len = bregexp_get_build_where_size(strip_prefix, add_prefix, add_suffix); + regexp = (char *)bmalloc(len * sizeof(char)); + + bregexp_build_where(regexp, len, strip_prefix, add_prefix, add_suffix); + rx.RegexWhere = regexp; + } + + /* TODO: add acl for regexwhere ? */ + + if (rx.RegexWhere) { + if (!acl_access_ok(ua, Where_ACL, rx.RegexWhere)) { + ua->error_msg(_("\"RegexWhere\" specification not authorized.\n")); + goto bail_out; + } + } + + if (rx.where) { if (!acl_access_ok(ua, Where_ACL, rx.where)) { ua->error_msg(_("\"where\" specification not authorized.\n")); goto bail_out; @@ -182,29 +224,39 @@ int restore_cmd(UAContext *ua, const char *cmd) ua->error_msg(_("No Client resource found!\n")); goto bail_out; } + get_restore_client_name(ua, rx); escaped_bsr_name = escape_filename(jcr->RestoreBootstrap); - escaped_where_name = escape_filename(rx.where); /* Build run command */ - if (rx.where) { - if (!acl_access_ok(ua, Where_ACL, rx.where)) { - ua->error_msg(_("\"where\" specification not authorized.\n")); - goto bail_out; - } + if (rx.RegexWhere) { + escaped_where_name = escape_filename(rx.RegexWhere); + Mmsg(ua->cmd, + "run job=\"%s\" client=\"%s\" restoreclient=\"%s\" storage=\"%s\"" + " bootstrap=\"%s\" regexwhere=\"%s\" files=%u catalog=\"%s\"", + job->name(), rx.ClientName, rx.RestoreClientName, + rx.store?rx.store->name():"", + escaped_bsr_name ? escaped_bsr_name : jcr->RestoreBootstrap, + escaped_where_name ? escaped_where_name : rx.RegexWhere, + rx.selected_files, ua->catalog->name()); + } else if (rx.where) { + escaped_where_name = escape_filename(rx.where); Mmsg(ua->cmd, - "run job=\"%s\" client=\"%s\" storage=\"%s\" bootstrap=\"%s\"" - " where=\"%s\" files=%d catalog=\"%s\"", - job->name(), rx.ClientName, rx.store?rx.store->name():"", + "run job=\"%s\" client=\"%s\" restoreclient=\"%s\" storage=\"%s\"" + " bootstrap=\"%s\" where=\"%s\" files=%u catalog=\"%s\"", + job->name(), rx.ClientName, rx.RestoreClientName, + rx.store?rx.store->name():"", escaped_bsr_name ? escaped_bsr_name : jcr->RestoreBootstrap, escaped_where_name ? escaped_where_name : rx.where, rx.selected_files, ua->catalog->name()); + } else { Mmsg(ua->cmd, - "run job=\"%s\" client=\"%s\" storage=\"%s\" bootstrap=\"%s\"" - " files=%d catalog=\"%s\"", - job->name(), rx.ClientName, rx.store?rx.store->name():"", + "run job=\"%s\" client=\"%s\" restoreclient=\"%s\" storage=\"%s\"" + " bootstrap=\"%s\" files=%u catalog=\"%s\"", + job->name(), rx.ClientName, rx.RestoreClientName, + rx.store?rx.store->name():"", escaped_bsr_name ? escaped_bsr_name : jcr->RestoreBootstrap, rx.selected_files, ua->catalog->name()); } @@ -216,6 +268,10 @@ int restore_cmd(UAContext *ua, const char *cmd) if (escaped_where_name != NULL) { bfree(escaped_where_name); } + + if (regexp) { + bfree(regexp); + } if (find_arg(ua, NT_("yes")) > 0) { pm_strcat(ua->cmd, " yes"); /* pass it on to the run command */ @@ -235,6 +291,10 @@ bail_out: bfree(escaped_where_name); } + if (regexp) { + bfree(regexp); + } + free_rx(&rx); return 0; @@ -272,6 +332,9 @@ static bool has_value(UAContext *ua, int i) return true; } +/* + * This gets the client name from which the backup was made + */ static int get_client_name(UAContext *ua, RESTORE_CTX *rx) { /* If no client name specified yet, get it now */ @@ -279,6 +342,9 @@ static int get_client_name(UAContext *ua, RESTORE_CTX *rx) CLIENT_DBR cr; /* try command line argument */ int i = find_arg_with_value(ua, NT_("client")); + if (i < 0) { + i = find_arg_with_value(ua, NT_("backupclient")); + } if (i >= 0) { if (!has_value(ua, i)) { return 0; @@ -295,6 +361,27 @@ static int get_client_name(UAContext *ua, RESTORE_CTX *rx) return 1; } +/* + * This is where we pick up a client name to restore to. + */ +static int get_restore_client_name(UAContext *ua, RESTORE_CTX &rx) +{ + /* Start with same name as backup client */ + bstrncpy(rx.RestoreClientName, rx.ClientName, sizeof(rx.RestoreClientName)); + + /* try command line argument */ + int i = find_arg_with_value(ua, NT_("restoreclient")); + if (i >= 0) { + if (!has_value(ua, i)) { + return 0; + } + bstrncpy(rx.RestoreClientName, ua->argv[i], sizeof(rx.RestoreClientName)); + return 1; + } + return 1; +} + + /* * The first step in the restore process is for the user to @@ -333,23 +420,28 @@ static int user_select_jobids_or_files(UAContext *ua, RESTORE_CTX *rx) const char *kw[] = { /* These keywords are handled in a for loop */ - "jobid", /* 0 */ - "current", /* 1 */ - "before", /* 2 */ - "file", /* 3 */ - "directory", /* 4 */ - "select", /* 5 */ - "pool", /* 6 */ - "all", /* 7 */ + "jobid", /* 0 */ + "current", /* 1 */ + "before", /* 2 */ + "file", /* 3 */ + "directory", /* 4 */ + "select", /* 5 */ + "pool", /* 6 */ + "all", /* 7 */ /* The keyword below are handled by individual arg lookups */ - "client", /* 8 */ - "storage", /* 9 */ - "fileset", /* 10 */ - "where", /* 11 */ - "yes", /* 12 */ - "bootstrap", /* 13 */ - "done", /* 14 */ + "client", /* 8 */ + "storage", /* 9 */ + "fileset", /* 10 */ + "where", /* 11 */ + "yes", /* 12 */ + "bootstrap", /* 13 */ + "done", /* 14 */ + "strip_prefix", /* 15 */ + "add_prefix", /* 16 */ + "add_suffix", /* 17 */ + "regexwhere", /* 18 */ + "restoreclient", /* 19 */ NULL }; @@ -389,7 +481,7 @@ static int user_select_jobids_or_files(UAContext *ua, RESTORE_CTX *rx) have_date = true; break; case 2: /* before */ - if (!has_value(ua, i)) { + if (have_date || !has_value(ua, i)) { return 0; } if (str_to_utime(ua->argv[i]) == 0) { @@ -489,8 +581,8 @@ static int user_select_jobids_or_files(UAContext *ua, RESTORE_CTX *rx) } len = strlen(ua->cmd); fname = (char *)malloc(len * 2 + 1); - db_escape_string(fname, ua->cmd, len); - Mmsg(rx->query, uar_file, rx->ClientName, fname); + db_escape_string(ua->jcr, ua->db, fname, ua->cmd, len); + Mmsg(rx->query, uar_file[db_type], rx->ClientName, fname); free(fname); gui_save = ua->jcr->gui; ua->jcr->gui = true; @@ -519,21 +611,27 @@ static int user_select_jobids_or_files(UAContext *ua, RESTORE_CTX *rx) done = false; break; case 4: /* Select the most recent backups */ - bstrutime(date, sizeof(date), now); + if (!have_date) { + bstrutime(date, sizeof(date), now); + } if (!select_backups_before_date(ua, rx, date)) { return 0; } break; case 5: /* select backup at specified time */ - if (!get_date(ua, date, sizeof(date))) { - return 0; + if (!have_date) { + if (!get_date(ua, date, sizeof(date))) { + return 0; + } } if (!select_backups_before_date(ua, rx, date)) { return 0; } break; case 6: /* Enter files */ - bstrutime(date, sizeof(date), now); + if (!have_date) { + bstrutime(date, sizeof(date), now); + } if (!get_client_name(ua, rx)) { return 0; } @@ -552,8 +650,10 @@ static int user_select_jobids_or_files(UAContext *ua, RESTORE_CTX *rx) } return 2; case 7: /* enter files backed up before specified time */ - if (!get_date(ua, date, sizeof(date))) { - return 0; + if (!have_date) { + if (!get_date(ua, date, sizeof(date))) { + return 0; + } } if (!get_client_name(ua, rx)) { return 0; @@ -574,7 +674,9 @@ static int user_select_jobids_or_files(UAContext *ua, RESTORE_CTX *rx) return 2; case 8: /* Find JobIds for current backup */ - bstrutime(date, sizeof(date), now); + if (!have_date) { + bstrutime(date, sizeof(date), now); + } if (!select_backups_before_date(ua, rx, date)) { return 0; } @@ -582,8 +684,10 @@ static int user_select_jobids_or_files(UAContext *ua, RESTORE_CTX *rx) break; case 9: /* Find JobIds for give date */ - if (!get_date(ua, date, sizeof(date))) { - return 0; + if (!have_date) { + if (!get_date(ua, date, sizeof(date))) { + return 0; + } } if (!select_backups_before_date(ua, rx, date)) { return 0; @@ -604,7 +708,9 @@ static int user_select_jobids_or_files(UAContext *ua, RESTORE_CTX *rx) if (*rx->JobIds == 0 || *rx->JobIds == '.') { return 0; /* nothing entered, return */ } - bstrutime(date, sizeof(date), now); + if (!have_date) { + bstrutime(date, sizeof(date), now); + } if (!get_client_name(ua, rx)) { return 0; } @@ -722,7 +828,7 @@ static void insert_one_file_or_dir(UAContext *ua, RESTORE_CTX *rx, char *date, b if ((ffd = fopen(p, "rb")) == NULL) { berrno be; ua->error_msg(_("Cannot open file %s: ERR=%s\n"), - p, be.strerror()); + p, be.bstrerror()); break; } while (fgets(file, sizeof(file), ffd)) { @@ -762,7 +868,7 @@ static bool insert_file_into_findex_list(UAContext *ua, RESTORE_CTX *rx, char *f char *date) { strip_trailing_newline(file); - split_path_and_filename(rx, file); + split_path_and_filename(ua, rx, file); if (*rx->JobIds == 0) { Mmsg(rx->query, uar_jobid_fileindex, date, rx->path, rx->fname, rx->ClientName); @@ -790,14 +896,13 @@ static bool insert_file_into_findex_list(UAContext *ua, RESTORE_CTX *rx, char *f */ static bool insert_dir_into_findex_list(UAContext *ua, RESTORE_CTX *rx, char *dir, char *date) -{ +{ strip_trailing_junk(dir); if (*rx->JobIds == 0) { ua->error_msg(_("No JobId specified cannot continue.\n")); return false; } else { - Mmsg(rx->query, uar_jobid_fileindex_from_dir, rx->JobIds, - dir, rx->ClientName); + Mmsg(rx->query, uar_jobid_fileindex_from_dir[db_type], rx->JobIds, dir, rx->ClientName); } rx->found = false; /* Find and insert jobid and File Index */ @@ -833,7 +938,7 @@ static bool insert_table_into_findex_list(UAContext *ua, RESTORE_CTX *rx, char * return true; } -static void split_path_and_filename(RESTORE_CTX *rx, char *name) +static void split_path_and_filename(UAContext *ua, RESTORE_CTX *rx, char *name) { char *p, *f; @@ -861,9 +966,8 @@ static void split_path_and_filename(RESTORE_CTX *rx, char *name) */ rx->fnl = p - f; if (rx->fnl > 0) { - rx->fname = check_pool_memory_size(rx->fname, rx->fnl+1); - memcpy(rx->fname, f, rx->fnl); /* copy filename */ - rx->fname[rx->fnl] = 0; + rx->fname = check_pool_memory_size(rx->fname, 2*(rx->fnl)+1); + db_escape_string(ua->jcr, ua->db, rx->fname, f, rx->fnl); } else { rx->fname[0] = 0; rx->fnl = 0; @@ -871,9 +975,8 @@ static void split_path_and_filename(RESTORE_CTX *rx, char *name) rx->pnl = f - name; if (rx->pnl > 0) { - rx->path = check_pool_memory_size(rx->path, rx->pnl+1); - memcpy(rx->path, name, rx->pnl); - rx->path[rx->pnl] = 0; + rx->path = check_pool_memory_size(rx->path, 2*(rx->pnl)+1); + db_escape_string(ua->jcr, ua->db, rx->path, name, rx->pnl); } else { rx->path[0] = 0; rx->pnl = 0; @@ -902,7 +1005,6 @@ static bool build_directory_tree(UAContext *ua, RESTORE_CTX *rx) * For display purposes, the same JobId, with different volumes may * appear more than once, however, we only insert it once. */ - int items = 0; p = rx->JobIds; tree.FileEstimate = 0; if (get_next_jobid_from_list(&p, &JobId) > 0) { @@ -917,23 +1019,12 @@ static bool build_directory_tree(UAContext *ua, RESTORE_CTX *rx) tree.DeltaCount = rx->JobId/50; /* print 50 ticks */ } } - for (p=rx->JobIds; get_next_jobid_from_list(&p, &JobId) > 0; ) { - char ed1[50]; - if (JobId == last_JobId) { - continue; /* eliminate duplicate JobIds */ - } - last_JobId = JobId; - ua->info_msg(_("\nBuilding directory tree for JobId %s ... "), - edit_int64(JobId, ed1)); - items++; - /* - * Find files for this JobId and insert them in the tree - */ - Mmsg(rx->query, uar_sel_files, edit_int64(JobId, ed1)); - if (!db_sql_query(ua->db, rx->query, insert_tree_handler, (void *)&tree)) { - ua->error_msg("%s", db_strerror(ua->db)); - } + ua->info_msg(_("\nBuilding directory tree for JobId(s) %s ... "), + rx->JobIds); + + if (!db_get_file_list(ua->jcr, ua->db, rx->JobIds, insert_tree_handler, (void *)&tree)) { + ua->error_msg("%s", db_strerror(ua->db)); } if (tree.FileCount == 0) { ua->send_msg(_("\nThere were no files inserted into the tree, so file selection\n" @@ -952,25 +1043,12 @@ static bool build_directory_tree(UAContext *ua, RESTORE_CTX *rx) } } else { char ec1[50]; - if (items==1) { - if (tree.all) { - ua->info_msg(_("\n1 Job, %s files inserted into the tree and marked for extraction.\n"), - edit_uint64_with_commas(tree.FileCount, ec1)); - } - else { - ua->info_msg(_("\n1 Job, %s files inserted into the tree.\n"), - edit_uint64_with_commas(tree.FileCount, ec1)); - } - } - else { - if (tree.all) { - ua->info_msg(_("\n%d Jobs, %s files inserted into the tree and marked for extraction.\n"), - items, edit_uint64_with_commas(tree.FileCount, ec1)); - } - else { - ua->info_msg(_("\n%d Jobs, %s files inserted into the tree.\n"), - items, edit_uint64_with_commas(tree.FileCount, ec1)); - } + if (tree.all) { + ua->info_msg(_("\n%s files inserted into the tree and marked for extraction.\n"), + edit_uint64_with_commas(tree.FileCount, ec1)); + } else { + ua->info_msg(_("\n%s files inserted into the tree.\n"), + edit_uint64_with_commas(tree.FileCount, ec1)); } if (find_arg(ua, NT_("done")) < 0) { @@ -1015,14 +1093,13 @@ static bool select_backups_before_date(UAContext *ua, RESTORE_CTX *rx, char *dat char pool_select[MAX_NAME_LENGTH]; int i; - /* Create temp tables */ db_sql_query(ua->db, uar_del_temp, NULL, NULL); db_sql_query(ua->db, uar_del_temp1, NULL, NULL); - if (!db_sql_query(ua->db, uar_create_temp, NULL, NULL)) { + if (!db_sql_query(ua->db, uar_create_temp[db_type], NULL, NULL)) { ua->error_msg("%s\n", db_strerror(ua->db)); } - if (!db_sql_query(ua->db, uar_create_temp1, NULL, NULL)) { + if (!db_sql_query(ua->db, uar_create_temp1[db_type], NULL, NULL)) { ua->error_msg("%s\n", db_strerror(ua->db)); } /* @@ -1096,6 +1173,7 @@ static bool select_backups_before_date(UAContext *ua, RESTORE_CTX *rx, char *dat ua->error_msg("%s\n", db_strerror(ua->db)); goto bail_out; } + /* Note, this is needed because I don't seem to get the callback * from the call just above. */ @@ -1163,11 +1241,12 @@ bail_out: */ int get_next_jobid_from_list(char **p, JobId_t *JobId) { - char jobid[30]; + const int maxlen = 30; + char jobid[maxlen+1]; char *q = *p; jobid[0] = 0; - for (int i=0; i<(int)sizeof(jobid); i++) { + for (int i=0; iJobId = str_to_int64(row[0]); add_findex(rx->bsr, rx->JobId, str_to_int64(row[1])); rx->found = true; @@ -1330,6 +1411,8 @@ void find_storage_resource(UAContext *ua, RESTORE_CTX &rx, char *Storage, char * /* Take command line arg, or ask user if none */ rx.store = get_storage_resource(ua, false /* don't use default */); - Dmsg1(200, "Set store=%s\n", rx.store->name()); + if (rx.store) { + Dmsg1(200, "Set store=%s\n", rx.store->name()); + } }