1
0
Fork 0

Fix support for writing into UNC paths, and comparing UNC paths correctly in InstalledVersions, fixes #9993

pull/9995/head
Jordi Boggiano 2021-07-12 13:54:01 +02:00
parent cc81f5bac3
commit 005c55185a
No known key found for this signature in database
GPG Key ID: 7BBD42C429EC80BC
4 changed files with 37 additions and 17 deletions

View File

@ -385,7 +385,7 @@ class Factory
$composer->setPackage($package); $composer->setPackage($package);
// load local repository // load local repository
$this->addLocalRepository($io, $rm, $vendorDir, $package); $this->addLocalRepository($io, $rm, $vendorDir, $package, $process);
// initialize installation manager // initialize installation manager
$im = $this->createInstallationManager($loop, $io, $dispatcher); $im = $this->createInstallationManager($loop, $io, $dispatcher);
@ -458,9 +458,14 @@ class Factory
* @param Repository\RepositoryManager $rm * @param Repository\RepositoryManager $rm
* @param string $vendorDir * @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));
} }
/** /**

View File

@ -31,6 +31,7 @@ class FilesystemRepository extends WritableArrayRepository
protected $file; protected $file;
private $dumpVersions; private $dumpVersions;
private $rootPackage; private $rootPackage;
private $filesystem;
/** /**
* Initializes filesystem repository. * Initializes filesystem repository.
@ -39,12 +40,13 @@ class FilesystemRepository extends WritableArrayRepository
* @param bool $dumpVersions * @param bool $dumpVersions
* @param ?RootPackageInterface $rootPackage Must be provided if $dumpVersions is true * @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(); parent::__construct();
$this->file = $repositoryFile; $this->file = $repositoryFile;
$this->dumpVersions = $dumpVersions; $this->dumpVersions = $dumpVersions;
$this->rootPackage = $rootPackage; $this->rootPackage = $rootPackage;
$this->filesystem = $filesystem ?: new Filesystem;
if ($dumpVersions && !$rootPackage) { if ($dumpVersions && !$rootPackage) {
throw new \InvalidArgumentException('Expected a root package instance if $dumpVersions is true'); 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()); $data = array('packages' => array(), 'dev' => $devMode, 'dev-package-names' => array());
$dumper = new ArrayDumper(); $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(); $installPaths = array();
foreach ($this->getCanonicalPackages() as $package) { foreach ($this->getCanonicalPackages() as $package) {
$pkgArray = $dumper->dump($package); $pkgArray = $dumper->dump($package);
$path = $installationManager->getInstallPath($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; $installPaths[$package->getName()] = $installPath;
$pkgArray['install-path'] = $installPath; $pkgArray['install-path'] = $installPath;
@ -130,9 +142,9 @@ class FilesystemRepository extends WritableArrayRepository
if ($this->dumpVersions) { if ($this->dumpVersions) {
$versions = $this->generateInstalledVersions($installationManager, $installPaths, $devMode, $repoDir); $versions = $this->generateInstalledVersions($installationManager, $installPaths, $devMode, $repoDir);
$fs->filePutContentsIfModified($repoDir.'/installed.php', '<?php return ' . $this->dumpToPhpCode($versions) . ';'."\n"); $this->filesystem->filePutContentsIfModified($repoDir.'/installed.php', '<?php return ' . $this->dumpToPhpCode($versions) . ';'."\n");
$installedVersionsClass = file_get_contents(__DIR__.'/../InstalledVersions.php'); $installedVersionsClass = file_get_contents(__DIR__.'/../InstalledVersions.php');
$fs->filePutContentsIfModified($repoDir.'/InstalledVersions.php', $installedVersionsClass); $this->filesystem->filePutContentsIfModified($repoDir.'/InstalledVersions.php', $installedVersionsClass);
\Composer\InstalledVersions::reload($versions); \Composer\InstalledVersions::reload($versions);
} }
@ -154,8 +166,7 @@ class FilesystemRepository extends WritableArrayRepository
$lines .= "array(),\n"; $lines .= "array(),\n";
} }
} elseif ($key === 'install_path' && is_string($value)) { } elseif ($key === 'install_path' && is_string($value)) {
$fs = new Filesystem(); if ($this->filesystem->isAbsolutePath($value)) {
if ($fs->isAbsolutePath($value)) {
$lines .= var_export($value, true) . ",\n"; $lines .= var_export($value, true) . ",\n";
} else { } else {
$lines .= "__DIR__ . " . var_export('/' . $value, true) . ",\n"; $lines .= "__DIR__ . " . var_export('/' . $value, true) . ",\n";
@ -203,9 +214,8 @@ class FilesystemRepository extends WritableArrayRepository
} }
if ($package instanceof RootPackageInterface) { if ($package instanceof RootPackageInterface) {
$fs = new Filesystem(); $to = $this->filesystem->normalizePath(realpath(getcwd()));
$to = getcwd(); $installPath = $this->filesystem->findShortestPath($repoDir, $to, true);
$installPath = $fs->findShortestPath($repoDir, $to, true);
} else { } else {
$installPath = $installPaths[$package->getName()]; $installPath = $installPaths[$package->getName()];
} }

View File

@ -46,7 +46,7 @@ class FactoryMock extends Factory
return new \Composer\Package\Loader\RootPackageLoader($rm, $config, $parser, new VersionGuesserMock(), $io); 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)
{ {
} }

View File

@ -16,6 +16,7 @@ use Composer\Package\RootPackageInterface;
use Composer\Repository\FilesystemRepository; use Composer\Repository\FilesystemRepository;
use Composer\Test\TestCase; use Composer\Test\TestCase;
use Composer\Json\JsonFile; use Composer\Json\JsonFile;
use Composer\Util\Filesystem;
class FilesystemRepositoryTest extends TestCase class FilesystemRepositoryTest extends TestCase
{ {
@ -83,13 +84,17 @@ class FilesystemRepositoryTest extends TestCase
{ {
$json = $this->createJsonFileMock(); $json = $this->createJsonFileMock();
$repoDir = realpath(sys_get_temp_dir()).'/repo_write_test/';
$fs = new Filesystem();
$fs->removeDirectory($repoDir);
$repository = new FilesystemRepository($json); $repository = new FilesystemRepository($json);
$im = $this->getMockBuilder('Composer\Installer\InstallationManager') $im = $this->getMockBuilder('Composer\Installer\InstallationManager')
->disableOriginalConstructor() ->disableOriginalConstructor()
->getMock(); ->getMock();
$im->expects($this->exactly(2)) $im->expects($this->exactly(2))
->method('getInstallPath') ->method('getInstallPath')
->will($this->returnValue('/foo/bar/vendor/woop/woop')); ->will($this->returnValue($repoDir.'/vendor/woop/woop'));
$json $json
->expects($this->once()) ->expects($this->once())
@ -98,7 +103,7 @@ class FilesystemRepositoryTest extends TestCase
$json $json
->expects($this->once()) ->expects($this->once())
->method('getPath') ->method('getPath')
->will($this->returnValue('/foo/bar/vendor/composer/installed.json')); ->will($this->returnValue($repoDir.'/vendor/composer/installed.json'));
$json $json
->expects($this->once()) ->expects($this->once())
->method('exists') ->method('exists')