Browse Source

Merge remote-tracking branch 'hakre/patch-2'

Jordi Boggiano 10 years ago
parent
commit
6e4a21d700
2 changed files with 81 additions and 2 deletions
  1. 47 2
      src/Composer/Util/Filesystem.php
  2. 34 0
      tests/Composer/Test/Util/FilesystemTest.php

+ 47 - 2
src/Composer/Util/Filesystem.php

@@ -95,8 +95,8 @@ class Filesystem
      */
      */
     public function removeDirectory($directory)
     public function removeDirectory($directory)
     {
     {
-        if (file_exists($directory) && is_link($directory)) {
-            return $this->unlink($directory);
+        if ($this->isSymlinkedDirectory($directory)) {
+            return $this->unlinkSymlinkedDirectory($directory);
         }
         }
 
 
         if (!file_exists($directory) || !is_dir($directory)) {
         if (!file_exists($directory) || !is_dir($directory)) {
@@ -507,4 +507,49 @@ class Filesystem
 
 
         return unlink($path);
         return unlink($path);
     }
     }
+
+    private function isSymlinkedDirectory($directory)
+    {
+        if (!is_dir($directory)) {
+            return false;
+        }
+
+        $resolved = $this->resolveSymlinkedDirectorySymlink($directory);
+
+        return is_link($resolved);
+    }
+
+    /**
+     * @param string $directory
+     *
+     * @return bool
+     */
+    private function unlinkSymlinkedDirectory($directory)
+    {
+        $resolved = $this->resolveSymlinkedDirectorySymlink($directory);
+
+        return $this->unlink($resolved);
+    }
+
+    /**
+     * resolve pathname to symbolic link of a directory
+     *
+     * @param string $pathname directory path to resolve
+     *
+     * @return string resolved path to symbolic link or original pathname (unresolved)
+     */
+    private function resolveSymlinkedDirectorySymlink($pathname)
+    {
+        if (!is_dir($pathname)) {
+            return $pathname;
+        }
+
+        $resolved = rtrim($pathname, '/');
+
+        if (!strlen($resolved)) {
+            return $pathname;
+        }
+
+        return $resolved;
+    }
 }
 }

+ 34 - 0
tests/Composer/Test/Util/FilesystemTest.php

@@ -203,4 +203,38 @@ class FilesystemTest extends TestCase
         $this->assertTrue($result);
         $this->assertTrue($result);
         $this->assertFalse(file_exists($symlinked));
         $this->assertFalse(file_exists($symlinked));
     }
     }
+
+    /**
+     * @link https://github.com/composer/composer/issues/3144
+     */
+    public function testRemoveSymlinkedDirectoryWithTrailingSlash()
+    {
+        $tmp = sys_get_temp_dir();
+        $basepath = $tmp . "/composer_testdir";
+        @mkdir($basepath . "/real", 0777, true);
+        touch($basepath . "/real/FILE");
+        $symlinked              = $basepath . "/linked";
+        $symlinkedTrailingSlash = $symlinked . "/";
+
+        $result = @symlink($basepath . "/real", $symlinked);
+
+        if (!$result) {
+            $this->markTestSkipped('Symbolic links for directories not supported on this platform');
+        }
+
+        if (!is_dir($symlinked)) {
+            $this->fail('Precondition assertion failed (is_dir is false on symbolic link to directory).');
+        }
+
+        if (!is_dir($symlinkedTrailingSlash)) {
+            $this->fail('Precondition assertion failed (is_dir false w trailing slash).');
+        }
+
+        $fs = new Filesystem();
+
+        $result = $fs->removeDirectory($symlinkedTrailingSlash);
+        $this->assertTrue($result);
+        $this->assertFalse(file_exists($symlinkedTrailingSlash));
+        $this->assertFalse(file_exists($symlinked));
+    }
 }
 }