Ensure packages that exist in a higher prio repo never get loaded in lower prio repos, fixes #5076
parent
4a7d42604f
commit
47a94b3a88
|
@ -97,9 +97,14 @@ class PoolBuilder
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO should we really pass the callable into here?
|
// TODO should we really pass the callable into here?
|
||||||
$packages = $repository->loadPackages($loadNames, $this->isPackageAcceptableCallable);
|
$result = $repository->loadPackages($loadNames, $this->isPackageAcceptableCallable);
|
||||||
|
|
||||||
|
foreach ($result['namesFound'] as $name) {
|
||||||
|
// avoid loading the same package again from other repositories once it has been found
|
||||||
|
unset($loadNames[$name]);
|
||||||
|
}
|
||||||
|
foreach ($result['packages'] as $package) {
|
||||||
|
|
||||||
foreach ($packages as $package) {
|
|
||||||
if (call_user_func($this->isPackageAcceptableCallable, $package->getNames(), $package->getStability())) {
|
if (call_user_func($this->isPackageAcceptableCallable, $package->getNames(), $package->getStability())) {
|
||||||
$newLoadNames += $this->loadPackage($request, $package, $key);
|
$newLoadNames += $this->loadPackage($request, $package, $key);
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,10 +31,11 @@ abstract class BaseRepository implements RepositoryInterface
|
||||||
$packages = $this->getPackages();
|
$packages = $this->getPackages();
|
||||||
|
|
||||||
$result = array();
|
$result = array();
|
||||||
|
$namesFound = array();
|
||||||
foreach ($packages as $package) {
|
foreach ($packages as $package) {
|
||||||
|
if (array_key_exists($package->getName(), $packageMap)) {
|
||||||
if (
|
if (
|
||||||
array_key_exists($package->getName(), $packageMap)
|
(!$packageMap[$package->getName()] || $packageMap[$package->getName()]->matches(new Constraint('==', $package->getVersion())))
|
||||||
&& (!$packageMap[$package->getName()] || $packageMap[$package->getName()]->matches(new Constraint('==', $package->getVersion())))
|
|
||||||
&& call_user_func($isPackageAcceptableCallable, $package->getNames(), $package->getStability())
|
&& call_user_func($isPackageAcceptableCallable, $package->getNames(), $package->getStability())
|
||||||
) {
|
) {
|
||||||
$result[spl_object_hash($package)] = $package;
|
$result[spl_object_hash($package)] = $package;
|
||||||
|
@ -42,6 +43,9 @@ abstract class BaseRepository implements RepositoryInterface
|
||||||
$result[spl_object_hash($package->getAliasOf())] = $package->getAliasOf();
|
$result[spl_object_hash($package->getAliasOf())] = $package->getAliasOf();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$namesFound[$package->getName()] = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
foreach ($packages as $package) {
|
foreach ($packages as $package) {
|
||||||
|
@ -52,7 +56,7 @@ abstract class BaseRepository implements RepositoryInterface
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $result;
|
return array('namesFound' => array_keys($namesFound), 'packages' => $result);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -143,7 +143,7 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
|
|
||||||
$packages = $this->loadAsyncPackages(array($name => $constraint));
|
$packages = $this->loadAsyncPackages(array($name => $constraint));
|
||||||
|
|
||||||
return reset($packages);
|
return reset($packages['packages']);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($hasProviders) {
|
if ($hasProviders) {
|
||||||
|
@ -181,7 +181,9 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
return array();
|
return array();
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->loadAsyncPackages(array($name => $constraint));
|
$result = $this->loadAsyncPackages(array($name => $constraint));
|
||||||
|
|
||||||
|
return $result['packages'];
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($hasProviders) {
|
if ($hasProviders) {
|
||||||
|
@ -239,7 +241,9 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
$packageMap[$name] = new EmptyConstraint();
|
$packageMap[$name] = new EmptyConstraint();
|
||||||
}
|
}
|
||||||
|
|
||||||
return array_values($this->loadAsyncPackages($packageMap));
|
$result = $this->loadAsyncPackages($packageMap);
|
||||||
|
|
||||||
|
return array_values($result['packages']);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->hasPartialPackages()) {
|
if ($this->hasPartialPackages()) {
|
||||||
|
@ -297,6 +301,7 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
}
|
}
|
||||||
|
|
||||||
$packages = array();
|
$packages = array();
|
||||||
|
$namesFound = array();
|
||||||
|
|
||||||
if ($hasProviders || $this->hasPartialPackages()) {
|
if ($hasProviders || $this->hasPartialPackages()) {
|
||||||
foreach ($packageNameMap as $name => $constraint) {
|
foreach ($packageNameMap as $name => $constraint) {
|
||||||
|
@ -313,6 +318,7 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
if ($candidate->getName() !== $name) {
|
if ($candidate->getName() !== $name) {
|
||||||
throw new \LogicException('whatProvides should never return a package with a different name than the requested one');
|
throw new \LogicException('whatProvides should never return a package with a different name than the requested one');
|
||||||
}
|
}
|
||||||
|
$namesFound[$name] = true;
|
||||||
if (!$constraint || $constraint->matches(new Constraint('==', $candidate->getVersion()))) {
|
if (!$constraint || $constraint->matches(new Constraint('==', $candidate->getVersion()))) {
|
||||||
$matches[spl_object_hash($candidate)] = $candidate;
|
$matches[spl_object_hash($candidate)] = $candidate;
|
||||||
if ($candidate instanceof AliasPackage && !isset($matches[spl_object_hash($candidate->getAliasOf())])) {
|
if ($candidate instanceof AliasPackage && !isset($matches[spl_object_hash($candidate->getAliasOf())])) {
|
||||||
|
@ -343,10 +349,12 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
}, ARRAY_FILTER_USE_KEY);
|
}, ARRAY_FILTER_USE_KEY);
|
||||||
}
|
}
|
||||||
|
|
||||||
$packages = array_merge($packages, $this->loadAsyncPackages($packageNameMap, $isPackageAcceptableCallable));
|
$result = $this->loadAsyncPackages($packageNameMap, $isPackageAcceptableCallable);
|
||||||
|
$packages = array_merge($packages, $result['packages']);
|
||||||
|
$namesFound = array_merge($namesFound, $result['namesFound']);
|
||||||
}
|
}
|
||||||
|
|
||||||
return $packages;
|
return array('namesFound' => array_keys($namesFound), 'packages' => $packages);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -586,6 +594,7 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
$this->loadRootServerFile();
|
$this->loadRootServerFile();
|
||||||
|
|
||||||
$packages = array();
|
$packages = array();
|
||||||
|
$namesFound = array();
|
||||||
$promises = array();
|
$promises = array();
|
||||||
$repo = $this;
|
$repo = $this;
|
||||||
|
|
||||||
|
@ -619,7 +628,7 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
}
|
}
|
||||||
|
|
||||||
$promises[] = $this->asyncFetchFile($url, $cacheKey, $lastModified)
|
$promises[] = $this->asyncFetchFile($url, $cacheKey, $lastModified)
|
||||||
->then(function ($response) use (&$packages, $contents, $realName, $constraint, $repo, $isPackageAcceptableCallable) {
|
->then(function ($response) use (&$packages, &$namesFound, $contents, $realName, $constraint, $repo, $isPackageAcceptableCallable) {
|
||||||
if (true === $response) {
|
if (true === $response) {
|
||||||
$response = $contents;
|
$response = $contents;
|
||||||
}
|
}
|
||||||
|
@ -657,6 +666,7 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
unset($expanded, $expandedVersion, $versionData);
|
unset($expanded, $expandedVersion, $versionData);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$namesFound[$realName] = true;
|
||||||
$versionsToLoad = array();
|
$versionsToLoad = array();
|
||||||
foreach ($versions as $version) {
|
foreach ($versions as $version) {
|
||||||
if (!isset($version['version_normalized'])) {
|
if (!isset($version['version_normalized'])) {
|
||||||
|
@ -683,7 +693,7 @@ class ComposerRepository extends ArrayRepository implements ConfigurableReposito
|
||||||
|
|
||||||
$this->loop->wait($promises);
|
$this->loop->wait($promises);
|
||||||
|
|
||||||
return $packages;
|
return array('namesFound' => $namesFound, 'packages' => $packages);
|
||||||
// RepositorySet should call loadMetadata, getMetadata when all promises resolved, then metadataComplete when done so we can GC the loaded json and whatnot then as needed
|
// RepositorySet should call loadMetadata, getMetadata when all promises resolved, then metadataComplete when done so we can GC the loaded json and whatnot then as needed
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -69,7 +69,7 @@ interface RepositoryInterface extends \Countable
|
||||||
*
|
*
|
||||||
* @param ConstraintInterface[] $packageNameMap package names pointing to constraints
|
* @param ConstraintInterface[] $packageNameMap package names pointing to constraints
|
||||||
* @param $isPackageAcceptableCallable
|
* @param $isPackageAcceptableCallable
|
||||||
* @return PackageInterface[]
|
* @return array [namesFound => string[], packages => PackageInterface[]]
|
||||||
*/
|
*/
|
||||||
public function loadPackages(array $packageNameMap, $isPackageAcceptableCallable);
|
public function loadPackages(array $packageNameMap, $isPackageAcceptableCallable);
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,42 @@
|
||||||
|
--TEST--
|
||||||
|
Packages found in a higher priority repository take precedence even if they are not found in the requested version
|
||||||
|
--COMPOSER--
|
||||||
|
{
|
||||||
|
"repositories": [
|
||||||
|
{
|
||||||
|
"type": "package",
|
||||||
|
"package": [
|
||||||
|
{ "name": "foo/a", "version": "1.0.0" }
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "package",
|
||||||
|
"package": [
|
||||||
|
{ "name": "foo/a", "version": "2.0.0" }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"require": {
|
||||||
|
"foo/a": "2.*"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
--RUN--
|
||||||
|
update
|
||||||
|
--EXPECT-OUTPUT--
|
||||||
|
Loading composer repositories with package information
|
||||||
|
Updating dependencies
|
||||||
|
Your requirements could not be resolved to an installable set of packages.
|
||||||
|
|
||||||
|
Problem 1
|
||||||
|
- The requested package foo/a could not be found in any version, there may be a typo in the package name.
|
||||||
|
|
||||||
|
Potential causes:
|
||||||
|
- A typo in the package name
|
||||||
|
- The package is not available in a stable-enough version according to your minimum-stability setting
|
||||||
|
see <https://getcomposer.org/doc/04-schema.md#minimum-stability> for more details.
|
||||||
|
- It's a private package and you forgot to add a custom repository to find it
|
||||||
|
|
||||||
|
Read <https://getcomposer.org/doc/articles/troubleshooting.md> for further common problems.
|
||||||
|
--EXPECT--
|
||||||
|
--EXPECT-EXIT-CODE--
|
||||||
|
2
|
|
@ -0,0 +1,26 @@
|
||||||
|
--TEST--
|
||||||
|
Packages found in a higher priority repository take precedence
|
||||||
|
--COMPOSER--
|
||||||
|
{
|
||||||
|
"repositories": [
|
||||||
|
{
|
||||||
|
"type": "package",
|
||||||
|
"package": [
|
||||||
|
{ "name": "foo/a", "version": "1.0.0" }
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "package",
|
||||||
|
"package": [
|
||||||
|
{ "name": "foo/a", "version": "1.1.0" }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"require": {
|
||||||
|
"foo/a": "1.*"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
--RUN--
|
||||||
|
update
|
||||||
|
--EXPECT--
|
||||||
|
Installing foo/a (1.0.0)
|
Loading…
Reference in New Issue