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
+ modify it under the terms of version three of the GNU Affero General Public
License as published by the Free Software Foundation and included
in the file LICENSE.
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
+ You should have received a copy of the GNU Affero General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
02110-1301, USA.
static bool dot_bvfs_lsdirs(UAContext *ua, const char *cmd);
static bool dot_bvfs_lsfiles(UAContext *ua, const char *cmd);
static bool dot_bvfs_update(UAContext *ua, const char *cmd);
+static bool dot_bvfs_get_jobids(UAContext *ua, const char *cmd);
+static bool dot_bvfs_versions(UAContext *ua, const char *cmd);
+static bool dot_bvfs_restore(UAContext *ua, const char *cmd);
+static bool dot_bvfs_cleanup(UAContext *ua, const char *cmd);
static bool api_cmd(UAContext *ua, const char *cmd);
static bool sql_cmd(UAContext *ua, const char *cmd);
static bool dot_quit_cmd(UAContext *ua, const char *cmd);
static bool dot_help_cmd(UAContext *ua, const char *cmd);
+static int one_handler(void *ctx, int num_field, char **row);
struct cmdstruct { const char *key; bool (*func)(UAContext *ua, const char *cmd); const char *help;const bool use_in_rs;};
static struct cmdstruct commands[] = { /* help */ /* can be used in runscript */
- { NT_(".api"), api_cmd, NULL, false},
- { NT_(".backups"), backupscmd, NULL, false},
- { NT_(".clients"), clientscmd, NULL, true},
- { NT_(".defaults"), defaultscmd, NULL, false},
- { NT_(".die"), admin_cmds, NULL, false},
- { NT_(".dump"), admin_cmds, NULL, false},
- { NT_(".exit"), admin_cmds, NULL, false},
- { NT_(".filesets"), filesetscmd, NULL, false},
- { NT_(".help"), dot_help_cmd, NULL, false},
- { NT_(".jobs"), jobscmd, NULL, true},
- { NT_(".levels"), levelscmd, NULL, false},
- { NT_(".messages"), getmsgscmd, NULL, false},
- { NT_(".msgs"), msgscmd, NULL, false},
- { NT_(".pools"), poolscmd, NULL, true},
- { NT_(".quit"), dot_quit_cmd, NULL, false},
- { NT_(".sql"), sql_cmd, NULL, false},
- { NT_(".status"), dot_status_cmd, NULL, false},
- { NT_(".storage"), storagecmd, NULL, true},
- { NT_(".volstatus"), volstatuscmd, NULL, true},
- { NT_(".media"), mediacmd, NULL, true},
- { NT_(".mediatypes"), mediatypescmd, NULL, true},
- { NT_(".locations"), locationscmd, NULL, true},
- { NT_(".actiononpurge"),aopcmd, NULL, true},
- { NT_(".bvfs_lsdirs"), dot_bvfs_lsdirs, NULL, true},
- { NT_(".bvfs_lsfiles"),dot_bvfs_lsfiles,NULL, true},
- { NT_(".bvfs_update"), dot_bvfs_update, NULL, true},
- { NT_(".types"), typescmd, NULL, false}
+ { NT_(".api"), api_cmd, NULL, false},
+ { NT_(".backups"), backupscmd, NULL, false},
+ { NT_(".clients"), clientscmd, NULL, true},
+ { NT_(".defaults"), defaultscmd, NULL, false},
+ { NT_(".die"), admin_cmds, NULL, false},
+ { NT_(".dump"), admin_cmds, NULL, false},
+ { NT_(".exit"), admin_cmds, NULL, false},
+ { NT_(".filesets"), filesetscmd, NULL, false},
+ { NT_(".help"), dot_help_cmd, NULL, false},
+ { NT_(".jobs"), jobscmd, NULL, true},
+ { NT_(".levels"), levelscmd, NULL, false},
+ { NT_(".messages"), getmsgscmd, NULL, false},
+ { NT_(".msgs"), msgscmd, NULL, false},
+ { NT_(".pools"), poolscmd, NULL, true},
+ { NT_(".quit"), dot_quit_cmd, NULL, false},
+ { NT_(".sql"), sql_cmd, NULL, false},
+ { NT_(".status"), dot_status_cmd, NULL, false},
+ { NT_(".storage"), storagecmd, NULL, true},
+ { NT_(".volstatus"), volstatuscmd, NULL, true},
+ { NT_(".media"), mediacmd, NULL, true},
+ { NT_(".mediatypes"), mediatypescmd, NULL, true},
+ { NT_(".locations"), locationscmd, NULL, true},
+ { NT_(".actiononpurge"),aopcmd, NULL, true},
+ { NT_(".bvfs_lsdirs"), dot_bvfs_lsdirs, NULL, true},
+ { NT_(".bvfs_lsfiles"),dot_bvfs_lsfiles, NULL, true},
+ { NT_(".bvfs_update"), dot_bvfs_update, NULL, true},
+ { NT_(".bvfs_get_jobids"), dot_bvfs_get_jobids, NULL, true},
+ { NT_(".bvfs_versions"), dot_bvfs_versions, NULL, true},
+ { NT_(".bvfs_restore"), dot_bvfs_restore, NULL, true},
+ { NT_(".bvfs_cleanup"), dot_bvfs_cleanup, NULL, true},
+ { NT_(".types"), typescmd, NULL, false}
};
#define comsize ((int)(sizeof(commands)/sizeof(struct cmdstruct)))
int pos = find_arg_with_value(ua, "jobid");
if (pos != -1 && is_a_number_list(ua->argv[pos])) {
- POOL_MEM jobids;
- pm_strcpy(jobids, ua->argv[pos]);
- bvfs_update_path_hierarchy_cache(ua->jcr, ua->db, jobids.c_str());
+ bvfs_update_path_hierarchy_cache(ua->jcr, ua->db, ua->argv[pos]);
} else {
/* update cache for all jobids */
bvfs_update_cache(ua->jcr, ua->db);
ua->send_msg("%s\t0\t%s\t%s\t%s\t%s\n", row[BVFS_PathId], fileid,
jobid, lstat, path);
+ } else if (bvfs_is_version(row)) {
+ ua->send_msg("%s\t%s\t%s\t%s\t%s\t%s\t%s\t%s\n", row[BVFS_PathId],
+ row[BVFS_FilenameId], fileid, jobid,
+ lstat, row[BVFS_Md5], row[BVFS_VolName],
+ row[BVFS_VolInchanger]);
+
} else if (bvfs_is_file(row)) {
ua->send_msg("%s\t%s\t%s\t%s\t%s\t%s\n", row[BVFS_PathId],
row[BVFS_FilenameId], fileid, jobid,
return 0;
}
+static bool bvfs_parse_arg_version(UAContext *ua,
+ char **client,
+ DBId_t *fnid,
+ bool *versions,
+ bool *copies)
+{
+ *fnid=0;
+ *client=NULL;
+ *versions=false;
+ *copies=false;
+
+ for (int i=1; i<ua->argc; i++) {
+ if (fnid && strcasecmp(ua->argk[i], NT_("fnid")) == 0) {
+ if (is_a_number(ua->argv[i])) {
+ *fnid = str_to_int64(ua->argv[i]);
+ }
+ }
+
+ if (strcasecmp(ua->argk[i], NT_("client")) == 0) {
+ *client = ua->argv[i];
+ }
+
+ if (copies && strcasecmp(ua->argk[i], NT_("copies")) == 0) {
+ *copies = true;
+ }
+
+ if (versions && strcasecmp(ua->argk[i], NT_("versions")) == 0) {
+ *versions = true;
+ }
+ }
+ return (*client && *fnid > 0);
+}
+
static bool bvfs_parse_arg(UAContext *ua,
DBId_t *pathid, char **path, char **jobid,
+ char **username,
int *limit, int *offset)
{
*pathid=0;
*offset=0;
*path=NULL;
*jobid=NULL;
+ *username=NULL;
for (int i=1; i<ua->argc; i++) {
if (strcasecmp(ua->argk[i], NT_("pathid")) == 0) {
*pathid = str_to_int64(ua->argv[i]);
}
}
+
if (strcasecmp(ua->argk[i], NT_("path")) == 0) {
*path = ua->argv[i];
}
+
+ if (strcasecmp(ua->argk[i], NT_("username")) == 0) {
+ *username = ua->argv[i];
+ }
if (strcasecmp(ua->argk[i], NT_("jobid")) == 0) {
if (is_a_number_list(ua->argv[i])) {
return true;
}
+/* .bvfs_cleanup path=b2XXXXX
+ */
+static bool dot_bvfs_cleanup(UAContext *ua, const char *cmd)
+{
+ int i;
+ if ((i = find_arg_with_value(ua, "path")) >= 0) {
+ open_client_db(ua);
+ Bvfs fs(ua->jcr, ua->db);
+ fs.drop_restore_list(ua->argv[i]);
+ }
+ return true;
+}
+
+/* .bvfs_restore path=b2XXXXX jobid=1,2 fileid=1,2 dirid=1,2 hardlink=1,2,3,4
+ */
+static bool dot_bvfs_restore(UAContext *ua, const char *cmd)
+{
+ DBId_t pathid=0;
+ int limit=2000, offset=0, i;
+ char *path=NULL, *jobid=NULL, *username=NULL;
+ char *empty = (char *)"";
+ char *fileid, *dirid, *hardlink, *id;
+ id = fileid = dirid = hardlink = empty;
+
+ if (!bvfs_parse_arg(ua, &pathid, &path, &jobid, &username,
+ &limit, &offset))
+ {
+ ua->error_msg("Can't find jobid, pathid or path argument\n");
+ return true; /* not enough param */
+ }
+
+ Bvfs fs(ua->jcr, ua->db);
+ fs.set_username(username);
+ fs.set_jobids(jobid);
+
+ if ((i = find_arg_with_value(ua, "fileid")) >= 0) {
+ fileid = ua->argv[i];
+ }
+ if ((i = find_arg_with_value(ua, "dirid")) >= 0) {
+ dirid = ua->argv[i];
+ }
+ if ((i = find_arg_with_value(ua, "hardlink")) >= 0) {
+ hardlink = ua->argv[i];
+ }
+
+ if (fs.compute_restore_list(fileid, dirid, hardlink, path)) {
+ ua->send_msg("OK\n");
+ } else {
+ ua->error_msg("Can't create restore list\n");
+ }
+ return true;
+}
+
/*
* .bvfs_lsfiles jobid=1,2,3,4 pathid=10
* .bvfs_lsfiles jobid=1,2,3,4 path=/
{
DBId_t pathid=0;
int limit=2000, offset=0;
- char *path=NULL, *jobid=NULL;
+ char *path=NULL, *jobid=NULL, *username=NULL;
+ char *pattern=NULL;
+ int i;
- if (!bvfs_parse_arg(ua, &pathid, &path, &jobid,
+ if (!bvfs_parse_arg(ua, &pathid, &path, &jobid, &username,
&limit, &offset))
{
ua->error_msg("Can't find jobid, pathid or path argument\n");
return true; /* not enough param */
}
+ if ((i = find_arg_with_value(ua, "pattern")) >= 0) {
+ pattern = ua->argv[i];
+ }
Bvfs fs(ua->jcr, ua->db);
+ fs.set_username(username);
fs.set_jobids(jobid);
fs.set_handler(bvfs_result_handler, ua);
fs.set_limit(limit);
-
+ if (pattern) {
+ fs.set_pattern(pattern);
+ }
if (pathid) {
fs.ch_dir(pathid);
} else {
fs.ch_dir(path);
}
-
+
fs.set_offset(offset);
fs.ls_files();
{
DBId_t pathid=0;
int limit=2000, offset=0;
- char *path=NULL, *jobid=NULL;
+ char *path=NULL, *jobid=NULL, *username=NULL;
- if (!bvfs_parse_arg(ua, &pathid, &path, &jobid,
+ if (!bvfs_parse_arg(ua, &pathid, &path, &jobid, &username,
&limit, &offset))
{
ua->error_msg("Can't find jobid, pathid or path argument\n");
}
Bvfs fs(ua->jcr, ua->db);
+ fs.set_username(username);
fs.set_jobids(jobid);
fs.set_limit(limit);
fs.set_handler(bvfs_result_handler, ua);
return true;
}
+/*
+ * .bvfs_versions jobid=x fnid=10 pathid=10 copies versions
+ * (jobid isn't used)
+ */
+static bool dot_bvfs_versions(UAContext *ua, const char *cmd)
+{
+ DBId_t pathid=0, fnid=0;
+ int limit=2000, offset=0;
+ char *path=NULL, *jobid=NULL, *client=NULL, *username=NULL;
+ bool copies=false, versions=false;
+ if (!bvfs_parse_arg(ua, &pathid, &path, &jobid, &username,
+ &limit, &offset))
+ {
+ ua->error_msg("Can't find jobid, pathid or path argument\n");
+ return true; /* not enough param */
+ }
+
+ if (!bvfs_parse_arg_version(ua, &client, &fnid, &versions, &copies))
+ {
+ ua->error_msg("Can't find client or fnid argument\n");
+ return true; /* not enough param */
+ }
+
+ Bvfs fs(ua->jcr, ua->db);
+ fs.set_limit(limit);
+ fs.set_see_all_versions(versions);
+ fs.set_see_copies(copies);
+ fs.set_handler(bvfs_result_handler, ua);
+ fs.set_offset(offset);
+ fs.get_all_file_versions(pathid, fnid, client);
+
+ return true;
+}
+
+/* .bvfs_get_jobids jobid=1
+ * -> returns needed jobids to restore
+ * .bvfs_get_jobids jobid=1 all
+ * -> returns needed jobids to restore with all filesets a JobId=1 time
+ */
+static bool dot_bvfs_get_jobids(UAContext *ua, const char *cmd)
+{
+ JOB_DBR jr;
+ db_list_ctx jobids, tempids;
+ int pos;
+ char ed1[50];
+ POOL_MEM query;
+ dbid_list ids; /* Store all FileSetIds for this client */
+
+ if (!open_client_db(ua)) {
+ return true;
+ }
+
+ memset(&jr, 0, sizeof(JOB_DBR));
+ if ((pos = find_arg_with_value(ua, "jobid")) >= 0) {
+ jr.JobId = str_to_int64(ua->argv[pos]);
+ }
+
+ if (!db_get_job_record(ua->jcr, ua->db, &jr)) {
+ ua->error_msg(_("Unable to get Job record for JobId=%s: ERR=%s\n"),
+ ua->cmd, db_strerror(ua->db));
+ return true;
+ }
+
+ /* If we have the "all" option, we do a search on all defined fileset
+ * for this client
+ */
+ if (find_arg(ua, "all") > 0) {
+ edit_int64(jr.ClientId, ed1);
+ Mmsg(query, uar_sel_filesetid, ed1);
+ db_get_query_dbids(ua->jcr, ua->db, query, ids);
+ } else {
+ ids.num_ids = 1;
+ ids.DBId[0] = jr.FileSetId;
+ }
+
+ jr.JobLevel = L_INCREMENTAL; /* Take Full+Diff+Incr */
+
+ /* Foreach different FileSet, we build a restore jobid list */
+ for (int i=0; i < ids.num_ids; i++) {
+ jr.FileSetId = ids.DBId[i];
+ if (!db_accurate_get_jobids(ua->jcr, ua->db, &jr, &tempids)) {
+ return true;
+ }
+ jobids.cat(tempids);
+ }
+
+ ua->send_msg("%s\n", jobids.list);
+ return true;
+}
+
static bool dot_quit_cmd(UAContext *ua, const char *cmd)
{
quit_cmd(ua, cmd);
/*
* Dummy routine for non-development version
*/
-static bool die_or_dump_cmd(UAContext *ua, const char *cmd)
+static bool admin_cmds(UAContext *ua, const char *cmd)
{
+ ua->error_msg(_("Unknown command: %s\n"), ua->argk[0]);
return true;
}