realpath: remove extraneous '/' for --relative-to edge cases
authorPádraig Brady <P@draigBrady.com>
Wed, 25 Jan 2012 16:42:42 +0000 (16:42 +0000)
committerPádraig Brady <P@draigBrady.com>
Wed, 25 Jan 2012 18:01:52 +0000 (18:01 +0000)
* src/realpath.c (path_common_prefix): Be consistent and
always include a leading '/' in the count returned.
(relpath): Account for the change in path_common_prefix()
and avoid outputting extra '/' chars in relative paths that
span the root dir.
* tests/misc/realpath: Add the two reported cases.
Reported by Mike Frysinger

src/realpath.c
tests/misc/realpath

index b39f7bf..b03f375 100644 (file)
@@ -136,7 +136,7 @@ path_common_prefix (const char *path1, const char *path2)
       if (*path1 != *path2)
         break;
       if (*path1 == '/')
-        ret = i;
+        ret = i + 1;
       path1++;
       path2++;
       i++;
@@ -171,11 +171,16 @@ relpath (const char *can_fname)
       const char *relto_suffix = can_relative_to + common_index;
       const char *fname_suffix = can_fname + common_index;
 
+      /* skip over extraneous '/'.  */
+      if (*relto_suffix == '/')
+        relto_suffix++;
+      if (*fname_suffix == '/')
+        fname_suffix++;
+
       /* Replace remaining components of --relative-to with '..', to get
          to a common directory.  Then output the remainder of fname.  */
       if (*relto_suffix)
         {
-          ++relto_suffix;
           printf ("%s", "..");
           for (; *relto_suffix; ++relto_suffix)
             {
@@ -183,12 +188,13 @@ relpath (const char *can_fname)
                 printf ("%s", "/..");
             }
 
-          printf ("%s", fname_suffix);
+          if (*fname_suffix)
+            printf ("/%s", fname_suffix);
         }
       else
         {
           if (*fname_suffix)
-            printf ("%s", ++fname_suffix);
+            printf ("%s", fname_suffix);
           else
             printf ("%c", '.');
         }
index fb01393..8a1f336 100755 (executable)
@@ -41,13 +41,18 @@ realpath -m '' && fail=1
 
 # symlink resolution
 this=$(realpath .)
-test "$(realpath $relative ldir2/..)" = "$this/dir1" || fail=1
-test "$(realpath -L $relative ldir2/..)" = "$this" || fail=1
-test "$(realpath -s $relative ldir2)" = "$this/ldir2" || fail=1
+test "$(realpath ldir2/..)" = "$this/dir1" || fail=1
+test "$(realpath -L ldir2/..)" = "$this" || fail=1
+test "$(realpath -s ldir2)" = "$this/ldir2" || fail=1
 
 # relative string handling
 test $(realpath -m --relative-to=prefix prefixed/1) = '../prefixed/1' || fail=1
 test $(realpath -m --relative-to=prefixed prefix/1) = '../prefix/1' || fail=1
 test $(realpath -m --relative-to=prefixed prefixed/1) = '1' || fail=1
 
+# Ensure no redundant trailing '/' present, as was the case in v8.15
+test $(realpath -sm --relative-to=/usr /) = '..' || fail=1
+# Ensure no redundant leading '../' present, as was the case in v8.15
+test $(realpath -sm --relative-to=/ /usr) = 'usr' || fail=1
+
 Exit $fail