| /* |
| * Routines only used by the receiving process. |
| * |
| * Copyright (C) 1996-2000 Andrew Tridgell |
| * Copyright (C) 1996 Paul Mackerras |
| * Copyright (C) 2003-2008 Wayne Davison |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 3 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * 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 along |
| * with this program; if not, visit the http://fsf.org website. |
| */ |
| |
| #include "rsync.h" |
| |
| extern int verbose; |
| extern int dry_run; |
| extern int do_xfers; |
| extern int am_server; |
| extern int do_progress; |
| extern int inc_recurse; |
| extern int log_before_transfer; |
| extern int stdout_format_has_i; |
| extern int logfile_format_has_i; |
| extern int csum_length; |
| extern int read_batch; |
| extern int write_batch; |
| extern int batch_gen_fd; |
| extern int protocol_version; |
| extern int relative_paths; |
| extern int preserve_hard_links; |
| extern int preserve_perms; |
| extern int preserve_xattrs; |
| extern int basis_dir_cnt; |
| extern int make_backups; |
| extern int cleanup_got_literal; |
| extern int remove_source_files; |
| extern int append_mode; |
| extern int sparse_files; |
| extern int keep_partial; |
| extern int checksum_seed; |
| extern int inplace; |
| extern int delay_updates; |
| extern mode_t orig_umask; |
| extern struct stats stats; |
| extern char *tmpdir; |
| extern char *partial_dir; |
| extern char *basis_dir[]; |
| extern struct file_list *cur_flist, *first_flist, *dir_flist; |
| extern struct filter_list_struct daemon_filter_list; |
| |
| static struct bitbag *delayed_bits = NULL; |
| static int phase = 0, redoing = 0; |
| /* We're either updating the basis file or an identical copy: */ |
| static int updating_basis_or_equiv; |
| |
| /* |
| * get_tmpname() - create a tmp filename for a given filename |
| * |
| * If a tmpdir is defined, use that as the directory to |
| * put it in. Otherwise, the tmp filename is in the same |
| * directory as the given name. Note that there may be no |
| * directory at all in the given name! |
| * |
| * The tmp filename is basically the given filename with a |
| * dot prepended, and .XXXXXX appended (for mkstemp() to |
| * put its unique gunk in). Take care to not exceed |
| * either the MAXPATHLEN or NAME_MAX, esp. the last, as |
| * the basename basically becomes 8 chars longer. In that |
| * case, the original name is shortened sufficiently to |
| * make it all fit. |
| * |
| * Of course, there's no real reason for the tmp name to |
| * look like the original, except to satisfy us humans. |
| * As long as it's unique, rsync will work. |
| */ |
| |
| int get_tmpname(char *fnametmp, const char *fname) |
| { |
| int maxname, added, length = 0; |
| const char *f; |
| |
| if (tmpdir) { |
| /* Note: this can't overflow, so the return value is safe */ |
| length = strlcpy(fnametmp, tmpdir, MAXPATHLEN - 2); |
| fnametmp[length++] = '/'; |
| } |
| |
| if ((f = strrchr(fname, '/')) != NULL) { |
| ++f; |
| if (!tmpdir) { |
| length = f - fname; |
| /* copy up to and including the slash */ |
| strlcpy(fnametmp, fname, length + 1); |
| } |
| } else |
| f = fname; |
| fnametmp[length++] = '.'; |
| |
| /* The maxname value is bufsize, and includes space for the '\0'. |
| * (Note that NAME_MAX get -8 for the leading '.' above.) */ |
| maxname = MIN(MAXPATHLEN - 7 - length, NAME_MAX - 8); |
| |
| if (maxname < 1) { |
| rprintf(FERROR_XFER, "temporary filename too long: %s\n", fname); |
| fnametmp[0] = '\0'; |
| return 0; |
| } |
| |
| added = strlcpy(fnametmp + length, f, maxname); |
| if (added >= maxname) |
| added = maxname - 1; |
| memcpy(fnametmp + length + added, ".XXXXXX", 8); |
| |
| return 1; |
| } |
| |
| /* Opens a temporary file for writing. |
| * Success: Writes name into fnametmp, returns fd. |
| * Failure: Clobbers fnametmp, returns -1. |
| * Calling cleanup_set() is the caller's job. */ |
| int open_tmpfile(char *fnametmp, const char *fname, struct file_struct *file) |
| { |
| int fd; |
| |
| if (!get_tmpname(fnametmp, fname)) |
| return -1; |
| |
| /* We initially set the perms without the setuid/setgid bits or group |
| * access to ensure that there is no race condition. They will be |
| * correctly updated after the right owner and group info is set. |
| * (Thanks to snabb@epipe.fi for pointing this out.) */ |
| fd = do_mkstemp(fnametmp, file->mode & INITACCESSPERMS); |
| |
| #if 0 |
| /* In most cases parent directories will already exist because their |
| * information should have been previously transferred, but that may |
| * not be the case with -R */ |
| if (fd == -1 && relative_paths && errno == ENOENT |
| && create_directory_path(fnametmp) == 0) { |
| /* Get back to name with XXXXXX in it. */ |
| get_tmpname(fnametmp, fname); |
| fd = do_mkstemp(fnametmp, file->mode & INITACCESSPERMS); |
| } |
| #endif |
| |
| if (fd == -1) { |
| rsyserr(FERROR_XFER, errno, "mkstemp %s failed", |
| full_fname(fnametmp)); |
| return -1; |
| } |
| |
| return fd; |
| } |
| |
| static int receive_data(int f_in, char *fname_r, int fd_r, OFF_T size_r, |
| const char *fname, int fd, OFF_T total_size) |
| { |
| static char file_sum1[MAX_DIGEST_LEN]; |
| static char file_sum2[MAX_DIGEST_LEN]; |
| struct map_struct *mapbuf; |
| struct sum_struct sum; |
| int32 len, sum_len; |
| OFF_T offset = 0; |
| OFF_T offset2; |
| char *data; |
| int32 i; |
| char *map = NULL; |
| |
| read_sum_head(f_in, &sum); |
| |
| if (fd_r >= 0 && size_r > 0) { |
| int32 read_size = MAX(sum.blength * 2, 16*1024); |
| mapbuf = map_file(fd_r, size_r, read_size, sum.blength); |
| if (verbose > 2) { |
| rprintf(FINFO, "recv mapped %s of size %.0f\n", |
| fname_r, (double)size_r); |
| } |
| } else |
| mapbuf = NULL; |
| |
| sum_init(checksum_seed); |
| |
| if (append_mode > 0) { |
| OFF_T j; |
| sum.flength = (OFF_T)sum.count * sum.blength; |
| if (sum.remainder) |
| sum.flength -= sum.blength - sum.remainder; |
| if (append_mode == 2) { |
| for (j = CHUNK_SIZE; j < sum.flength; j += CHUNK_SIZE) { |
| if (do_progress) |
| show_progress(offset, total_size); |
| sum_update(map_ptr(mapbuf, offset, CHUNK_SIZE), |
| CHUNK_SIZE); |
| offset = j; |
| } |
| if (offset < sum.flength) { |
| int32 len = (int32)(sum.flength - offset); |
| if (do_progress) |
| show_progress(offset, total_size); |
| sum_update(map_ptr(mapbuf, offset, len), len); |
| } |
| } |
| offset = sum.flength; |
| if (fd != -1 && (j = do_lseek(fd, offset, SEEK_SET)) != offset) { |
| rsyserr(FERROR_XFER, errno, "lseek of %s returned %.0f, not %.0f", |
| full_fname(fname), (double)j, (double)offset); |
| exit_cleanup(RERR_FILEIO); |
| } |
| } |
| |
| while ((i = recv_token(f_in, &data)) != 0) { |
| if (do_progress) |
| show_progress(offset, total_size); |
| |
| if (i > 0) { |
| if (verbose > 3) { |
| rprintf(FINFO,"data recv %d at %.0f\n", |
| i,(double)offset); |
| } |
| |
| stats.literal_data += i; |
| cleanup_got_literal = 1; |
| |
| sum_update(data, i); |
| |
| if (fd != -1 && write_file(fd,data,i) != i) |
| goto report_write_error; |
| offset += i; |
| continue; |
| } |
| |
| i = -(i+1); |
| offset2 = i * (OFF_T)sum.blength; |
| len = sum.blength; |
| if (i == (int)sum.count-1 && sum.remainder != 0) |
| len = sum.remainder; |
| |
| stats.matched_data += len; |
| |
| if (verbose > 3) { |
| rprintf(FINFO, |
| "chunk[%d] of size %ld at %.0f offset=%.0f\n", |
| i, (long)len, (double)offset2, (double)offset); |
| } |
| |
| if (mapbuf) { |
| map = map_ptr(mapbuf,offset2,len); |
| |
| see_token(map, len); |
| sum_update(map, len); |
| } |
| |
| if (updating_basis_or_equiv) { |
| if (offset == offset2 && fd != -1) { |
| OFF_T pos; |
| if (flush_write_file(fd) < 0) |
| goto report_write_error; |
| offset += len; |
| if ((pos = do_lseek(fd, len, SEEK_CUR)) != offset) { |
| rsyserr(FERROR_XFER, errno, |
| "lseek of %s returned %.0f, not %.0f", |
| full_fname(fname), |
| (double)pos, (double)offset); |
| exit_cleanup(RERR_FILEIO); |
| } |
| continue; |
| } |
| } |
| if (fd != -1 && map && write_file(fd, map, len) != (int)len) |
| goto report_write_error; |
| offset += len; |
| } |
| |
| if (flush_write_file(fd) < 0) |
| goto report_write_error; |
| |
| #ifdef HAVE_FTRUNCATE |
| if (inplace && fd != -1) |
| ftruncate(fd, offset); |
| #endif |
| |
| if (do_progress) |
| end_progress(total_size); |
| |
| if (fd != -1 && offset > 0 && sparse_end(fd) != 0) { |
| report_write_error: |
| rsyserr(FERROR_XFER, errno, "write failed on %s", |
| full_fname(fname)); |
| exit_cleanup(RERR_FILEIO); |
| } |
| |
| sum_len = sum_end(file_sum1); |
| |
| if (mapbuf) |
| unmap_file(mapbuf); |
| |
| read_buf(f_in, file_sum2, sum_len); |
| if (verbose > 2) |
| rprintf(FINFO,"got file_sum\n"); |
| if (fd != -1 && memcmp(file_sum1, file_sum2, sum_len) != 0) |
| return 0; |
| return 1; |
| } |
| |
| |
| static void discard_receive_data(int f_in, OFF_T length) |
| { |
| receive_data(f_in, NULL, -1, 0, NULL, -1, length); |
| } |
| |
| static void handle_delayed_updates(char *local_name) |
| { |
| char *fname, *partialptr; |
| int ndx; |
| |
| for (ndx = -1; (ndx = bitbag_next_bit(delayed_bits, ndx)) >= 0; ) { |
| struct file_struct *file = cur_flist->files[ndx]; |
| fname = local_name ? local_name : f_name(file, NULL); |
| if ((partialptr = partial_dir_fname(fname)) != NULL) { |
| if (make_backups > 0 && !make_backup(fname)) |
| continue; |
| if (verbose > 2) { |
| rprintf(FINFO, "renaming %s to %s\n", |
| partialptr, fname); |
| } |
| /* We don't use robust_rename() here because the |
| * partial-dir must be on the same drive. */ |
| if (do_rename(partialptr, fname) < 0) { |
| rsyserr(FERROR_XFER, errno, |
| "rename failed for %s (from %s)", |
| full_fname(fname), partialptr); |
| } else { |
| if (remove_source_files |
| || (preserve_hard_links && F_IS_HLINKED(file))) |
| send_msg_int(MSG_SUCCESS, ndx); |
| handle_partial_dir(partialptr, PDIR_DELETE); |
| } |
| } |
| } |
| } |
| |
| static int get_next_gen_ndx(int fd, int next_gen_ndx, int desired_ndx) |
| { |
| while (next_gen_ndx < desired_ndx) { |
| if (next_gen_ndx >= 0) { |
| struct file_struct *file = cur_flist->files[next_gen_ndx]; |
| rprintf(FERROR_XFER, |
| "(No batched update for%s \"%s\")\n", |
| file->flags & FLAG_FILE_SENT ? " resend of" : "", |
| f_name(file, NULL)); |
| } |
| next_gen_ndx = read_int(fd); |
| if (next_gen_ndx == -1) { |
| if (inc_recurse) |
| next_gen_ndx = first_flist->prev->used + first_flist->prev->ndx_start; |
| else |
| next_gen_ndx = cur_flist->used; |
| } |
| } |
| return next_gen_ndx; |
| } |
| |
| /** |
| * main routine for receiver process. |
| * |
| * Receiver process runs on the same host as the generator process. */ |
| int recv_files(int f_in, char *local_name) |
| { |
| int next_gen_ndx = -1; |
| int fd1,fd2; |
| STRUCT_STAT st; |
| int iflags, xlen; |
| char *fname, fbuf[MAXPATHLEN]; |
| char xname[MAXPATHLEN]; |
| char fnametmp[MAXPATHLEN]; |
| char *fnamecmp, *partialptr; |
| char fnamecmpbuf[MAXPATHLEN]; |
| uchar fnamecmp_type; |
| struct file_struct *file; |
| struct stats initial_stats; |
| int itemizing = am_server ? logfile_format_has_i : stdout_format_has_i; |
| enum logcode log_code = log_before_transfer ? FLOG : FINFO; |
| int max_phase = protocol_version >= 29 ? 2 : 1; |
| int dflt_perms = (ACCESSPERMS & ~orig_umask); |
| #ifdef SUPPORT_ACLS |
| const char *parent_dirname = ""; |
| #endif |
| int ndx, recv_ok; |
| |
| if (verbose > 2) |
| rprintf(FINFO, "recv_files(%d) starting\n", cur_flist->used); |
| |
| if (delay_updates) |
| delayed_bits = bitbag_create(cur_flist->used + 1); |
| |
| while (1) { |
| cleanup_disable(); |
| |
| /* This call also sets cur_flist. */ |
| ndx = read_ndx_and_attrs(f_in, &iflags, &fnamecmp_type, |
| xname, &xlen); |
| if (ndx == NDX_DONE) { |
| if (inc_recurse && first_flist) { |
| flist_free(first_flist); |
| if (first_flist) |
| continue; |
| } |
| if (read_batch && cur_flist) { |
| int high = inc_recurse |
| ? first_flist->prev->used + first_flist->prev->ndx_start |
| : cur_flist->used; |
| get_next_gen_ndx(batch_gen_fd, next_gen_ndx, high); |
| next_gen_ndx = -1; |
| } |
| if (++phase > max_phase) |
| break; |
| if (verbose > 2) |
| rprintf(FINFO, "recv_files phase=%d\n", phase); |
| if (phase == 2 && delay_updates) |
| handle_delayed_updates(local_name); |
| send_msg(MSG_DONE, "", 0, 0); |
| continue; |
| } |
| |
| if (ndx - cur_flist->ndx_start >= 0) |
| file = cur_flist->files[ndx - cur_flist->ndx_start]; |
| else |
| file = dir_flist->files[cur_flist->parent_ndx]; |
| fname = local_name ? local_name : f_name(file, fbuf); |
| |
| if (verbose > 2) |
| rprintf(FINFO, "recv_files(%s)\n", fname); |
| |
| #ifdef SUPPORT_XATTRS |
| if (iflags & ITEM_REPORT_XATTR && !dry_run) |
| recv_xattr_request(file, f_in); |
| #endif |
| |
| if (!(iflags & ITEM_TRANSFER)) { |
| maybe_log_item(file, iflags, itemizing, xname); |
| #ifdef SUPPORT_XATTRS |
| if (preserve_xattrs && iflags & ITEM_REPORT_XATTR && !dry_run) |
| set_file_attrs(fname, file, NULL, fname, 0); |
| #endif |
| continue; |
| } |
| if (phase == 2) { |
| rprintf(FERROR, |
| "got transfer request in phase 2 [%s]\n", |
| who_am_i()); |
| exit_cleanup(RERR_PROTOCOL); |
| } |
| |
| if (file->flags & FLAG_FILE_SENT) { |
| if (csum_length == SHORT_SUM_LENGTH) { |
| if (keep_partial && !partial_dir) |
| make_backups = -make_backups; /* prevents double backup */ |
| if (append_mode) |
| sparse_files = -sparse_files; |
| append_mode = -append_mode; |
| csum_length = SUM_LENGTH; |
| redoing = 1; |
| } |
| } else { |
| if (csum_length != SHORT_SUM_LENGTH) { |
| if (keep_partial && !partial_dir) |
| make_backups = -make_backups; |
| if (append_mode) |
| sparse_files = -sparse_files; |
| append_mode = -append_mode; |
| csum_length = SHORT_SUM_LENGTH; |
| redoing = 0; |
| } |
| } |
| |
| if (!am_server && do_progress) |
| set_current_file_index(file, ndx); |
| stats.num_transferred_files++; |
| stats.total_transferred_size += F_LENGTH(file); |
| |
| cleanup_got_literal = 0; |
| |
| if (daemon_filter_list.head |
| && check_filter(&daemon_filter_list, FLOG, fname, 0) < 0) { |
| rprintf(FERROR, "attempt to hack rsync failed.\n"); |
| exit_cleanup(RERR_PROTOCOL); |
| } |
| |
| if (!do_xfers) { /* log the transfer */ |
| log_item(FCLIENT, file, &stats, iflags, NULL); |
| if (read_batch) |
| discard_receive_data(f_in, F_LENGTH(file)); |
| continue; |
| } |
| if (write_batch < 0) { |
| log_item(FCLIENT, file, &stats, iflags, NULL); |
| if (!am_server) |
| discard_receive_data(f_in, F_LENGTH(file)); |
| continue; |
| } |
| |
| if (read_batch) { |
| next_gen_ndx = get_next_gen_ndx(batch_gen_fd, next_gen_ndx, ndx); |
| if (ndx < next_gen_ndx) { |
| rprintf(FINFO, |
| "(Skipping batched update for \"%s\")\n", |
| fname); |
| discard_receive_data(f_in, F_LENGTH(file)); |
| if (inc_recurse) |
| send_msg_int(MSG_NO_SEND, ndx); |
| continue; |
| } |
| next_gen_ndx = -1; |
| } |
| |
| partialptr = partial_dir ? partial_dir_fname(fname) : fname; |
| |
| if (protocol_version >= 29) { |
| switch (fnamecmp_type) { |
| case FNAMECMP_FNAME: |
| fnamecmp = fname; |
| break; |
| case FNAMECMP_PARTIAL_DIR: |
| fnamecmp = partialptr; |
| break; |
| case FNAMECMP_BACKUP: |
| fnamecmp = get_backup_name(fname); |
| break; |
| case FNAMECMP_FUZZY: |
| if (file->dirname) { |
| pathjoin(fnamecmpbuf, MAXPATHLEN, |
| file->dirname, xname); |
| fnamecmp = fnamecmpbuf; |
| } else |
| fnamecmp = xname; |
| break; |
| default: |
| if (fnamecmp_type >= basis_dir_cnt) { |
| rprintf(FERROR, |
| "invalid basis_dir index: %d.\n", |
| fnamecmp_type); |
| exit_cleanup(RERR_PROTOCOL); |
| } |
| pathjoin(fnamecmpbuf, sizeof fnamecmpbuf, |
| basis_dir[fnamecmp_type], fname); |
| fnamecmp = fnamecmpbuf; |
| break; |
| } |
| if (!fnamecmp || (daemon_filter_list.head |
| && check_filter(&daemon_filter_list, FLOG, fname, 0) < 0)) { |
| fnamecmp = fname; |
| fnamecmp_type = FNAMECMP_FNAME; |
| } |
| } else { |
| /* Reminder: --inplace && --partial-dir are never |
| * enabled at the same time. */ |
| if (inplace && make_backups > 0) { |
| if (!(fnamecmp = get_backup_name(fname))) |
| fnamecmp = fname; |
| else |
| fnamecmp_type = FNAMECMP_BACKUP; |
| } else if (partial_dir && partialptr) |
| fnamecmp = partialptr; |
| else |
| fnamecmp = fname; |
| } |
| |
| initial_stats = stats; |
| |
| /* open the file */ |
| fd1 = do_open(fnamecmp, O_RDONLY, 0); |
| |
| if (fd1 == -1 && protocol_version < 29) { |
| if (fnamecmp != fname) { |
| fnamecmp = fname; |
| fd1 = do_open(fnamecmp, O_RDONLY, 0); |
| } |
| |
| if (fd1 == -1 && basis_dir[0]) { |
| /* pre-29 allowed only one alternate basis */ |
| pathjoin(fnamecmpbuf, sizeof fnamecmpbuf, |
| basis_dir[0], fname); |
| fnamecmp = fnamecmpbuf; |
| fd1 = do_open(fnamecmp, O_RDONLY, 0); |
| } |
| } |
| |
| updating_basis_or_equiv = inplace |
| && (fnamecmp == fname || fnamecmp_type == FNAMECMP_BACKUP); |
| |
| if (fd1 == -1) { |
| st.st_mode = 0; |
| st.st_size = 0; |
| } else if (do_fstat(fd1,&st) != 0) { |
| rsyserr(FERROR_XFER, errno, "fstat %s failed", |
| full_fname(fnamecmp)); |
| discard_receive_data(f_in, F_LENGTH(file)); |
| close(fd1); |
| if (inc_recurse) |
| send_msg_int(MSG_NO_SEND, ndx); |
| continue; |
| } |
| |
| if (fd1 != -1 && S_ISDIR(st.st_mode) && fnamecmp == fname) { |
| /* this special handling for directories |
| * wouldn't be necessary if robust_rename() |
| * and the underlying robust_unlink could cope |
| * with directories |
| */ |
| rprintf(FERROR_XFER, "recv_files: %s is a directory\n", |
| full_fname(fnamecmp)); |
| discard_receive_data(f_in, F_LENGTH(file)); |
| close(fd1); |
| if (inc_recurse) |
| send_msg_int(MSG_NO_SEND, ndx); |
| continue; |
| } |
| |
| if (fd1 != -1 && !S_ISREG(st.st_mode)) { |
| close(fd1); |
| fd1 = -1; |
| } |
| |
| /* If we're not preserving permissions, change the file-list's |
| * mode based on the local permissions and some heuristics. */ |
| if (!preserve_perms) { |
| int exists = fd1 != -1; |
| #ifdef SUPPORT_ACLS |
| const char *dn = file->dirname ? file->dirname : "."; |
| if (parent_dirname != dn |
| && strcmp(parent_dirname, dn) != 0) { |
| dflt_perms = default_perms_for_dir(dn); |
| parent_dirname = dn; |
| } |
| #endif |
| file->mode = dest_mode(file->mode, st.st_mode, |
| dflt_perms, exists); |
| } |
| |
| /* We now check to see if we are writing the file "inplace" */ |
| if (inplace) { |
| fd2 = do_open(fname, O_WRONLY|O_CREAT, 0600); |
| if (fd2 == -1) { |
| rsyserr(FERROR_XFER, errno, "open %s failed", |
| full_fname(fname)); |
| } |
| } else { |
| fd2 = open_tmpfile(fnametmp, fname, file); |
| if (fd2 != -1) |
| cleanup_set(fnametmp, partialptr, file, fd1, fd2); |
| } |
| |
| if (fd2 == -1) { |
| discard_receive_data(f_in, F_LENGTH(file)); |
| if (fd1 != -1) |
| close(fd1); |
| if (inc_recurse) |
| send_msg_int(MSG_NO_SEND, ndx); |
| continue; |
| } |
| |
| /* log the transfer */ |
| if (log_before_transfer) |
| log_item(FCLIENT, file, &initial_stats, iflags, NULL); |
| else if (!am_server && verbose && do_progress) |
| rprintf(FINFO, "%s\n", fname); |
| |
| /* recv file data */ |
| recv_ok = receive_data(f_in, fnamecmp, fd1, st.st_size, |
| fname, fd2, F_LENGTH(file)); |
| |
| log_item(log_code, file, &initial_stats, iflags, NULL); |
| |
| if (fd1 != -1) |
| close(fd1); |
| if (close(fd2) < 0) { |
| rsyserr(FERROR, errno, "close failed on %s", |
| full_fname(fnametmp)); |
| exit_cleanup(RERR_FILEIO); |
| } |
| |
| if ((recv_ok && (!delay_updates || !partialptr)) || inplace) { |
| if (partialptr == fname) |
| partialptr = NULL; |
| if (!finish_transfer(fname, fnametmp, fnamecmp, |
| partialptr, file, recv_ok, 1)) |
| recv_ok = -1; |
| else if (fnamecmp == partialptr) { |
| do_unlink(partialptr); |
| handle_partial_dir(partialptr, PDIR_DELETE); |
| } |
| } else if (keep_partial && partialptr |
| && handle_partial_dir(partialptr, PDIR_CREATE)) { |
| if (!finish_transfer(partialptr, fnametmp, fnamecmp, NULL, |
| file, recv_ok, !partial_dir)) |
| recv_ok = -1; |
| else if (delay_updates && recv_ok) { |
| bitbag_set_bit(delayed_bits, ndx); |
| recv_ok = 2; |
| } |
| } else { |
| partialptr = NULL; |
| do_unlink(fnametmp); |
| } |
| |
| cleanup_disable(); |
| |
| switch (recv_ok) { |
| case 1: |
| if (remove_source_files || inc_recurse |
| || (preserve_hard_links && F_IS_HLINKED(file))) |
| send_msg_int(MSG_SUCCESS, ndx); |
| break; |
| case 0: { |
| enum logcode msgtype = redoing ? FERROR_XFER : FWARNING; |
| if (msgtype == FERROR_XFER || verbose) { |
| char *errstr, *redostr, *keptstr; |
| if (!(keep_partial && partialptr) && !inplace) |
| keptstr = "discarded"; |
| else if (partial_dir) |
| keptstr = "put into partial-dir"; |
| else |
| keptstr = "retained"; |
| if (msgtype == FERROR_XFER) { |
| errstr = "ERROR"; |
| redostr = ""; |
| } else { |
| errstr = "WARNING"; |
| redostr = read_batch ? " (may try again)" |
| : " (will try again)"; |
| } |
| rprintf(msgtype, |
| "%s: %s failed verification -- update %s%s.\n", |
| errstr, local_name ? f_name(file, NULL) : fname, |
| keptstr, redostr); |
| } |
| if (!redoing) { |
| send_msg_int(MSG_REDO, ndx); |
| file->flags |= FLAG_FILE_SENT; |
| } else if (inc_recurse) |
| send_msg_int(MSG_NO_SEND, ndx); |
| break; |
| } |
| case -1: |
| if (inc_recurse) |
| send_msg_int(MSG_NO_SEND, ndx); |
| break; |
| } |
| } |
| if (make_backups < 0) |
| make_backups = -make_backups; |
| |
| if (phase == 2 && delay_updates) /* for protocol_version < 29 */ |
| handle_delayed_updates(local_name); |
| |
| if (verbose > 2) |
| rprintf(FINFO,"recv_files finished\n"); |
| |
| return 0; |
| } |