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);
201 /* If there was a successful job, make sure it is recent enough */
202 if (jcr->get_JobLevel() == L_INCREMENTAL && have_full && jcr->job->MaxDiffInterval > 0) {
203 /* Lookup last diff job */
204 if (db_find_last_job_start_time(jcr, jcr->db, &jcr->jr, &stime, L_DIFFERENTIAL)) {
205 last_diff_time = str_to_utime(stime);
206 do_diff = ((now - last_diff_time) >= jcr->job->MaxDiffInterval);
209 if (have_full && jcr->job->MaxFullInterval > 0 &&
210 db_find_last_job_start_time(jcr, jcr->db, &jcr->jr, &stime, L_FULL)) {
211 last_full_time = str_to_utime(stime);
212 do_full = ((now - last_full_time) >= jcr->job->MaxFullInterval);
214 free_pool_memory(stime);
216 if (!have_full || do_full) {
217 /* No recent Full job found, so upgrade this one to Full */
218 Jmsg(jcr, M_INFO, 0, "%s", db_strerror(jcr->db));
219 Jmsg(jcr, M_INFO, 0, _("No prior or suitable Full backup found in catalog. Doing FULL backup.\n"));
220 bsnprintf(since, since_len, _(" (upgraded from %s)"),
221 level_to_str(jcr->get_JobLevel()));
222 jcr->set_JobLevel(jcr->jr.JobLevel = L_FULL);
223 } else if (do_diff) {
224 /* No recent diff job found, so upgrade this one to Full */
225 Jmsg(jcr, M_INFO, 0, _("No prior or suitable Differential backup found in catalog. Doing Differential backup.\n"));
226 bsnprintf(since, since_len, _(" (upgraded from %s)"),
227 level_to_str(jcr->get_JobLevel()));
228 jcr->set_JobLevel(jcr->jr.JobLevel = L_DIFFERENTIAL);
230 if (jcr->job->rerun_failed_levels) {
231 if (db_find_failed_job_since(jcr, jcr->db, &jcr->jr, jcr->stime, JobLevel)) {
232 Jmsg(jcr, M_INFO, 0, _("Prior failed job found in catalog. Upgrading to %s.\n"),
233 level_to_str(JobLevel));
234 bsnprintf(since, since_len, _(" (upgraded from %s)"),
235 level_to_str(jcr->get_JobLevel()));
236 jcr->set_JobLevel(jcr->jr.JobLevel = JobLevel);
237 jcr->jr.JobId = jcr->JobId;
241 bstrncpy(since, _(", since="), since_len);
242 bstrncat(since, jcr->stime, since_len);
244 jcr->jr.JobId = jcr->JobId;
247 Dmsg2(100, "Level=%c last start time=%s\n", jcr->get_JobLevel(), jcr->stime);
250 static void send_since_time(JCR *jcr)
252 BSOCK *fd = jcr->file_bsock;
256 stime = str_to_utime(jcr->stime);
257 fd->fsend(levelcmd, "", NT_("since_utime "), edit_uint64(stime, ed1), 0);
258 while (bget_dirmsg(fd) >= 0) { /* allow him to poll us to sync clocks */
259 Jmsg(jcr, M_INFO, 0, "%s\n", fd->msg);
264 * Send level command to FD.
265 * Used for backup jobs and estimate command.
267 bool send_level_command(JCR *jcr)
269 BSOCK *fd = jcr->file_bsock;
270 const char *accurate = jcr->job->accurate?"accurate_":"";
271 const char *not_accurate = "";
273 * Send Level command to File daemon
275 switch (jcr->get_JobLevel()) {
277 fd->fsend(levelcmd, not_accurate, "base", " ", 0);
279 /* L_NONE is the console, sending something off to the FD */
282 fd->fsend(levelcmd, not_accurate, "full", " ", 0);
285 fd->fsend(levelcmd, accurate, "differential", " ", 0);
286 send_since_time(jcr);
289 fd->fsend(levelcmd, accurate, "incremental", " ", 0);
290 send_since_time(jcr);
294 Jmsg2(jcr, M_FATAL, 0, _("Unimplemented backup level %d %c\n"),
295 jcr->get_JobLevel(), jcr->get_JobLevel());
298 Dmsg1(120, ">filed: %s", fd->msg);
299 if (!response(jcr, fd, OKlevel, "Level", DISPLAY_ERROR)) {
306 * Send either an Included or an Excluded list to FD
308 static bool send_fileset(JCR *jcr)
310 FILESET *fileset = jcr->fileset;
311 BSOCK *fd = jcr->file_bsock;
317 num = fileset->num_includes;
319 num = fileset->num_excludes;
321 for (int i=0; i<num; i++) {
327 ie = fileset->include_items[i];
330 ie = fileset->exclude_items[i];
333 for (j=0; j<ie->num_opts; j++) {
334 FOPTS *fo = ie->opts_list[j];
335 fd->fsend("O %s\n", fo->opts);
337 bool enhanced_wild = false;
338 for (k=0; fo->opts[k]!='\0'; k++) {
339 if (fo->opts[k]=='W') {
340 enhanced_wild = true;
345 for (k=0; k<fo->regex.size(); k++) {
346 fd->fsend("R %s\n", fo->regex.get(k));
348 for (k=0; k<fo->regexdir.size(); k++) {
349 fd->fsend("RD %s\n", fo->regexdir.get(k));
351 for (k=0; k<fo->regexfile.size(); k++) {
352 fd->fsend("RF %s\n", fo->regexfile.get(k));
354 for (k=0; k<fo->wild.size(); k++) {
355 fd->fsend("W %s\n", fo->wild.get(k));
357 for (k=0; k<fo->wilddir.size(); k++) {
358 fd->fsend("WD %s\n", fo->wilddir.get(k));
360 for (k=0; k<fo->wildfile.size(); k++) {
361 fd->fsend("WF %s\n", fo->wildfile.get(k));
363 for (k=0; k<fo->wildbase.size(); k++) {
364 fd->fsend("W%c %s\n", enhanced_wild ? 'B' : 'F', fo->wildbase.get(k));
366 for (k=0; k<fo->base.size(); k++) {
367 fd->fsend("B %s\n", fo->base.get(k));
369 for (k=0; k<fo->fstype.size(); k++) {
370 fd->fsend("X %s\n", fo->fstype.get(k));
372 for (k=0; k<fo->drivetype.size(); k++) {
373 fd->fsend("XD %s\n", fo->drivetype.get(k));
376 fd->fsend("G %s\n", fo->plugin);
379 bnet_fsend(fd, "Z %s\n", fo->ignoredir);
382 fd->fsend("D %s\n", fo->reader);
385 fd->fsend("T %s\n", fo->writer);
390 for (j=0; j<ie->name_list.size(); j++) {
391 item = (char *)ie->name_list.get(j);
392 if (!send_list_item(jcr, "F ", item, fd)) {
397 for (j=0; j<ie->plugin_list.size(); j++) {
398 item = (char *)ie->plugin_list.get(j);
399 if (!send_list_item(jcr, "P ", item, fd)) {
405 if (!include) { /* If we just did excludes */
406 break; /* all done */
408 include = false; /* Now do excludes */
411 fd->signal(BNET_EOD); /* end of data */
412 if (!response(jcr, fd, OKinc, "Include", DISPLAY_ERROR)) {
418 set_jcr_job_status(jcr, JS_ErrorTerminated);
423 static bool send_list_item(JCR *jcr, const char *code, char *item, BSOCK *fd)
433 p++; /* skip over the | */
434 fd->msg = edit_job_codes(jcr, fd->msg, p, "");
435 bpipe = open_bpipe(fd->msg, 0, "r");
438 Jmsg(jcr, M_FATAL, 0, _("Cannot run program: %s. ERR=%s\n"),
442 bstrncpy(buf, code, sizeof(buf));
443 Dmsg1(500, "code=%s\n", buf);
444 optlen = strlen(buf);
445 while (fgets(buf+optlen, sizeof(buf)-optlen, bpipe->rfd)) {
446 fd->msglen = Mmsg(fd->msg, "%s", buf);
447 Dmsg2(500, "Inc/exc len=%d: %s", fd->msglen, fd->msg);
448 if (!bnet_send(fd)) {
449 Jmsg(jcr, M_FATAL, 0, _(">filed: write error on socket\n"));
453 if ((stat=close_bpipe(bpipe)) != 0) {
455 Jmsg(jcr, M_FATAL, 0, _("Error running program: %s. ERR=%s\n"),
456 p, be.bstrerror(stat));
461 p++; /* skip over < */
462 if ((ffd = fopen(p, "rb")) == NULL) {
464 Jmsg(jcr, M_FATAL, 0, _("Cannot open included file: %s. ERR=%s\n"),
468 bstrncpy(buf, code, sizeof(buf));
469 Dmsg1(500, "code=%s\n", buf);
470 optlen = strlen(buf);
471 while (fgets(buf+optlen, sizeof(buf)-optlen, ffd)) {
472 fd->msglen = Mmsg(fd->msg, "%s", buf);
473 if (!bnet_send(fd)) {
474 Jmsg(jcr, M_FATAL, 0, _(">filed: write error on socket\n"));
481 p++; /* skip over \ */
482 /* Note, fall through wanted */
484 pm_strcpy(fd->msg, code);
485 fd->msglen = pm_strcat(fd->msg, p);
486 Dmsg1(500, "Inc/Exc name=%s\n", fd->msg);
488 Jmsg(jcr, M_FATAL, 0, _(">filed: write error on socket\n"));
498 * Send include list to File daemon
500 bool send_include_list(JCR *jcr)
502 BSOCK *fd = jcr->file_bsock;
503 if (jcr->fileset->new_include) {
504 fd->fsend(filesetcmd, jcr->fileset->enable_vss ? " vss=1" : "");
505 return send_fileset(jcr);
512 * Send exclude list to File daemon
513 * Under the new scheme, the Exclude list
514 * is part of the FileSet sent with the
515 * "include_list" above.
517 bool send_exclude_list(JCR *jcr)
524 * Send bootstrap file if any to the socket given (FD or SD).
525 * This is used for restore, verify VolumeToCatalog, and
528 bool send_bootstrap_file(JCR *jcr, BSOCK *sock)
532 const char *bootstrap = "bootstrap\n";
534 Dmsg1(400, "send_bootstrap_file: %s\n", jcr->RestoreBootstrap);
535 if (!jcr->RestoreBootstrap) {
538 bs = fopen(jcr->RestoreBootstrap, "rb");
541 Jmsg(jcr, M_FATAL, 0, _("Could not open bootstrap file %s: ERR=%s\n"),
542 jcr->RestoreBootstrap, be.bstrerror());
543 set_jcr_job_status(jcr, JS_ErrorTerminated);
546 sock->fsend(bootstrap);
547 while (fgets(buf, sizeof(buf), bs)) {
548 sock->fsend("%s", buf);
550 sock->signal(BNET_EOD);
552 if (jcr->unlink_bsr) {
553 unlink(jcr->RestoreBootstrap);
554 jcr->unlink_bsr = false;
559 /* TODO: drop this with runscript.old_proto in bacula 1.42 */
560 static char runbefore[] = "RunBeforeJob %s\n";
561 static char runafter[] = "RunAfterJob %s\n";
562 static char OKRunBefore[] = "2000 OK RunBefore\n";
563 static char OKRunAfter[] = "2000 OK RunAfter\n";
565 int send_runscript_with_old_proto(JCR *jcr, int when, POOLMEM *msg)
568 Dmsg1(120, "bdird: sending old runcommand to fd '%s'\n",msg);
569 if (when & SCRIPT_Before) {
570 bnet_fsend(jcr->file_bsock, runbefore, msg);
571 ret = response(jcr, jcr->file_bsock, OKRunBefore, "ClientRunBeforeJob", DISPLAY_ERROR);
573 bnet_fsend(jcr->file_bsock, runafter, msg);
574 ret = response(jcr, jcr->file_bsock, OKRunAfter, "ClientRunAfterJob", DISPLAY_ERROR);
580 * Send RunScripts to File daemon
581 * 1) We send all runscript to FD, they can be executed Before, After, or twice
582 * 2) Then, we send a "RunBeforeNow" command to the FD to tell him to do the
583 * first run_script() call. (ie ClientRunBeforeJob)
585 int send_runscripts_commands(JCR *jcr)
587 POOLMEM *msg = get_pool_memory(PM_FNAME);
588 BSOCK *fd = jcr->file_bsock;
590 bool launch_before_cmd = false;
591 POOLMEM *ehost = get_pool_memory(PM_FNAME);
594 Dmsg0(120, "bdird: sending runscripts to fd\n");
596 foreach_alist(cmd, jcr->job->RunScripts) {
597 if (cmd->can_run_at_level(jcr->get_JobLevel()) && cmd->target) {
598 ehost = edit_job_codes(jcr, ehost, cmd->target, "");
599 Dmsg2(200, "bdird: runscript %s -> %s\n", cmd->target, ehost);
601 if (strcmp(ehost, jcr->client->name()) == 0) {
602 pm_strcpy(msg, cmd->command);
605 Dmsg1(120, "bdird: sending runscripts to fd '%s'\n", cmd->command);
607 /* TODO: remove this with bacula 1.42 */
608 if (cmd->old_proto) {
609 result = send_runscript_with_old_proto(jcr, cmd->when, msg);
612 fd->fsend(runscript, cmd->on_success,
618 result = response(jcr, fd, OKRunScript, "RunScript", DISPLAY_ERROR);
619 launch_before_cmd = true;
626 /* TODO : we have to play with other client */
629 send command to an other client
635 /* Tell the FD to execute the ClientRunBeforeJob */
636 if (launch_before_cmd) {
637 fd->fsend(runbeforenow);
638 if (!response(jcr, fd, OKRunBeforeNow, "RunBeforeNow", DISPLAY_ERROR)) {
642 free_pool_memory(msg);
643 free_pool_memory(ehost);
647 Jmsg(jcr, M_FATAL, 0, _("Client \"%s\" RunScript failed.\n"), ehost);
648 free_pool_memory(msg);
649 free_pool_memory(ehost);
656 * Read the attributes from the File daemon for
657 * a Verify job and store them in the catalog.
659 int get_attributes_and_put_in_catalog(JCR *jcr)
664 char digest[MAXSTRING];
666 fd = jcr->file_bsock;
667 jcr->jr.FirstIndex = 1;
669 /* Start transaction allocates jcr->attr and jcr->ar if needed */
670 db_start_transaction(jcr, jcr->db); /* start transaction if not already open */
673 Dmsg0(120, "bdird: waiting to receive file attributes\n");
674 /* Pickup file attributes and digest */
675 while (!fd->errors && (n = bget_dirmsg(fd)) > 0) {
679 char Digest[MAXSTRING]; /* either Verify opts or MD5/SHA1 digest */
681 if ((len = sscanf(fd->msg, "%ld %d %s", &file_index, &stream, Digest)) != 3) {
682 Jmsg(jcr, M_FATAL, 0, _("<filed: bad attributes, expected 3 fields got %d\n"
683 "msglen=%d msg=%s\n"), len, fd->msglen, fd->msg);
684 set_jcr_job_status(jcr, JS_ErrorTerminated);
688 /* The following three fields were sscanf'ed above so skip them */
689 skip_nonspaces(&p); /* skip FileIndex */
691 skip_nonspaces(&p); /* skip Stream */
693 skip_nonspaces(&p); /* skip Opts_Digest */
694 p++; /* skip space */
695 Dmsg1(dbglvl, "Stream=%d\n", stream);
696 if (stream == STREAM_UNIX_ATTRIBUTES || stream == STREAM_UNIX_ATTRIBUTES_EX) {
697 if (jcr->cached_attribute) {
698 Dmsg3(dbglvl, "Cached attr. Stream=%d fname=%s\n", ar->Stream, ar->fname,
700 if (!db_create_file_attributes_record(jcr, jcr->db, ar)) {
701 Jmsg1(jcr, M_FATAL, 0, _("Attribute create error. %s"), db_strerror(jcr->db));
704 /* Any cached attr is flushed so we can reuse jcr->attr and jcr->ar */
705 fn = jcr->fname = check_pool_memory_size(jcr->fname, fd->msglen);
707 *fn++ = *p++; /* copy filename */
709 *fn = *p++; /* term filename and point p to attribs */
710 pm_strcpy(jcr->attr, p); /* save attributes */
712 jcr->FileIndex = file_index;
713 ar->attr = jcr->attr;
714 ar->fname = jcr->fname;
715 ar->FileIndex = file_index;
718 ar->JobId = jcr->JobId;
719 ar->ClientId = jcr->ClientId;
723 ar->DigestType = CRYPTO_DIGEST_NONE;
724 jcr->cached_attribute = true;
726 Dmsg2(dbglvl, "dird<filed: stream=%d %s\n", stream, jcr->fname);
727 Dmsg1(dbglvl, "dird<filed: attr=%s\n", ar->attr);
728 jcr->FileId = ar->FileId;
730 * First, get STREAM_UNIX_ATTRIBUTES and fill ATTR_DBR structure
731 * Next, we CAN have a CRYPTO_DIGEST, so we fill ATTR_DBR with it (or not)
732 * When we get a new STREAM_UNIX_ATTRIBUTES, we known that we can add file to the catalog
733 * At the end, we have to add the last file
735 } else if (crypto_digest_stream_type(stream) != CRYPTO_DIGEST_NONE) {
736 if (jcr->FileIndex != (uint32_t)file_index) {
737 Jmsg3(jcr, M_ERROR, 0, _("%s index %d not same as attributes %d\n"),
738 stream_to_ascii(stream), file_index, jcr->FileIndex);
742 ar->DigestType = crypto_digest_stream_type(stream);
743 db_escape_string(jcr, jcr->db, digest, Digest, strlen(Digest));
744 Dmsg4(dbglvl, "stream=%d DigestLen=%d Digest=%s type=%d\n", stream,
745 strlen(digest), digest, ar->DigestType);
747 jcr->jr.JobFiles = jcr->JobFiles = file_index;
748 jcr->jr.LastIndex = file_index;
750 if (is_bnet_error(fd)) {
751 Jmsg1(jcr, M_FATAL, 0, _("<filed: Network error getting attributes. ERR=%s\n"),
755 if (jcr->cached_attribute) {
756 Dmsg3(dbglvl, "Cached attr with digest. Stream=%d fname=%s attr=%s\n", ar->Stream,
757 ar->fname, ar->attr);
758 if (!db_create_file_attributes_record(jcr, jcr->db, ar)) {
759 Jmsg1(jcr, M_FATAL, 0, _("Attribute create error. %s"), db_strerror(jcr->db));
761 jcr->cached_attribute = false;
763 set_jcr_job_status(jcr, JS_Terminated);