From cc81f5bac395dbf49ecf6dfd780e01fed9e4b2c3 Mon Sep 17 00:00:00 2001 From: Jordi Boggiano Date: Mon, 12 Jul 2021 13:36:57 +0200 Subject: [PATCH 1/2] Fix support for UNC paths in normalizePath, refs #9993 --- src/Composer/Util/Filesystem.php | 14 ++++++++++---- tests/Composer/Test/Util/FilesystemTest.php | 1 + 2 files changed, 11 insertions(+), 4 deletions(-) diff --git a/src/Composer/Util/Filesystem.php b/src/Composer/Util/Filesystem.php index 2f5192096..69f097958 100644 --- a/src/Composer/Util/Filesystem.php +++ b/src/Composer/Util/Filesystem.php @@ -544,7 +544,13 @@ class Filesystem $parts = array(); $path = strtr($path, '\\', '/'); $prefix = ''; - $absolute = false; + $absolute = ''; + + // extract windows UNC paths e.g. \\foo\bar + if (strpos($path, '//') === 0 && \strlen($path) > 2) { + $absolute = '//'; + $path = substr($path, 2); + } // extract a prefix being a protocol://, protocol:, protocol://drive: or simply drive: if (preg_match('{^( [0-9a-z]{2,}+: (?: // (?: [a-z]: )? )? | [a-z]: )}ix', $path, $match)) { @@ -553,13 +559,13 @@ class Filesystem } if (strpos($path, '/') === 0) { - $absolute = true; + $absolute = '/'; $path = substr($path, 1); } $up = false; foreach (explode('/', $path) as $chunk) { - if ('..' === $chunk && ($absolute || $up)) { + if ('..' === $chunk && ($absolute !== '' || $up)) { array_pop($parts); $up = !(empty($parts) || '..' === end($parts)); } elseif ('.' !== $chunk && '' !== $chunk) { @@ -568,7 +574,7 @@ class Filesystem } } - return $prefix.($absolute ? '/' : '').implode('/', $parts); + return $prefix.((string) $absolute).implode('/', $parts); } /** diff --git a/tests/Composer/Test/Util/FilesystemTest.php b/tests/Composer/Test/Util/FilesystemTest.php index 0fb4babd8..7ac013a78 100644 --- a/tests/Composer/Test/Util/FilesystemTest.php +++ b/tests/Composer/Test/Util/FilesystemTest.php @@ -220,6 +220,7 @@ class FilesystemTest extends TestCase array('../src', 'Foo/Bar/../../../src'), array('c:../b', 'c:.\\..\\a\\..\\b'), array('phar://c:../Foo', 'phar://c:../Foo'), + array('//foo/bar', '\\\\foo\\bar'), ); } From 005c55185af77c81c438074c0e10041c1c4e6324 Mon Sep 17 00:00:00 2001 From: Jordi Boggiano Date: Mon, 12 Jul 2021 13:54:01 +0200 Subject: [PATCH 2/2] Fix support for writing into UNC paths, and comparing UNC paths correctly in InstalledVersions, fixes #9993 --- src/Composer/Factory.php | 11 +++++-- .../Repository/FilesystemRepository.php | 32 ++++++++++++------- tests/Composer/Test/Mock/FactoryMock.php | 2 +- .../Repository/FilesystemRepositoryTest.php | 9 ++++-- 4 files changed, 37 insertions(+), 17 deletions(-) diff --git a/src/Composer/Factory.php b/src/Composer/Factory.php index f4fe06274..6aed45454 100644 --- a/src/Composer/Factory.php +++ b/src/Composer/Factory.php @@ -385,7 +385,7 @@ class Factory $composer->setPackage($package); // load local repository - $this->addLocalRepository($io, $rm, $vendorDir, $package); + $this->addLocalRepository($io, $rm, $vendorDir, $package, $process); // initialize installation manager $im = $this->createInstallationManager($loop, $io, $dispatcher); @@ -458,9 +458,14 @@ class Factory * @param Repository\RepositoryManager $rm * @param string $vendorDir */ - protected function addLocalRepository(IOInterface $io, RepositoryManager $rm, $vendorDir, RootPackageInterface $rootPackage) + protected function addLocalRepository(IOInterface $io, RepositoryManager $rm, $vendorDir, RootPackageInterface $rootPackage, ProcessExecutor $process = null) { - $rm->setLocalRepository(new Repository\InstalledFilesystemRepository(new JsonFile($vendorDir.'/composer/installed.json', null, $io), true, $rootPackage)); + $fs = null; + if ($process) { + $fs = new Filesystem($process); + } + + $rm->setLocalRepository(new Repository\InstalledFilesystemRepository(new JsonFile($vendorDir.'/composer/installed.json', null, $io), true, $rootPackage, $fs)); } /** diff --git a/src/Composer/Repository/FilesystemRepository.php b/src/Composer/Repository/FilesystemRepository.php index 696b00541..976eb957d 100644 --- a/src/Composer/Repository/FilesystemRepository.php +++ b/src/Composer/Repository/FilesystemRepository.php @@ -31,6 +31,7 @@ class FilesystemRepository extends WritableArrayRepository protected $file; private $dumpVersions; private $rootPackage; + private $filesystem; /** * Initializes filesystem repository. @@ -39,12 +40,13 @@ class FilesystemRepository extends WritableArrayRepository * @param bool $dumpVersions * @param ?RootPackageInterface $rootPackage Must be provided if $dumpVersions is true */ - public function __construct(JsonFile $repositoryFile, $dumpVersions = false, RootPackageInterface $rootPackage = null) + public function __construct(JsonFile $repositoryFile, $dumpVersions = false, RootPackageInterface $rootPackage = null, Filesystem $filesystem = null) { parent::__construct(); $this->file = $repositoryFile; $this->dumpVersions = $dumpVersions; $this->rootPackage = $rootPackage; + $this->filesystem = $filesystem ?: new Filesystem; if ($dumpVersions && !$rootPackage) { throw new \InvalidArgumentException('Expected a root package instance if $dumpVersions is true'); } @@ -100,14 +102,24 @@ class FilesystemRepository extends WritableArrayRepository { $data = array('packages' => array(), 'dev' => $devMode, 'dev-package-names' => array()); $dumper = new ArrayDumper(); - $fs = new Filesystem(); - $repoDir = dirname($fs->normalizePath($this->file->getPath())); + + // make sure the directory is created so we can realpath it + // as realpath() does some additional normalizations with network paths that normalizePath does not + // and we need to find shortest path correctly + $repoDir = dirname($this->file->getPath()); + $this->filesystem->ensureDirectoryExists($repoDir); + + $repoDir = $this->filesystem->normalizePath(realpath($repoDir)); $installPaths = array(); foreach ($this->getCanonicalPackages() as $package) { $pkgArray = $dumper->dump($package); $path = $installationManager->getInstallPath($package); - $installPath = ('' !== $path && null !== $path) ? $fs->findShortestPath($repoDir, $fs->isAbsolutePath($path) ? $path : getcwd() . '/' . $path, true) : null; + $installPath = null; + if ('' !== $path && null !== $path) { + $normalizedPath = $this->filesystem->normalizePath($this->filesystem->isAbsolutePath($path) ? $path : getcwd() . '/' . $path); + $installPath = $this->filesystem->findShortestPath($repoDir, $normalizedPath, true); + } $installPaths[$package->getName()] = $installPath; $pkgArray['install-path'] = $installPath; @@ -130,9 +142,9 @@ class FilesystemRepository extends WritableArrayRepository if ($this->dumpVersions) { $versions = $this->generateInstalledVersions($installationManager, $installPaths, $devMode, $repoDir); - $fs->filePutContentsIfModified($repoDir.'/installed.php', 'dumpToPhpCode($versions) . ';'."\n"); + $this->filesystem->filePutContentsIfModified($repoDir.'/installed.php', 'dumpToPhpCode($versions) . ';'."\n"); $installedVersionsClass = file_get_contents(__DIR__.'/../InstalledVersions.php'); - $fs->filePutContentsIfModified($repoDir.'/InstalledVersions.php', $installedVersionsClass); + $this->filesystem->filePutContentsIfModified($repoDir.'/InstalledVersions.php', $installedVersionsClass); \Composer\InstalledVersions::reload($versions); } @@ -154,8 +166,7 @@ class FilesystemRepository extends WritableArrayRepository $lines .= "array(),\n"; } } elseif ($key === 'install_path' && is_string($value)) { - $fs = new Filesystem(); - if ($fs->isAbsolutePath($value)) { + if ($this->filesystem->isAbsolutePath($value)) { $lines .= var_export($value, true) . ",\n"; } else { $lines .= "__DIR__ . " . var_export('/' . $value, true) . ",\n"; @@ -203,9 +214,8 @@ class FilesystemRepository extends WritableArrayRepository } if ($package instanceof RootPackageInterface) { - $fs = new Filesystem(); - $to = getcwd(); - $installPath = $fs->findShortestPath($repoDir, $to, true); + $to = $this->filesystem->normalizePath(realpath(getcwd())); + $installPath = $this->filesystem->findShortestPath($repoDir, $to, true); } else { $installPath = $installPaths[$package->getName()]; } diff --git a/tests/Composer/Test/Mock/FactoryMock.php b/tests/Composer/Test/Mock/FactoryMock.php index d8b618647..c095bc639 100644 --- a/tests/Composer/Test/Mock/FactoryMock.php +++ b/tests/Composer/Test/Mock/FactoryMock.php @@ -46,7 +46,7 @@ class FactoryMock extends Factory return new \Composer\Package\Loader\RootPackageLoader($rm, $config, $parser, new VersionGuesserMock(), $io); } - protected function addLocalRepository(IOInterface $io, RepositoryManager $rm, $vendorDir, RootPackageInterface $rootPackage) + protected function addLocalRepository(IOInterface $io, RepositoryManager $rm, $vendorDir, RootPackageInterface $rootPackage, ProcessExecutor $process = null) { } diff --git a/tests/Composer/Test/Repository/FilesystemRepositoryTest.php b/tests/Composer/Test/Repository/FilesystemRepositoryTest.php index da754661b..0ea13fce6 100644 --- a/tests/Composer/Test/Repository/FilesystemRepositoryTest.php +++ b/tests/Composer/Test/Repository/FilesystemRepositoryTest.php @@ -16,6 +16,7 @@ use Composer\Package\RootPackageInterface; use Composer\Repository\FilesystemRepository; use Composer\Test\TestCase; use Composer\Json\JsonFile; +use Composer\Util\Filesystem; class FilesystemRepositoryTest extends TestCase { @@ -83,13 +84,17 @@ class FilesystemRepositoryTest extends TestCase { $json = $this->createJsonFileMock(); + $repoDir = realpath(sys_get_temp_dir()).'/repo_write_test/'; + $fs = new Filesystem(); + $fs->removeDirectory($repoDir); + $repository = new FilesystemRepository($json); $im = $this->getMockBuilder('Composer\Installer\InstallationManager') ->disableOriginalConstructor() ->getMock(); $im->expects($this->exactly(2)) ->method('getInstallPath') - ->will($this->returnValue('/foo/bar/vendor/woop/woop')); + ->will($this->returnValue($repoDir.'/vendor/woop/woop')); $json ->expects($this->once()) @@ -98,7 +103,7 @@ class FilesystemRepositoryTest extends TestCase $json ->expects($this->once()) ->method('getPath') - ->will($this->returnValue('/foo/bar/vendor/composer/installed.json')); + ->will($this->returnValue($repoDir.'/vendor/composer/installed.json')); $json ->expects($this->once()) ->method('exists')