Annotation of embedaddon/rsync/patches/backup-dir-dels.diff, revision 1.1
1.1 ! misho 1: This patches creates two new command line options as follows:
! 2: --backup-dir-dels=DIR
! 3: --suffix-dels=SUFFIX
! 4:
! 5: The backup-dir-dels and suffix-dels options give the ability to store
! 6: backup of removed files on the receiver in different directories or with
! 7: different suffix than the backup of files that have been changed but that
! 8: are still on the source drive. Both commands can be combined.
! 9:
! 10: The default behaviour if one or both of the options are not specified
! 11: is the previous behaviour, both backups use the same directory or
! 12: suffix.
! 13:
! 14: Marc St-Onge
! 15:
! 16: To use this patch, run these commands for a successful build:
! 17:
! 18: patch -p1 <patches/backup-deleted.diff
! 19: patch -p1 <patches/backup-dir-dels.diff
! 20: ./configure (optional if already run)
! 21: make
! 22:
! 23: based-on: patch/master/backup-deleted
! 24: diff --git a/backup.c b/backup.c
! 25: --- a/backup.c
! 26: +++ b/backup.c
! 27: @@ -29,25 +29,32 @@ extern int preserve_specials;
! 28: extern int preserve_links;
! 29: extern int safe_symlinks;
! 30: extern int backup_dir_len;
! 31: +extern int backup_dir_dels_len;
! 32: extern unsigned int backup_dir_remainder;
! 33: +extern unsigned int backup_dir_dels_remainder;
! 34: extern char backup_dir_buf[MAXPATHLEN];
! 35: +extern char backup_dir_dels_buf[MAXPATHLEN];
! 36: extern char *backup_suffix;
! 37: +extern char *backup_suffix_dels;
! 38: extern char *backup_dir;
! 39: +extern char *backup_dir_dels;
! 40: +
! 41: +static BOOL deleting;
! 42:
! 43: /* Returns -1 on error, 0 on missing dir, and 1 on present dir. */
! 44: -static int validate_backup_dir(void)
! 45: +static int validate_backup_dir(char *buf)
! 46: {
! 47: STRUCT_STAT st;
! 48:
! 49: - if (do_lstat(backup_dir_buf, &st) < 0) {
! 50: + if (do_lstat(buf, &st) < 0) {
! 51: if (errno == ENOENT)
! 52: return 0;
! 53: - rsyserr(FERROR, errno, "backup lstat %s failed", backup_dir_buf);
! 54: + rsyserr(FERROR, errno, "backup lstat %s failed", buf);
! 55: return -1;
! 56: }
! 57: if (!S_ISDIR(st.st_mode)) {
! 58: int flags = get_del_for_flag(st.st_mode) | DEL_FOR_BACKUP | DEL_RECURSE;
! 59: - if (delete_item(backup_dir_buf, st.st_mode, flags) == 0)
! 60: + if (delete_item(buf, st.st_mode, flags) == 0)
! 61: return 0;
! 62: return -1;
! 63: }
! 64: @@ -58,20 +65,20 @@ static int validate_backup_dir(void)
! 65: * backup_dir_buf. Any new directories (compared to the prior backup
! 66: * path) are ensured to exist as directories, replacing anything else
! 67: * that may be in the way (e.g. a symlink). */
! 68: -static BOOL copy_valid_path(const char *fname)
! 69: +static BOOL copy_valid_path(const char *fname, char *buf, int prefix_len, unsigned int remainder, const char *suffix)
! 70: {
! 71: const char *f;
! 72: int val;
! 73: BOOL ret = True;
! 74: stat_x sx;
! 75: - char *b, *rel = backup_dir_buf + backup_dir_len, *name = rel;
! 76: + char *b, *rel = buf + prefix_len, *name = rel;
! 77:
! 78: for (f = fname, b = rel; *f && *f == *b; f++, b++) {
! 79: if (*b == '/')
! 80: name = b + 1;
! 81: }
! 82:
! 83: - if (stringjoin(rel, backup_dir_remainder, fname, backup_suffix, NULL) >= backup_dir_remainder) {
! 84: + if (stringjoin(rel, remainder, fname, suffix, NULL) >= remainder) {
! 85: rprintf(FERROR, "backup filename too long\n");
! 86: *name = '\0';
! 87: return False;
! 88: @@ -82,7 +89,7 @@ static BOOL copy_valid_path(const char *fname)
! 89: return True;
! 90: *b = '\0';
! 91:
! 92: - val = validate_backup_dir();
! 93: + val = validate_backup_dir(buf);
! 94: if (val == 0)
! 95: break;
! 96: if (val < 0) {
! 97: @@ -98,9 +105,9 @@ static BOOL copy_valid_path(const char *fname)
! 98: for ( ; b; name = b + 1, b = strchr(name, '/')) {
! 99: *b = '\0';
! 100:
! 101: - while (do_mkdir(backup_dir_buf, ACCESSPERMS) < 0) {
! 102: + while (do_mkdir(buf, ACCESSPERMS) < 0) {
! 103: if (errno == EEXIST) {
! 104: - val = validate_backup_dir();
! 105: + val = validate_backup_dir(buf);
! 106: if (val > 0)
! 107: break;
! 108: if (val == 0)
! 109: @@ -134,7 +141,7 @@ static BOOL copy_valid_path(const char *fname)
! 110: free_xattr(&sx);
! 111: }
! 112: #endif
! 113: - set_file_attrs(backup_dir_buf, file, NULL, NULL, 0);
! 114: + set_file_attrs(buf, file, NULL, NULL, 0);
! 115: unmake_file(file);
! 116: }
! 117:
! 118: @@ -156,7 +163,12 @@ static BOOL copy_valid_path(const char *fname)
! 119: /* Make a complete pathname for backup file and verify any new path elements. */
! 120: char *get_backup_name(const char *fname)
! 121: {
! 122: + char *buf = deleting ? backup_dir_dels_buf : backup_dir_buf;
! 123: + char *suffix = deleting ? backup_suffix_dels : backup_suffix;
! 124: +
! 125: if (backup_dir) {
! 126: + int prefix_len = deleting ? backup_dir_dels_len : backup_dir_len;
! 127: + unsigned int remainder = deleting ? backup_dir_dels_remainder : backup_dir_remainder;
! 128: static int initialized = 0;
! 129: if (!initialized) {
! 130: int ret;
! 131: @@ -170,14 +182,14 @@ char *get_backup_name(const char *fname)
! 132: initialized = 1;
! 133: }
! 134: /* copy fname into backup_dir_buf while validating the dirs. */
! 135: - if (copy_valid_path(fname))
! 136: - return backup_dir_buf;
! 137: + if (copy_valid_path(fname, buf, prefix_len, remainder, suffix))
! 138: + return buf;
! 139: /* copy_valid_path() has printed an error message. */
! 140: return NULL;
! 141: }
! 142:
! 143: - if (stringjoin(backup_dir_buf, MAXPATHLEN, fname, backup_suffix, NULL) < MAXPATHLEN)
! 144: - return backup_dir_buf;
! 145: + if (stringjoin(backup_dir_buf, MAXPATHLEN, fname, suffix, NULL) < MAXPATHLEN)
! 146: + return buf;
! 147:
! 148: rprintf(FERROR, "backup filename too long\n");
! 149: return NULL;
! 150: @@ -352,3 +364,13 @@ int make_backup(const char *fname, BOOL prefer_rename)
! 151: rprintf(FINFO, "backed up %s to %s\n", fname, buf);
! 152: return ret;
! 153: }
! 154: +
! 155: +/* backup switch routine called only when backing-up removed file */
! 156: +int safe_delete(const char *fname)
! 157: +{
! 158: + int ret;
! 159: + deleting = 1;
! 160: + ret = make_backup(fname, True);
! 161: + deleting = 0;
! 162: + return ret;
! 163: +}
! 164: diff --git a/delete.c b/delete.c
! 165: --- a/delete.c
! 166: +++ b/delete.c
! 167: @@ -28,16 +28,23 @@ extern int max_delete;
! 168: extern char *backup_dir;
! 169: extern char *backup_suffix;
! 170: extern int backup_suffix_len;
! 171: +extern char *backup_dir_dels;
! 172: +extern char *backup_suffix_dels;
! 173: +extern int backup_suffix_dels_len;
! 174: extern struct stats stats;
! 175:
! 176: int ignore_perishable = 0;
! 177: int non_perishable_cnt = 0;
! 178: int skipped_deletes = 0;
! 179:
! 180: +/* Function now compares both backup_suffix and backup_suffix_dels. */
! 181: static inline int is_backup_file(char *fn)
! 182: {
! 183: int k = strlen(fn) - backup_suffix_len;
! 184: - return k > 0 && strcmp(fn+k, backup_suffix) == 0;
! 185: + if (k > 0 && strcmp(fn+k, backup_suffix) == 0)
! 186: + return 1;
! 187: + k += backup_suffix_len - backup_suffix_dels_len;
! 188: + return k > 0 && strcmp(fn+k, backup_suffix_dels) == 0;
! 189: }
! 190:
! 191: /* The directory is about to be deleted: if DEL_RECURSE is given, delete all
! 192: @@ -162,9 +169,9 @@ enum delret delete_item(char *fbuf, uint16 mode, uint16 flags)
! 193: what = "rmdir";
! 194: ok = do_rmdir(fbuf) == 0;
! 195: } else {
! 196: - if (make_backups > 0 && !(flags & DEL_FOR_BACKUP) && (backup_dir || !is_backup_file(fbuf))) {
! 197: + if (make_backups > 0 && !(flags & DEL_FOR_BACKUP) && (backup_dir_dels || !is_backup_file(fbuf))) {
! 198: what = "make_backup";
! 199: - ok = make_backup(fbuf, True);
! 200: + ok = safe_delete(fbuf);
! 201: if (ok == 2) {
! 202: what = "unlink";
! 203: ok = robust_unlink(fbuf) == 0;
! 204: diff --git a/options.c b/options.c
! 205: --- a/options.c
! 206: +++ b/options.c
! 207: @@ -157,10 +157,14 @@ int no_detach
! 208: int write_batch = 0;
! 209: int read_batch = 0;
! 210: int backup_dir_len = 0;
! 211: +int backup_dir_dels_len = 0;
! 212: int backup_suffix_len;
! 213: +int backup_suffix_dels_len;
! 214: unsigned int backup_dir_remainder;
! 215: +unsigned int backup_dir_dels_remainder;
! 216:
! 217: char *backup_suffix = NULL;
! 218: +char *backup_suffix_dels = NULL;
! 219: char *tmpdir = NULL;
! 220: char *partial_dir = NULL;
! 221: char *basis_dir[MAX_BASIS_DIRS+1];
! 222: @@ -173,7 +177,9 @@ char *password_file = NULL;
! 223: char *early_input_file = NULL;
! 224: char *rsync_path = RSYNC_PATH;
! 225: char *backup_dir = NULL;
! 226: +char *backup_dir_dels = NULL;
! 227: char backup_dir_buf[MAXPATHLEN];
! 228: +char backup_dir_dels_buf[MAXPATHLEN];
! 229: char *sockopts = NULL;
! 230: char *usermap = NULL;
! 231: char *groupmap = NULL;
! 232: @@ -770,7 +776,9 @@ static struct poptOption long_options[] = {
! 233: {"backup-deleted", 0, POPT_ARG_VAL, &make_backups, 1, 0, 0 },
! 234: {"no-backup", 0, POPT_ARG_VAL, &make_backups, 0, 0, 0 },
! 235: {"backup-dir", 0, POPT_ARG_STRING, &backup_dir, 0, 0, 0 },
! 236: + {"backup-dir-dels", 0, POPT_ARG_STRING, &backup_dir_dels, 0, 0, 0 },
! 237: {"suffix", 0, POPT_ARG_STRING, &backup_suffix, 0, 0, 0 },
! 238: + {"suffix-dels", 0, POPT_ARG_STRING, &backup_suffix_dels, 0, 0, 0 },
! 239: {"list-only", 0, POPT_ARG_VAL, &list_only, 2, 0, 0 },
! 240: {"read-batch", 0, POPT_ARG_STRING, &batch_name, OPT_READ_BATCH, 0, 0 },
! 241: {"write-batch", 0, POPT_ARG_STRING, &batch_name, OPT_WRITE_BATCH, 0, 0 },
! 242: @@ -2187,6 +2195,8 @@ int parse_arguments(int *argc_p, const char ***argv_p)
! 243: tmpdir = sanitize_path(NULL, tmpdir, NULL, 0, SP_DEFAULT);
! 244: if (backup_dir)
! 245: backup_dir = sanitize_path(NULL, backup_dir, NULL, 0, SP_DEFAULT);
! 246: + if (backup_dir_dels)
! 247: + backup_dir_dels = sanitize_path(NULL, backup_dir_dels, NULL, 0, SP_DEFAULT);
! 248: }
! 249: if (daemon_filter_list.head && !am_sender) {
! 250: filter_rule_list *elp = &daemon_filter_list;
! 251: @@ -2208,6 +2218,14 @@ int parse_arguments(int *argc_p, const char ***argv_p)
! 252: if (check_filter(elp, FLOG, dir, 1) < 0)
! 253: goto options_rejected;
! 254: }
! 255: + /* Clean backup_dir_dels same as for backup_dir */
! 256: + if (backup_dir_dels) {
! 257: + if (!*backup_dir_dels)
! 258: + goto options_rejected;
! 259: + clean_fname(backup_dir_dels, 1);
! 260: + if (check_filter(elp, FLOG, backup_dir_dels, 1) < 0)
! 261: + goto options_rejected;
! 262: + }
! 263: }
! 264:
! 265: if (!backup_suffix)
! 266: @@ -2219,6 +2237,20 @@ int parse_arguments(int *argc_p, const char ***argv_p)
! 267: backup_suffix);
! 268: return 0;
! 269: }
! 270: + /* --suffix-dels defaults to --suffix, or empty for a client given an
! 271: + * explicit --backup-dir-dels (just as --suffix defaults to empty when
! 272: + * a --backup-dir is given). The second case does not apply to the
! 273: + * server for consistency with server_options, which sends --suffix-dels
! 274: + * to the server iff it differs from --suffix. */
! 275: + if (!backup_suffix_dels)
! 276: + backup_suffix_dels = backup_dir_dels && !am_server ? "" : backup_suffix;
! 277: + backup_suffix_dels_len = strlen(backup_suffix_dels);
! 278: + if (strchr(backup_suffix_dels, '/') != NULL) {
! 279: + snprintf(err_buf, sizeof err_buf,
! 280: + "--suffix-dels cannot contain slashes: %s\n",
! 281: + backup_suffix_dels);
! 282: + return 0;
! 283: + }
! 284: if (backup_dir) {
! 285: size_t len;
! 286: make_backups = 1; /* --backup-dir implies --backup */
! 287: @@ -2255,6 +2287,34 @@ int parse_arguments(int *argc_p, const char ***argv_p)
! 288: "P *%s", backup_suffix);
! 289: parse_filter_str(&filter_list, backup_dir_buf, rule_template(0), 0);
! 290: }
! 291: + if (backup_dir_dels) {
! 292: + backup_dir_dels_len = strlcpy(backup_dir_dels_buf, backup_dir_dels, sizeof backup_dir_dels_buf);
! 293: + backup_dir_dels_remainder = sizeof backup_dir_dels_buf - backup_dir_dels_len;
! 294: + if (backup_dir_dels_remainder < 32) {
! 295: + snprintf(err_buf, sizeof err_buf,
! 296: + "the --backup-dir-dels path is WAY too long.\n");
! 297: + return 0;
! 298: + }
! 299: + if (backup_dir_dels_buf[backup_dir_dels_len - 1] != '/') {
! 300: + backup_dir_dels_buf[backup_dir_dels_len++] = '/';
! 301: + backup_dir_dels_buf[backup_dir_dels_len] = '\0';
! 302: + }
! 303: + if (INFO_GTE(BACKUP, 1) && !am_sender)
! 304: + rprintf(FINFO, "backup_dir_dels is %s\n", backup_dir_dels_buf);
! 305: + } else if (backup_dir) {
! 306: + backup_dir_dels = backup_dir;
! 307: + backup_dir_dels_len = backup_dir_len;
! 308: + backup_dir_dels_remainder = backup_dir_remainder;
! 309: + strlcpy(backup_dir_dels_buf, backup_dir_buf, sizeof backup_dir_buf);
! 310: + } else if (!backup_suffix_dels_len && (!am_server || !am_sender)) {
! 311: + snprintf(err_buf, sizeof err_buf,
! 312: + "--suffix-dels cannot be a null string without --backup-dir-dels\n");
! 313: + return 0;
! 314: + } else if (make_backups && delete_mode && !delete_excluded && !am_server) {
! 315: + snprintf(backup_dir_dels_buf, sizeof backup_dir_dels_buf,
! 316: + "P *%s", backup_suffix_dels);
! 317: + parse_filter_str(&filter_list, backup_dir_dels_buf, rule_template(0), 0);
! 318: + }
! 319:
! 320: if (preserve_times) {
! 321: preserve_times = PRESERVE_FILE_TIMES;
! 322: @@ -2701,6 +2761,10 @@ void server_options(char **args, int *argc_p)
! 323: args[ac++] = "--backup-dir";
! 324: args[ac++] = backup_dir;
! 325: }
! 326: + if (backup_dir_dels && backup_dir_dels != backup_dir) {
! 327: + args[ac++] = "--backup-dir-dels";
! 328: + args[ac++] = backup_dir_dels;
! 329: + }
! 330:
! 331: /* Only send --suffix if it specifies a non-default value. */
! 332: if (strcmp(backup_suffix, backup_dir ? "" : BACKUP_SUFFIX) != 0) {
! 333: @@ -2709,6 +2773,14 @@ void server_options(char **args, int *argc_p)
! 334: goto oom;
! 335: args[ac++] = arg;
! 336: }
! 337: + /* Only send --suffix-dels if it specifies a value different from the
! 338: + * --suffix value, which would normally be used for deletions too. */
! 339: + if (strcmp(backup_suffix_dels, backup_suffix) != 0) {
! 340: + /* We use the following syntax to avoid weirdness with '~'. */
! 341: + if (asprintf(&arg, "--suffix-dels=%s", backup_suffix_dels) < 0)
! 342: + goto oom;
! 343: + args[ac++] = arg;
! 344: + }
! 345:
! 346: if (checksum_choice) {
! 347: if (asprintf(&arg, "--checksum-choice=%s", checksum_choice) < 0)
! 348: diff --git a/rsync.1.md b/rsync.1.md
! 349: --- a/rsync.1.md
! 350: +++ b/rsync.1.md
! 351: @@ -346,7 +346,9 @@ detailed description below for a complete description.
! 352: --backup, -b make backups (see --suffix & --backup-dir)
! 353: --backup-deleted make backups only of deleted files
! 354: --backup-dir=DIR make backups into hierarchy based in DIR
! 355: +--backup-dir-dels=DIR backup removed files into hierarchy based in DIR
! 356: --suffix=SUFFIX backup suffix (default ~ w/o --backup-dir)
! 357: +--suffix-dels=SUFFIX set removed-files suffix (def. --suffix w/o b-d-d)
! 358: --update, -u skip files that are newer on the receiver
! 359: --inplace update destination files in-place
! 360: --append append data onto shorter files
! 361: diff -Nurp a/rsync.1 b/rsync.1
! 362: --- a/rsync.1
! 363: +++ b/rsync.1
! 364: @@ -422,7 +422,9 @@ detailed description below for a complet
! 365: --backup, -b make backups (see --suffix & --backup-dir)
! 366: --backup-deleted make backups only of deleted files
! 367: --backup-dir=DIR make backups into hierarchy based in DIR
! 368: +--backup-dir-dels=DIR backup removed files into hierarchy based in DIR
! 369: --suffix=SUFFIX backup suffix (default ~ w/o --backup-dir)
! 370: +--suffix-dels=SUFFIX set removed-files suffix (def. --suffix w/o b-d-d)
! 371: --update, -u skip files that are newer on the receiver
! 372: --inplace update destination files in-place
! 373: --append append data onto shorter files
! 374: diff -Nurp a/rsync.1.html b/rsync.1.html
! 375: --- a/rsync.1.html
! 376: +++ b/rsync.1.html
! 377: @@ -337,7 +337,9 @@ detailed description below for a complet
! 378: --backup, -b make backups (see --suffix & --backup-dir)
! 379: --backup-deleted make backups only of deleted files
! 380: --backup-dir=DIR make backups into hierarchy based in DIR
! 381: +--backup-dir-dels=DIR backup removed files into hierarchy based in DIR
! 382: --suffix=SUFFIX backup suffix (default ~ w/o --backup-dir)
! 383: +--suffix-dels=SUFFIX set removed-files suffix (def. --suffix w/o b-d-d)
! 384: --update, -u skip files that are newer on the receiver
! 385: --inplace update destination files in-place
! 386: --append append data onto shorter files
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>