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 /* Make sure the last diff is recent enough */
205 if (have_full && jcr->get_JobLevel() == L_INCREMENTAL && jcr->job->MaxDiffInterval > 0) {
206 /* Lookup last diff job */
207 if (db_find_last_job_start_time(jcr, jcr->db, &jcr->jr, &stime, L_DIFFERENTIAL)) {
208 last_diff_time = str_to_utime(stime);
210 /* No last differential, so use last full time */
211 last_diff_time = last_full_time;
213 do_diff = ((now - last_diff_time) >= jcr->job->MaxDiffInterval);
215 /* Note, do_full takes precedence over do_diff */
216 if (have_full && jcr->job->MaxFullInterval > 0) {
217 do_full = ((now - last_full_time) >= jcr->job->MaxFullInterval);
221 free_pool_memory(stime);
224 /* No recent Full job found, so upgrade this one to Full */
225 Jmsg(jcr, M_INFO, 0, "%s", db_strerror(jcr->db));
226 Jmsg(jcr, M_INFO, 0, _("No prior or suitable Full backup found in catalog. Doing FULL backup.\n"));
227 bsnprintf(since, since_len, _(" (upgraded from %s)"),
228 level_to_str(jcr->get_JobLevel()));
229 jcr->set_JobLevel(jcr->jr.JobLevel = L_FULL);
230 } else if (do_diff) {
231 /* No recent diff job found, so upgrade this one to Diff */
232 Jmsg(jcr, M_INFO, 0, _("No prior or suitable Differential backup found in catalog. Doing Differential backup.\n"));
233 bsnprintf(since, since_len, _(" (upgraded from %s)"),
234 level_to_str(jcr->get_JobLevel()));
235 jcr->set_JobLevel(jcr->jr.JobLevel = L_DIFFERENTIAL);
237 if (jcr->job->rerun_failed_levels) {
238 if (db_find_failed_job_since(jcr, jcr->db, &jcr->jr, jcr->stime, JobLevel)) {
239 Jmsg(jcr, M_INFO, 0, _("Prior failed job found in catalog. Upgrading to %s.\n"),
240 level_to_str(JobLevel));
241 bsnprintf(since, since_len, _(" (upgraded from %s)"),
242 level_to_str(jcr->get_JobLevel()));
243 jcr->set_JobLevel(jcr->jr.JobLevel = JobLevel);
244 jcr->jr.JobId = jcr->JobId;
248 bstrncpy(since, _(", since="), since_len);
249 bstrncat(since, jcr->stime, since_len);
251 jcr->jr.JobId = jcr->JobId;
254 Dmsg2(100, "Level=%c last start time=%s\n", jcr->get_JobLevel(), jcr->stime);
257 static void send_since_time(JCR *jcr)
259 BSOCK *fd = jcr->file_bsock;
263 stime = str_to_utime(jcr->stime);
264 fd->fsend(levelcmd, "", NT_("since_utime "), edit_uint64(stime, ed1), 0);
265 while (bget_dirmsg(fd) >= 0) { /* allow him to poll us to sync clocks */
266 Jmsg(jcr, M_INFO, 0, "%s\n", fd->msg);
271 * Send level command to FD.
272 * Used for backup jobs and estimate command.
274 bool send_level_command(JCR *jcr)
276 BSOCK *fd = jcr->file_bsock;
277 const char *accurate = jcr->job->accurate?"accurate_":"";
278 const char *not_accurate = "";
280 * Send Level command to File daemon
282 switch (jcr->get_JobLevel()) {
284 fd->fsend(levelcmd, not_accurate, "base", " ", 0);
286 /* L_NONE is the console, sending something off to the FD */
289 fd->fsend(levelcmd, not_accurate, "full", " ", 0);
292 fd->fsend(levelcmd, accurate, "differential", " ", 0);
293 send_since_time(jcr);
296 fd->fsend(levelcmd, accurate, "incremental", " ", 0);
297 send_since_time(jcr);
301 Jmsg2(jcr, M_FATAL, 0, _("Unimplemented backup level %d %c\n"),
302 jcr->get_JobLevel(), jcr->get_JobLevel());
305 Dmsg1(120, ">filed: %s", fd->msg);
306 if (!response(jcr, fd, OKlevel, "Level", DISPLAY_ERROR)) {
313 * Send either an Included or an Excluded list to FD
315 static bool send_fileset(JCR *jcr)
317 FILESET *fileset = jcr->fileset;
318 BSOCK *fd = jcr->file_bsock;
324 num = fileset->num_includes;
326 num = fileset->num_excludes;
328 for (int i=0; i<num; i++) {
334 ie = fileset->include_items[i];
337 ie = fileset->exclude_items[i];
340 for (j=0; j<ie->num_opts; j++) {
341 FOPTS *fo = ie->opts_list[j];
342 fd->fsend("O %s\n", fo->opts);
344 bool enhanced_wild = false;
345 for (k=0; fo->opts[k]!='\0'; k++) {
346 if (fo->opts[k]=='W') {
347 enhanced_wild = true;
352 for (k=0; k<fo->regex.size(); k++) {
353 fd->fsend("R %s\n", fo->regex.get(k));
355 for (k=0; k<fo->regexdir.size(); k++) {
356 fd->fsend("RD %s\n", fo->regexdir.get(k));
358 for (k=0; k<fo->regexfile.size(); k++) {
359 fd->fsend("RF %s\n", fo->regexfile.get(k));
361 for (k=0; k<fo->wild.size(); k++) {
362 fd->fsend("W %s\n", fo->wild.get(k));
364 for (k=0; k<fo->wilddir.size(); k++) {
365 fd->fsend("WD %s\n", fo->wilddir.get(k));
367 for (k=0; k<fo->wildfile.size(); k++) {
368 fd->fsend("WF %s\n", fo->wildfile.get(k));
370 for (k=0; k<fo->wildbase.size(); k++) {
371 fd->fsend("W%c %s\n", enhanced_wild ? 'B' : 'F', fo->wildbase.get(k));
373 for (k=0; k<fo->base.size(); k++) {
374 fd->fsend("B %s\n", fo->base.get(k));
376 for (k=0; k<fo->fstype.size(); k++) {
377 fd->fsend("X %s\n", fo->fstype.get(k));
379 for (k=0; k<fo->drivetype.size(); k++) {
380 fd->fsend("XD %s\n", fo->drivetype.get(k));
383 fd->fsend("G %s\n", fo->plugin);
386 bnet_fsend(fd, "Z %s\n", fo->ignoredir);
389 fd->fsend("D %s\n", fo->reader);
392 fd->fsend("T %s\n", fo->writer);
397 for (j=0; j<ie->name_list.size(); j++) {
398 item = (char *)ie->name_list.get(j);
399 if (!send_list_item(jcr, "F ", item, fd)) {
404 for (j=0; j<ie->plugin_list.size(); j++) {
405 item = (char *)ie->plugin_list.get(j);
406 if (!send_list_item(jcr, "P ", item, fd)) {
412 if (!include) { /* If we just did excludes */
413 break; /* all done */
415 include = false; /* Now do excludes */
418 fd->signal(BNET_EOD); /* end of data */
419 if (!response(jcr, fd, OKinc, "Include", DISPLAY_ERROR)) {
425 set_jcr_job_status(jcr, JS_ErrorTerminated);
430 static bool send_list_item(JCR *jcr, const char *code, char *item, BSOCK *fd)
440 p++; /* skip over the | */
441 fd->msg = edit_job_codes(jcr, fd->msg, p, "");
442 bpipe = open_bpipe(fd->msg, 0, "r");
445 Jmsg(jcr, M_FATAL, 0, _("Cannot run program: %s. ERR=%s\n"),
449 bstrncpy(buf, code, sizeof(buf));
450 Dmsg1(500, "code=%s\n", buf);
451 optlen = strlen(buf);
452 while (fgets(buf+optlen, sizeof(buf)-optlen, bpipe->rfd)) {
453 fd->msglen = Mmsg(fd->msg, "%s", buf);
454 Dmsg2(500, "Inc/exc len=%d: %s", fd->msglen, fd->msg);
455 if (!bnet_send(fd)) {
456 Jmsg(jcr, M_FATAL, 0, _(">filed: write error on socket\n"));
460 if ((stat=close_bpipe(bpipe)) != 0) {
462 Jmsg(jcr, M_FATAL, 0, _("Error running program: %s. ERR=%s\n"),
463 p, be.bstrerror(stat));
468 p++; /* skip over < */
469 if ((ffd = fopen(p, "rb")) == NULL) {
471 Jmsg(jcr, M_FATAL, 0, _("Cannot open included file: %s. ERR=%s\n"),
475 bstrncpy(buf, code, sizeof(buf));
476 Dmsg1(500, "code=%s\n", buf);
477 optlen = strlen(buf);
478 while (fgets(buf+optlen, sizeof(buf)-optlen, ffd)) {
479 fd->msglen = Mmsg(fd->msg, "%s", buf);
480 if (!bnet_send(fd)) {
481 Jmsg(jcr, M_FATAL, 0, _(">filed: write error on socket\n"));
488 p++; /* skip over \ */
489 /* Note, fall through wanted */
491 pm_strcpy(fd->msg, code);
492 fd->msglen = pm_strcat(fd->msg, p);
493 Dmsg1(500, "Inc/Exc name=%s\n", fd->msg);
495 Jmsg(jcr, M_FATAL, 0, _(">filed: write error on socket\n"));
505 * Send include list to File daemon
507 bool send_include_list(JCR *jcr)
509 BSOCK *fd = jcr->file_bsock;
510 if (jcr->fileset->new_include) {
511 fd->fsend(filesetcmd, jcr->fileset->enable_vss ? " vss=1" : "");
512 return send_fileset(jcr);
519 * Send exclude list to File daemon
520 * Under the new scheme, the Exclude list
521 * is part of the FileSet sent with the
522 * "include_list" above.
524 bool send_exclude_list(JCR *jcr)
531 * Send bootstrap file if any to the socket given (FD or SD).
532 * This is used for restore, verify VolumeToCatalog, and
535 bool send_bootstrap_file(JCR *jcr, BSOCK *sock)
539 const char *bootstrap = "bootstrap\n";
541 Dmsg1(400, "send_bootstrap_file: %s\n", jcr->RestoreBootstrap);
542 if (!jcr->RestoreBootstrap) {
545 bs = fopen(jcr->RestoreBootstrap, "rb");
548 Jmsg(jcr, M_FATAL, 0, _("Could not open bootstrap file %s: ERR=%s\n"),
549 jcr->RestoreBootstrap, be.bstrerror());
550 set_jcr_job_status(jcr, JS_ErrorTerminated);
553 sock->fsend(bootstrap);
554 while (fgets(buf, sizeof(buf), bs)) {
555 sock->fsend("%s", buf);
557 sock->signal(BNET_EOD);
559 if (jcr->unlink_bsr) {
560 unlink(jcr->RestoreBootstrap);
561 jcr->unlink_bsr = false;
566 /* TODO: drop this with runscript.old_proto in bacula 1.42 */
567 static char runbefore[] = "RunBeforeJob %s\n";
568 static char runafter[] = "RunAfterJob %s\n";
569 static char OKRunBefore[] = "2000 OK RunBefore\n";
570 static char OKRunAfter[] = "2000 OK RunAfter\n";
572 int send_runscript_with_old_proto(JCR *jcr, int when, POOLMEM *msg)
575 Dmsg1(120, "bdird: sending old runcommand to fd '%s'\n",msg);
576 if (when & SCRIPT_Before) {
577 bnet_fsend(jcr->file_bsock, runbefore, msg);
578 ret = response(jcr, jcr->file_bsock, OKRunBefore, "ClientRunBeforeJob", DISPLAY_ERROR);
580 bnet_fsend(jcr->file_bsock, runafter, msg);
581 ret = response(jcr, jcr->file_bsock, OKRunAfter, "ClientRunAfterJob", DISPLAY_ERROR);
587 * Send RunScripts to File daemon
588 * 1) We send all runscript to FD, they can be executed Before, After, or twice
589 * 2) Then, we send a "RunBeforeNow" command to the FD to tell him to do the
590 * first run_script() call. (ie ClientRunBeforeJob)
592 int send_runscripts_commands(JCR *jcr)
594 POOLMEM *msg = get_pool_memory(PM_FNAME);
595 BSOCK *fd = jcr->file_bsock;
597 bool launch_before_cmd = false;
598 POOLMEM *ehost = get_pool_memory(PM_FNAME);
601 Dmsg0(120, "bdird: sending runscripts to fd\n");
603 foreach_alist(cmd, jcr->job->RunScripts) {
604 if (cmd->can_run_at_level(jcr->get_JobLevel()) && cmd->target) {
605 ehost = edit_job_codes(jcr, ehost, cmd->target, "");
606 Dmsg2(200, "bdird: runscript %s -> %s\n", cmd->target, ehost);
608 if (strcmp(ehost, jcr->client->name()) == 0) {
609 pm_strcpy(msg, cmd->command);
612 Dmsg1(120, "bdird: sending runscripts to fd '%s'\n", cmd->command);
614 /* TODO: remove this with bacula 1.42 */
615 if (cmd->old_proto) {
616 result = send_runscript_with_old_proto(jcr, cmd->when, msg);
619 fd->fsend(runscript, cmd->on_success,
625 result = response(jcr, fd, OKRunScript, "RunScript", DISPLAY_ERROR);
626 launch_before_cmd = true;
633 /* TODO : we have to play with other client */
636 send command to an other client
642 /* Tell the FD to execute the ClientRunBeforeJob */
643 if (launch_before_cmd) {
644 fd->fsend(runbeforenow);
645 if (!response(jcr, fd, OKRunBeforeNow, "RunBeforeNow", DISPLAY_ERROR)) {
649 free_pool_memory(msg);
650 free_pool_memory(ehost);
654 Jmsg(jcr, M_FATAL, 0, _("Client \"%s\" RunScript failed.\n"), ehost);
655 free_pool_memory(msg);
656 free_pool_memory(ehost);
663 * Read the attributes from the File daemon for
664 * a Verify job and store them in the catalog.
666 int get_attributes_and_put_in_catalog(JCR *jcr)
671 char digest[MAXSTRING];
673 fd = jcr->file_bsock;
674 jcr->jr.FirstIndex = 1;
676 /* Start transaction allocates jcr->attr and jcr->ar if needed */
677 db_start_transaction(jcr, jcr->db); /* start transaction if not already open */
680 Dmsg0(120, "bdird: waiting to receive file attributes\n");
681 /* Pickup file attributes and digest */
682 while (!fd->errors && (n = bget_dirmsg(fd)) > 0) {
686 char Digest[MAXSTRING]; /* either Verify opts or MD5/SHA1 digest */
688 if ((len = sscanf(fd->msg, "%ld %d %s", &file_index, &stream, Digest)) != 3) {
689 Jmsg(jcr, M_FATAL, 0, _("<filed: bad attributes, expected 3 fields got %d\n"
690 "msglen=%d msg=%s\n"), len, fd->msglen, fd->msg);
691 set_jcr_job_status(jcr, JS_ErrorTerminated);
695 /* The following three fields were sscanf'ed above so skip them */
696 skip_nonspaces(&p); /* skip FileIndex */
698 skip_nonspaces(&p); /* skip Stream */
700 skip_nonspaces(&p); /* skip Opts_Digest */
701 p++; /* skip space */
702 Dmsg1(dbglvl, "Stream=%d\n", stream);
703 if (stream == STREAM_UNIX_ATTRIBUTES || stream == STREAM_UNIX_ATTRIBUTES_EX) {
704 if (jcr->cached_attribute) {
705 Dmsg3(dbglvl, "Cached attr. Stream=%d fname=%s\n", ar->Stream, ar->fname,
707 if (!db_create_file_attributes_record(jcr, jcr->db, ar)) {
708 Jmsg1(jcr, M_FATAL, 0, _("Attribute create error. %s"), db_strerror(jcr->db));
711 /* Any cached attr is flushed so we can reuse jcr->attr and jcr->ar */
712 fn = jcr->fname = check_pool_memory_size(jcr->fname, fd->msglen);
714 *fn++ = *p++; /* copy filename */
716 *fn = *p++; /* term filename and point p to attribs */
717 pm_strcpy(jcr->attr, p); /* save attributes */
719 jcr->FileIndex = file_index;
720 ar->attr = jcr->attr;
721 ar->fname = jcr->fname;
722 ar->FileIndex = file_index;
725 ar->JobId = jcr->JobId;
726 ar->ClientId = jcr->ClientId;
730 ar->DigestType = CRYPTO_DIGEST_NONE;
731 jcr->cached_attribute = true;
733 Dmsg2(dbglvl, "dird<filed: stream=%d %s\n", stream, jcr->fname);
734 Dmsg1(dbglvl, "dird<filed: attr=%s\n", ar->attr);
735 jcr->FileId = ar->FileId;
737 * First, get STREAM_UNIX_ATTRIBUTES and fill ATTR_DBR structure
738 * Next, we CAN have a CRYPTO_DIGEST, so we fill ATTR_DBR with it (or not)
739 * When we get a new STREAM_UNIX_ATTRIBUTES, we known that we can add file to the catalog
740 * At the end, we have to add the last file
742 } else if (crypto_digest_stream_type(stream) != CRYPTO_DIGEST_NONE) {
743 if (jcr->FileIndex != (uint32_t)file_index) {
744 Jmsg3(jcr, M_ERROR, 0, _("%s index %d not same as attributes %d\n"),
745 stream_to_ascii(stream), file_index, jcr->FileIndex);
749 ar->DigestType = crypto_digest_stream_type(stream);
750 db_escape_string(jcr, jcr->db, digest, Digest, strlen(Digest));
751 Dmsg4(dbglvl, "stream=%d DigestLen=%d Digest=%s type=%d\n", stream,
752 strlen(digest), digest, ar->DigestType);
754 jcr->jr.JobFiles = jcr->JobFiles = file_index;
755 jcr->jr.LastIndex = file_index;
757 if (is_bnet_error(fd)) {
758 Jmsg1(jcr, M_FATAL, 0, _("<filed: Network error getting attributes. ERR=%s\n"),
762 if (jcr->cached_attribute) {
763 Dmsg3(dbglvl, "Cached attr with digest. Stream=%d fname=%s attr=%s\n", ar->Stream,
764 ar->fname, ar->attr);
765 if (!db_create_file_attributes_record(jcr, jcr->db, ar)) {
766 Jmsg1(jcr, M_FATAL, 0, _("Attribute create error. %s"), db_strerror(jcr->db));
768 jcr->cached_attribute = false;
770 set_jcr_job_status(jcr, JS_Terminated);