2 Bacula® - The Network Backup Solution
4 Copyright (C) 2000-2008 Free Software Foundation Europe e.V.
6 The main author of Bacula is Kern Sibbald, with contributions from
7 many others, a complete list can be found in the file AUTHORS.
8 This program is Free Software; you can redistribute it and/or
9 modify it under the terms of version two of the GNU General Public
10 License as published by the Free Software Foundation and included
13 This program is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
23 Bacula® is a registered trademark of Kern Sibbald.
24 The licensor of Bacula is the Free Software Foundation Europe
25 (FSFE), Fiduciary Program, Sumatrastrasse 25, 8006 Zürich,
26 Switzerland, email:ftf@fsfeurope.org.
30 * Bacula Director -- fd_cmds.c -- send commands to File daemon
32 * Kern Sibbald, October MM
34 * This routine is run as a separate thread. There may be more
35 * work to be done to make it totally reentrant!!!!
37 * Utility functions for sending info to File Daemon.
38 * These functions are used by both backup and verify.
45 #include "findlib/find.h"
47 const int dbglvl = 400;
49 /* Commands sent to File daemon */
50 static char filesetcmd[] = "fileset%s\n"; /* set full fileset */
51 static char jobcmd[] = "JobId=%s Job=%s SDid=%u SDtime=%u Authorization=%s\n";
52 /* Note, mtime_only is not used here -- implemented as file option */
53 static char levelcmd[] = "level = %s%s%s mtime_only=%d\n";
54 static char runscript[] = "Run OnSuccess=%u OnFailure=%u AbortOnError=%u When=%u Command=%s\n";
55 static char runbeforenow[]= "RunBeforeNow\n";
57 /* Responses received from File daemon */
58 static char OKinc[] = "2000 OK include\n";
59 static char OKjob[] = "2000 OK Job";
60 static char OKlevel[] = "2000 OK level\n";
61 static char OKRunScript[] = "2000 OK RunScript\n";
62 static char OKRunBeforeNow[] = "2000 OK RunBeforeNow\n";
64 /* Forward referenced functions */
65 static bool send_list_item(JCR *jcr, const char *code, char *item, BSOCK *fd);
67 /* External functions */
68 extern DIRRES *director;
69 extern int FDConnectTimeout;
75 * Open connection with File daemon.
76 * Try connecting every retry_interval (default 10 sec), and
77 * give up after max_retry_time (default 30 mins).
80 int connect_to_file_daemon(JCR *jcr, int retry_interval, int max_retry_time,
83 BSOCK *fd = new_bsock();
87 if (jcr->client->heartbeat_interval) {
88 heart_beat = jcr->client->heartbeat_interval;
90 heart_beat = director->heartbeat_interval;
93 if (!jcr->file_bsock) {
94 char name[MAX_NAME_LENGTH + 100];
95 bstrncpy(name, _("Client: "), sizeof(name));
96 bstrncat(name, jcr->client->name(), sizeof(name));
98 fd->set_source_address(director->DIRsrc_addr);
99 if (!fd->connect(jcr,retry_interval,max_retry_time, heart_beat, name, jcr->client->address,
100 NULL, jcr->client->FDport, verbose)) {
106 set_jcr_job_status(jcr, JS_ErrorTerminated);
109 Dmsg0(10, "Opened connection with File daemon\n");
111 fd = jcr->file_bsock; /* use existing connection */
113 fd->res = (RES *)jcr->client; /* save resource in BSOCK */
114 jcr->file_bsock = fd;
115 set_jcr_job_status(jcr, JS_Running);
117 if (!authenticate_file_daemon(jcr)) {
118 set_jcr_job_status(jcr, JS_ErrorTerminated);
123 * Now send JobId and authorization key
125 fd->fsend(jobcmd, edit_int64(jcr->JobId, ed1), jcr->Job, jcr->VolSessionId,
126 jcr->VolSessionTime, jcr->sd_auth_key);
127 if (strcmp(jcr->sd_auth_key, "dummy") != 0) {
128 memset(jcr->sd_auth_key, 0, strlen(jcr->sd_auth_key));
130 Dmsg1(100, ">filed: %s", fd->msg);
131 if (bget_dirmsg(fd) > 0) {
132 Dmsg1(110, "<filed: %s", fd->msg);
133 if (strncmp(fd->msg, OKjob, strlen(OKjob)) != 0) {
134 Jmsg(jcr, M_FATAL, 0, _("File daemon \"%s\" rejected Job command: %s\n"),
135 jcr->client->hdr.name, fd->msg);
136 set_jcr_job_status(jcr, JS_ErrorTerminated);
138 } else if (jcr->db) {
140 memset(&cr, 0, sizeof(cr));
141 bstrncpy(cr.Name, jcr->client->hdr.name, sizeof(cr.Name));
142 cr.AutoPrune = jcr->client->AutoPrune;
143 cr.FileRetention = jcr->client->FileRetention;
144 cr.JobRetention = jcr->client->JobRetention;
145 bstrncpy(cr.Uname, fd->msg+strlen(OKjob)+1, sizeof(cr.Uname));
146 if (!db_update_client_record(jcr, jcr->db, &cr)) {
147 Jmsg(jcr, M_WARNING, 0, _("Error updating Client record. ERR=%s\n"),
148 db_strerror(jcr->db));
152 Jmsg(jcr, M_FATAL, 0, _("FD gave bad response to JobId command: %s\n"),
154 set_jcr_job_status(jcr, JS_ErrorTerminated);
161 * This subroutine edits the last job start time into a
162 * "since=date/time" buffer that is returned in the
163 * variable since. This is used for display purposes in
164 * the job report. The time in jcr->stime is later
165 * passed to tell the File daemon what to do.
167 void get_level_since_time(JCR *jcr, char *since, int since_len)
171 bool do_full = false;
172 bool do_diff = false;
174 utime_t last_full_time;
175 utime_t last_diff_time;
178 /* If job cloned and a since time already given, use it */
179 if (jcr->cloned && jcr->stime && jcr->stime[0]) {
180 bstrncpy(since, _(", since="), since_len);
181 bstrncat(since, jcr->stime, since_len);
184 /* Make sure stime buffer is allocated */
186 jcr->stime = get_pool_memory(PM_MESSAGE);
190 * Lookup the last FULL backup job to get the time/date for a
191 * differential or incremental save.
193 switch (jcr->get_JobLevel()) {
196 POOLMEM *stime = get_pool_memory(PM_MESSAGE);
197 /* Look up start time of last Full job */
198 now = (utime_t)time(NULL);
199 jcr->jr.JobId = 0; /* flag to return since time */
200 have_full = db_find_job_start_time(jcr, jcr->db, &jcr->jr, &jcr->stime);
202 last_full_time = str_to_utime(jcr->stime);
204 do_full = true; /* No full, upgrade to one */
206 /* Make sure the last diff is recent enough */
207 if (have_full && jcr->get_JobLevel() == L_INCREMENTAL && jcr->job->MaxDiffInterval > 0) {
208 /* Lookup last diff job */
209 if (db_find_last_job_start_time(jcr, jcr->db, &jcr->jr, &stime, L_DIFFERENTIAL)) {
210 last_diff_time = str_to_utime(stime);
211 /* If no Diff since Full, use Full time */
212 if (last_diff_time < last_full_time) {
213 last_diff_time = last_full_time;
216 /* No last differential, so use last full time */
217 last_diff_time = last_full_time;
219 do_diff = ((now - last_diff_time) >= jcr->job->MaxDiffInterval);
221 /* Note, do_full takes precedence over do_diff */
222 if (have_full && jcr->job->MaxFullInterval > 0) {
223 do_full = ((now - last_full_time) >= jcr->job->MaxFullInterval);
225 free_pool_memory(stime);
228 /* No recent Full job found, so upgrade this one to Full */
229 Jmsg(jcr, M_INFO, 0, "%s", db_strerror(jcr->db));
230 Jmsg(jcr, M_INFO, 0, _("No prior or suitable Full backup found in catalog. Doing FULL backup.\n"));
231 bsnprintf(since, since_len, _(" (upgraded from %s)"),
232 level_to_str(jcr->get_JobLevel()));
233 jcr->set_JobLevel(jcr->jr.JobLevel = L_FULL);
234 } else if (do_diff) {
235 /* No recent diff job found, so upgrade this one to Diff */
236 Jmsg(jcr, M_INFO, 0, _("No prior or suitable Differential backup found in catalog. Doing Differential backup.\n"));
237 bsnprintf(since, since_len, _(" (upgraded from %s)"),
238 level_to_str(jcr->get_JobLevel()));
239 jcr->set_JobLevel(jcr->jr.JobLevel = L_DIFFERENTIAL);
241 if (jcr->job->rerun_failed_levels) {
242 if (db_find_failed_job_since(jcr, jcr->db, &jcr->jr, jcr->stime, JobLevel)) {
243 Jmsg(jcr, M_INFO, 0, _("Prior failed job found in catalog. Upgrading to %s.\n"),
244 level_to_str(JobLevel));
245 bsnprintf(since, since_len, _(" (upgraded from %s)"),
246 level_to_str(jcr->get_JobLevel()));
247 jcr->set_JobLevel(jcr->jr.JobLevel = JobLevel);
248 jcr->jr.JobId = jcr->JobId;
252 bstrncpy(since, _(", since="), since_len);
253 bstrncat(since, jcr->stime, since_len);
255 jcr->jr.JobId = jcr->JobId;
258 Dmsg2(100, "Level=%c last start time=%s\n", jcr->get_JobLevel(), jcr->stime);
261 static void send_since_time(JCR *jcr)
263 BSOCK *fd = jcr->file_bsock;
267 stime = str_to_utime(jcr->stime);
268 fd->fsend(levelcmd, "", NT_("since_utime "), edit_uint64(stime, ed1), 0);
269 while (bget_dirmsg(fd) >= 0) { /* allow him to poll us to sync clocks */
270 Jmsg(jcr, M_INFO, 0, "%s\n", fd->msg);
275 * Send level command to FD.
276 * Used for backup jobs and estimate command.
278 bool send_level_command(JCR *jcr)
280 BSOCK *fd = jcr->file_bsock;
281 const char *accurate = jcr->job->accurate?"accurate_":"";
282 const char *not_accurate = "";
284 * Send Level command to File daemon
286 switch (jcr->get_JobLevel()) {
288 fd->fsend(levelcmd, not_accurate, "base", " ", 0);
290 /* L_NONE is the console, sending something off to the FD */
293 fd->fsend(levelcmd, not_accurate, "full", " ", 0);
296 fd->fsend(levelcmd, accurate, "differential", " ", 0);
297 send_since_time(jcr);
300 fd->fsend(levelcmd, accurate, "incremental", " ", 0);
301 send_since_time(jcr);
305 Jmsg2(jcr, M_FATAL, 0, _("Unimplemented backup level %d %c\n"),
306 jcr->get_JobLevel(), jcr->get_JobLevel());
309 Dmsg1(120, ">filed: %s", fd->msg);
310 if (!response(jcr, fd, OKlevel, "Level", DISPLAY_ERROR)) {
317 * Send either an Included or an Excluded list to FD
319 static bool send_fileset(JCR *jcr)
321 FILESET *fileset = jcr->fileset;
322 BSOCK *fd = jcr->file_bsock;
328 num = fileset->num_includes;
330 num = fileset->num_excludes;
332 for (int i=0; i<num; i++) {
338 ie = fileset->include_items[i];
341 ie = fileset->exclude_items[i];
344 for (j=0; j<ie->num_opts; j++) {
345 FOPTS *fo = ie->opts_list[j];
346 fd->fsend("O %s\n", fo->opts);
348 bool enhanced_wild = false;
349 for (k=0; fo->opts[k]!='\0'; k++) {
350 if (fo->opts[k]=='W') {
351 enhanced_wild = true;
356 for (k=0; k<fo->regex.size(); k++) {
357 fd->fsend("R %s\n", fo->regex.get(k));
359 for (k=0; k<fo->regexdir.size(); k++) {
360 fd->fsend("RD %s\n", fo->regexdir.get(k));
362 for (k=0; k<fo->regexfile.size(); k++) {
363 fd->fsend("RF %s\n", fo->regexfile.get(k));
365 for (k=0; k<fo->wild.size(); k++) {
366 fd->fsend("W %s\n", fo->wild.get(k));
368 for (k=0; k<fo->wilddir.size(); k++) {
369 fd->fsend("WD %s\n", fo->wilddir.get(k));
371 for (k=0; k<fo->wildfile.size(); k++) {
372 fd->fsend("WF %s\n", fo->wildfile.get(k));
374 for (k=0; k<fo->wildbase.size(); k++) {
375 fd->fsend("W%c %s\n", enhanced_wild ? 'B' : 'F', fo->wildbase.get(k));
377 for (k=0; k<fo->base.size(); k++) {
378 fd->fsend("B %s\n", fo->base.get(k));
380 for (k=0; k<fo->fstype.size(); k++) {
381 fd->fsend("X %s\n", fo->fstype.get(k));
383 for (k=0; k<fo->drivetype.size(); k++) {
384 fd->fsend("XD %s\n", fo->drivetype.get(k));
387 fd->fsend("G %s\n", fo->plugin);
390 bnet_fsend(fd, "Z %s\n", fo->ignoredir);
393 fd->fsend("D %s\n", fo->reader);
396 fd->fsend("T %s\n", fo->writer);
401 for (j=0; j<ie->name_list.size(); j++) {
402 item = (char *)ie->name_list.get(j);
403 if (!send_list_item(jcr, "F ", item, fd)) {
408 for (j=0; j<ie->plugin_list.size(); j++) {
409 item = (char *)ie->plugin_list.get(j);
410 if (!send_list_item(jcr, "P ", item, fd)) {
416 if (!include) { /* If we just did excludes */
417 break; /* all done */
419 include = false; /* Now do excludes */
422 fd->signal(BNET_EOD); /* end of data */
423 if (!response(jcr, fd, OKinc, "Include", DISPLAY_ERROR)) {
429 set_jcr_job_status(jcr, JS_ErrorTerminated);
434 static bool send_list_item(JCR *jcr, const char *code, char *item, BSOCK *fd)
444 p++; /* skip over the | */
445 fd->msg = edit_job_codes(jcr, fd->msg, p, "");
446 bpipe = open_bpipe(fd->msg, 0, "r");
449 Jmsg(jcr, M_FATAL, 0, _("Cannot run program: %s. ERR=%s\n"),
453 bstrncpy(buf, code, sizeof(buf));
454 Dmsg1(500, "code=%s\n", buf);
455 optlen = strlen(buf);
456 while (fgets(buf+optlen, sizeof(buf)-optlen, bpipe->rfd)) {
457 fd->msglen = Mmsg(fd->msg, "%s", buf);
458 Dmsg2(500, "Inc/exc len=%d: %s", fd->msglen, fd->msg);
459 if (!bnet_send(fd)) {
460 Jmsg(jcr, M_FATAL, 0, _(">filed: write error on socket\n"));
464 if ((stat=close_bpipe(bpipe)) != 0) {
466 Jmsg(jcr, M_FATAL, 0, _("Error running program: %s. ERR=%s\n"),
467 p, be.bstrerror(stat));
472 p++; /* skip over < */
473 if ((ffd = fopen(p, "rb")) == NULL) {
475 Jmsg(jcr, M_FATAL, 0, _("Cannot open included file: %s. ERR=%s\n"),
479 bstrncpy(buf, code, sizeof(buf));
480 Dmsg1(500, "code=%s\n", buf);
481 optlen = strlen(buf);
482 while (fgets(buf+optlen, sizeof(buf)-optlen, ffd)) {
483 fd->msglen = Mmsg(fd->msg, "%s", buf);
484 if (!bnet_send(fd)) {
485 Jmsg(jcr, M_FATAL, 0, _(">filed: write error on socket\n"));
492 p++; /* skip over \ */
493 /* Note, fall through wanted */
495 pm_strcpy(fd->msg, code);
496 fd->msglen = pm_strcat(fd->msg, p);
497 Dmsg1(500, "Inc/Exc name=%s\n", fd->msg);
499 Jmsg(jcr, M_FATAL, 0, _(">filed: write error on socket\n"));
509 * Send include list to File daemon
511 bool send_include_list(JCR *jcr)
513 BSOCK *fd = jcr->file_bsock;
514 if (jcr->fileset->new_include) {
515 fd->fsend(filesetcmd, jcr->fileset->enable_vss ? " vss=1" : "");
516 return send_fileset(jcr);
523 * Send exclude list to File daemon
524 * Under the new scheme, the Exclude list
525 * is part of the FileSet sent with the
526 * "include_list" above.
528 bool send_exclude_list(JCR *jcr)
535 * Send bootstrap file if any to the socket given (FD or SD).
536 * This is used for restore, verify VolumeToCatalog, and
539 bool send_bootstrap_file(JCR *jcr, BSOCK *sock)
543 const char *bootstrap = "bootstrap\n";
545 Dmsg1(400, "send_bootstrap_file: %s\n", jcr->RestoreBootstrap);
546 if (!jcr->RestoreBootstrap) {
549 bs = fopen(jcr->RestoreBootstrap, "rb");
552 Jmsg(jcr, M_FATAL, 0, _("Could not open bootstrap file %s: ERR=%s\n"),
553 jcr->RestoreBootstrap, be.bstrerror());
554 set_jcr_job_status(jcr, JS_ErrorTerminated);
557 sock->fsend(bootstrap);
558 while (fgets(buf, sizeof(buf), bs)) {
559 sock->fsend("%s", buf);
561 sock->signal(BNET_EOD);
563 if (jcr->unlink_bsr) {
564 unlink(jcr->RestoreBootstrap);
565 jcr->unlink_bsr = false;
570 /* TODO: drop this with runscript.old_proto in bacula 1.42 */
571 static char runbefore[] = "RunBeforeJob %s\n";
572 static char runafter[] = "RunAfterJob %s\n";
573 static char OKRunBefore[] = "2000 OK RunBefore\n";
574 static char OKRunAfter[] = "2000 OK RunAfter\n";
576 int send_runscript_with_old_proto(JCR *jcr, int when, POOLMEM *msg)
579 Dmsg1(120, "bdird: sending old runcommand to fd '%s'\n",msg);
580 if (when & SCRIPT_Before) {
581 bnet_fsend(jcr->file_bsock, runbefore, msg);
582 ret = response(jcr, jcr->file_bsock, OKRunBefore, "ClientRunBeforeJob", DISPLAY_ERROR);
584 bnet_fsend(jcr->file_bsock, runafter, msg);
585 ret = response(jcr, jcr->file_bsock, OKRunAfter, "ClientRunAfterJob", DISPLAY_ERROR);
591 * Send RunScripts to File daemon
592 * 1) We send all runscript to FD, they can be executed Before, After, or twice
593 * 2) Then, we send a "RunBeforeNow" command to the FD to tell him to do the
594 * first run_script() call. (ie ClientRunBeforeJob)
596 int send_runscripts_commands(JCR *jcr)
598 POOLMEM *msg = get_pool_memory(PM_FNAME);
599 BSOCK *fd = jcr->file_bsock;
601 bool launch_before_cmd = false;
602 POOLMEM *ehost = get_pool_memory(PM_FNAME);
605 Dmsg0(120, "bdird: sending runscripts to fd\n");
607 foreach_alist(cmd, jcr->job->RunScripts) {
608 if (cmd->can_run_at_level(jcr->get_JobLevel()) && cmd->target) {
609 ehost = edit_job_codes(jcr, ehost, cmd->target, "");
610 Dmsg2(200, "bdird: runscript %s -> %s\n", cmd->target, ehost);
612 if (strcmp(ehost, jcr->client->name()) == 0) {
613 pm_strcpy(msg, cmd->command);
616 Dmsg1(120, "bdird: sending runscripts to fd '%s'\n", cmd->command);
618 /* TODO: remove this with bacula 1.42 */
619 if (cmd->old_proto) {
620 result = send_runscript_with_old_proto(jcr, cmd->when, msg);
623 fd->fsend(runscript, cmd->on_success,
629 result = response(jcr, fd, OKRunScript, "RunScript", DISPLAY_ERROR);
630 launch_before_cmd = true;
637 /* TODO : we have to play with other client */
640 send command to an other client
646 /* Tell the FD to execute the ClientRunBeforeJob */
647 if (launch_before_cmd) {
648 fd->fsend(runbeforenow);
649 if (!response(jcr, fd, OKRunBeforeNow, "RunBeforeNow", DISPLAY_ERROR)) {
653 free_pool_memory(msg);
654 free_pool_memory(ehost);
658 Jmsg(jcr, M_FATAL, 0, _("Client \"%s\" RunScript failed.\n"), ehost);
659 free_pool_memory(msg);
660 free_pool_memory(ehost);
667 * Read the attributes from the File daemon for
668 * a Verify job and store them in the catalog.
670 int get_attributes_and_put_in_catalog(JCR *jcr)
675 char digest[MAXSTRING];
677 fd = jcr->file_bsock;
678 jcr->jr.FirstIndex = 1;
680 /* Start transaction allocates jcr->attr and jcr->ar if needed */
681 db_start_transaction(jcr, jcr->db); /* start transaction if not already open */
684 Dmsg0(120, "bdird: waiting to receive file attributes\n");
685 /* Pickup file attributes and digest */
686 while (!fd->errors && (n = bget_dirmsg(fd)) > 0) {
690 char Digest[MAXSTRING]; /* either Verify opts or MD5/SHA1 digest */
692 if ((len = sscanf(fd->msg, "%ld %d %s", &file_index, &stream, Digest)) != 3) {
693 Jmsg(jcr, M_FATAL, 0, _("<filed: bad attributes, expected 3 fields got %d\n"
694 "msglen=%d msg=%s\n"), len, fd->msglen, fd->msg);
695 set_jcr_job_status(jcr, JS_ErrorTerminated);
699 /* The following three fields were sscanf'ed above so skip them */
700 skip_nonspaces(&p); /* skip FileIndex */
702 skip_nonspaces(&p); /* skip Stream */
704 skip_nonspaces(&p); /* skip Opts_Digest */
705 p++; /* skip space */
706 Dmsg1(dbglvl, "Stream=%d\n", stream);
707 if (stream == STREAM_UNIX_ATTRIBUTES || stream == STREAM_UNIX_ATTRIBUTES_EX) {
708 if (jcr->cached_attribute) {
709 Dmsg3(dbglvl, "Cached attr. Stream=%d fname=%s\n", ar->Stream, ar->fname,
711 if (!db_create_file_attributes_record(jcr, jcr->db, ar)) {
712 Jmsg1(jcr, M_FATAL, 0, _("Attribute create error. %s"), db_strerror(jcr->db));
715 /* Any cached attr is flushed so we can reuse jcr->attr and jcr->ar */
716 fn = jcr->fname = check_pool_memory_size(jcr->fname, fd->msglen);
718 *fn++ = *p++; /* copy filename */
720 *fn = *p++; /* term filename and point p to attribs */
721 pm_strcpy(jcr->attr, p); /* save attributes */
723 jcr->FileIndex = file_index;
724 ar->attr = jcr->attr;
725 ar->fname = jcr->fname;
726 ar->FileIndex = file_index;
729 ar->JobId = jcr->JobId;
730 ar->ClientId = jcr->ClientId;
734 ar->DigestType = CRYPTO_DIGEST_NONE;
735 jcr->cached_attribute = true;
737 Dmsg2(dbglvl, "dird<filed: stream=%d %s\n", stream, jcr->fname);
738 Dmsg1(dbglvl, "dird<filed: attr=%s\n", ar->attr);
739 jcr->FileId = ar->FileId;
741 * First, get STREAM_UNIX_ATTRIBUTES and fill ATTR_DBR structure
742 * Next, we CAN have a CRYPTO_DIGEST, so we fill ATTR_DBR with it (or not)
743 * When we get a new STREAM_UNIX_ATTRIBUTES, we known that we can add file to the catalog
744 * At the end, we have to add the last file
746 } else if (crypto_digest_stream_type(stream) != CRYPTO_DIGEST_NONE) {
747 if (jcr->FileIndex != (uint32_t)file_index) {
748 Jmsg3(jcr, M_ERROR, 0, _("%s index %d not same as attributes %d\n"),
749 stream_to_ascii(stream), file_index, jcr->FileIndex);
753 ar->DigestType = crypto_digest_stream_type(stream);
754 db_escape_string(jcr, jcr->db, digest, Digest, strlen(Digest));
755 Dmsg4(dbglvl, "stream=%d DigestLen=%d Digest=%s type=%d\n", stream,
756 strlen(digest), digest, ar->DigestType);
758 jcr->jr.JobFiles = jcr->JobFiles = file_index;
759 jcr->jr.LastIndex = file_index;
761 if (is_bnet_error(fd)) {
762 Jmsg1(jcr, M_FATAL, 0, _("<filed: Network error getting attributes. ERR=%s\n"),
766 if (jcr->cached_attribute) {
767 Dmsg3(dbglvl, "Cached attr with digest. Stream=%d fname=%s attr=%s\n", ar->Stream,
768 ar->fname, ar->attr);
769 if (!db_create_file_attributes_record(jcr, jcr->db, ar)) {
770 Jmsg1(jcr, M_FATAL, 0, _("Attribute create error. %s"), db_strerror(jcr->db));
772 jcr->cached_attribute = false;
774 set_jcr_job_status(jcr, JS_Terminated);