File:  [ELWIX - Embedded LightWeight unIX -] / embedaddon / rsync / patches / backup-dir-dels.diff
Revision 1.1.1.1 (vendor branch): download - view: text, annotated - select for diffs - revision graph
Wed Mar 17 00:32:36 2021 UTC (4 years ago) by misho
Branches: rsync, MAIN
CVS tags: v3_2_3, HEAD
rsync 3.2.3

    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 &amp; --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>