Merge pull request #9995 from Seldaek/fix_wsl
Fix UNC/WSL-path issues when running in Windowspull/9996/head
commit
5152eeebdc
|
@ -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));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -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()];
|
||||||
}
|
}
|
||||||
|
|
|
@ -544,7 +544,13 @@ class Filesystem
|
||||||
$parts = array();
|
$parts = array();
|
||||||
$path = strtr($path, '\\', '/');
|
$path = strtr($path, '\\', '/');
|
||||||
$prefix = '';
|
$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:
|
// 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)) {
|
if (preg_match('{^( [0-9a-z]{2,}+: (?: // (?: [a-z]: )? )? | [a-z]: )}ix', $path, $match)) {
|
||||||
|
@ -553,13 +559,13 @@ class Filesystem
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strpos($path, '/') === 0) {
|
if (strpos($path, '/') === 0) {
|
||||||
$absolute = true;
|
$absolute = '/';
|
||||||
$path = substr($path, 1);
|
$path = substr($path, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
$up = false;
|
$up = false;
|
||||||
foreach (explode('/', $path) as $chunk) {
|
foreach (explode('/', $path) as $chunk) {
|
||||||
if ('..' === $chunk && ($absolute || $up)) {
|
if ('..' === $chunk && ($absolute !== '' || $up)) {
|
||||||
array_pop($parts);
|
array_pop($parts);
|
||||||
$up = !(empty($parts) || '..' === end($parts));
|
$up = !(empty($parts) || '..' === end($parts));
|
||||||
} elseif ('.' !== $chunk && '' !== $chunk) {
|
} elseif ('.' !== $chunk && '' !== $chunk) {
|
||||||
|
@ -568,7 +574,7 @@ class Filesystem
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $prefix.($absolute ? '/' : '').implode('/', $parts);
|
return $prefix.((string) $absolute).implode('/', $parts);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -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)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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')
|
||||||
|
|
|
@ -220,6 +220,7 @@ class FilesystemTest extends TestCase
|
||||||
array('../src', 'Foo/Bar/../../../src'),
|
array('../src', 'Foo/Bar/../../../src'),
|
||||||
array('c:../b', 'c:.\\..\\a\\..\\b'),
|
array('c:../b', 'c:.\\..\\a\\..\\b'),
|
||||||
array('phar://c:../Foo', 'phar://c:../Foo'),
|
array('phar://c:../Foo', 'phar://c:../Foo'),
|
||||||
|
array('//foo/bar', '\\\\foo\\bar'),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue