1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116
|
The attached patch gives the user ability to define how many symbolic
links rsync should follow before actually adding it to the file list.
The patch has been heavily modified from its original form to work
with the latest codebase, but even in its original form it didn't
handle relative symlinks properly, and that has not yet been fixed
in this modified version.
To use this patch, run these commands for a successful build:
patch -p1 <patches/links-depth.diff
./configure (optional if already run)
make
--- old/flist.c
+++ new/flist.c
@@ -41,6 +41,7 @@ extern int one_file_system;
extern int copy_dirlinks;
extern int keep_dirlinks;
extern int preserve_links;
+extern int follow_links_depth;
extern int preserve_hard_links;
extern int preserve_devices;
extern int preserve_specials;
@@ -702,6 +703,30 @@ static struct file_struct *receive_file_
return file;
}
+#if SUPPORT_LINKS
+static int links_depth(char *linkname, STRUCT_STAT *st_ptr)
+{
+ char buf[MAXPATHLEN];
+ STRUCT_STAT st;
+ int i;
+
+ for (i = 0; i < follow_links_depth; i++) {
+ /* XXX This doesn't handle relative symlinks! */
+ if (readlink_stat(linkname, &st, buf) != 0)
+ break;
+ *st_ptr = st;
+ if (!S_ISLNK(st.st_mode))
+ return 1;
+ strlcpy(linkname, buf, MAXPATHLEN);
+#if 0
+ fprintf(stderr, "\n%s:%i [#%i] %s -> %s\n", __FILE__, __LINE__, i, file->u.link, linkname);
+#endif
+ }
+
+ return 0;
+}
+#endif
+
/**
* Create a file_struct for a named file by reading its stat()
* information and performing extensive checks against global
@@ -837,7 +862,13 @@ struct file_struct *make_file(char *fnam
basename_len = strlen(basename) + 1; /* count the '\0' */
#ifdef SUPPORT_LINKS
- linkname_len = S_ISLNK(st.st_mode) ? strlen(linkname) + 1 : 0;
+ if (S_ISLNK(st.st_mode)) {
+ if (follow_links_depth && links_depth(linkname, &st))
+ linkname_len = 0;
+ else
+ linkname_len = strlen(linkname) + 1;
+ } else
+ linkname_len = 0;
#else
linkname_len = 0;
#endif
--- old/options.c
+++ new/options.c
@@ -46,6 +46,7 @@ int keep_dirlinks = 0;
int copy_dirlinks = 0;
int copy_links = 0;
int preserve_links = 0;
+int follow_links_depth = 0;
int preserve_hard_links = 0;
int preserve_perms = 0;
int preserve_executability = 0;
@@ -297,6 +298,7 @@ void usage(enum logcode F)
rprintf(F," --append append data onto shorter files\n");
rprintf(F," -d, --dirs transfer directories without recursing\n");
rprintf(F," -l, --links copy symlinks as symlinks\n");
+ rprintf(F," --links-depth=NUM follow symlinks up to NUM depth\n");
rprintf(F," -L, --copy-links transform symlink into referent file/dir\n");
rprintf(F," --copy-unsafe-links only \"unsafe\" symlinks are transformed\n");
rprintf(F," --safe-links ignore symlinks that point outside the source tree\n");
@@ -447,6 +449,7 @@ static struct poptOption long_options[]
{"links", 'l', POPT_ARG_VAL, &preserve_links, 1, 0, 0 },
{"no-links", 0, POPT_ARG_VAL, &preserve_links, 0, 0, 0 },
{"no-l", 0, POPT_ARG_VAL, &preserve_links, 0, 0, 0 },
+ {"links-depth", 0, POPT_ARG_INT, &follow_links_depth , 0, 0, 0 },
{"copy-links", 'L', POPT_ARG_NONE, ©_links, 0, 0, 0 },
{"copy-unsafe-links",0, POPT_ARG_NONE, ©_unsafe_links, 0, 0, 0 },
{"safe-links", 0, POPT_ARG_NONE, &safe_symlinks, 0, 0, 0 },
--- old/rsync.yo
+++ new/rsync.yo
@@ -314,6 +314,7 @@ to the detailed description below for a
--append append data onto shorter files
-d, --dirs transfer directories without recursing
-l, --links copy symlinks as symlinks
+ --links-depth=NUM follow symlinks up to NUM depth
-L, --copy-links transform symlink into referent file/dir
--copy-unsafe-links only "unsafe" symlinks are transformed
--safe-links ignore symlinks that point outside the tree
--- old/rsync.1
+++ new/rsync.1
@@ -380,6 +380,7 @@ to the detailed description below for a
\-\-append append data onto shorter files
\-d, \-\-dirs transfer directories without recursing
\-l, \-\-links copy symlinks as symlinks
+ \-\-links\-depth=NUM follow symlinks up to NUM depth
\-L, \-\-copy\-links transform symlink into referent file/dir
\-\-copy\-unsafe\-links only "unsafe" symlinks are transformed
\-\-safe\-links ignore symlinks that point outside the tree
|