1
0
Fork 0

Make sure the root aliases always get installed when a package is updated, fixes #9448

pull/9473/head
Jordi Boggiano 2020-11-12 17:05:50 +01:00
parent 8936f724d4
commit 7b183956d8
No known key found for this signature in database
GPG Key ID: 7BBD42C429EC80BC
8 changed files with 207 additions and 6 deletions

View File

@ -133,6 +133,10 @@ class LockTransaction extends Transaction
} }
} }
usort($usedAliases, function ($a, $b) {
return strcmp($a['package'], $b['package']);
});
return $usedAliases; return $usedAliases;
} }
} }

View File

@ -55,6 +55,11 @@ class Pool implements \Countable
} }
} }
public function getPackages()
{
return $this->packages;
}
/** /**
* Retrieves the package object for a given package id. * Retrieves the package object for a given package id.
* *

View File

@ -231,7 +231,7 @@ class RuleSetGenerator
} }
// otherwise, looks like a bug // otherwise, looks like a bug
throw new \LogicException("Fixed package ".$package->getName()." ".$package->getVersion().($package instanceof AliasPackage ? " (alias)" : "")." was not added to solver pool."); throw new \LogicException("Fixed package ".$package->getPrettyString()." was not added to solver pool.");
} }
$this->addRulesForPackage($package, $ignorePlatformReqs); $this->addRulesForPackage($package, $ignorePlatformReqs);
@ -262,6 +262,17 @@ class RuleSetGenerator
} }
} }
protected function addRulesForRootAliases()
{
foreach ($this->pool->getPackages() as $package) {
// if it is a root alias, make sure that if the aliased version gets installed
// the alias must be installed too
if ($package instanceof AliasPackage && $package->isRootPackageAlias()) {
$this->addRule(RuleSet::TYPE_PACKAGE, $this->createRequireRule($package->getAliasOf(), array($package), Rule::RULE_PACKAGE_REQUIRES, $package->getAliasOf()));
}
}
}
/** /**
* @param bool|array $ignorePlatformReqs * @param bool|array $ignorePlatformReqs
*/ */
@ -277,6 +288,8 @@ class RuleSetGenerator
$this->addRulesForRequest($request, $ignorePlatformReqs); $this->addRulesForRequest($request, $ignorePlatformReqs);
$this->addRulesForRootAliases();
$this->addConflictRules($ignorePlatformReqs); $this->addConflictRules($ignorePlatformReqs);
// Remove references to packages // Remove references to packages

View File

@ -789,7 +789,7 @@ class SolverTest extends TestCase
$this->checkSolverResult(array( $this->checkSolverResult(array(
array('job' => 'install', 'package' => $packageB), array('job' => 'install', 'package' => $packageB),
array('job' => 'install', 'package' => $packageA2), array('job' => 'install', 'package' => $packageA2),
array('job' => 'install', 'package' => $packageA2Alias), array('job' => 'markAliasInstalled', 'package' => $packageA2Alias),
)); ));
} }
@ -811,11 +811,40 @@ class SolverTest extends TestCase
$this->checkSolverResult(array( $this->checkSolverResult(array(
array('job' => 'install', 'package' => $packageA), array('job' => 'install', 'package' => $packageA),
array('job' => 'install', 'package' => $packageAAlias), array('job' => 'markAliasInstalled', 'package' => $packageAAlias),
array('job' => 'install', 'package' => $packageB), array('job' => 'install', 'package' => $packageB),
)); ));
} }
public function testInstallRootAliasesIfAliasOfIsInstalled()
{
// root aliased, required
$this->repo->addPackage($packageA = $this->getPackage('A', '1.0'));
$this->repo->addPackage($packageAAlias = $this->getAliasPackage($packageA, '1.1'));
$packageAAlias->setRootPackageAlias(true);
// root aliased, not required, should still be installed as it is root alias
$this->repo->addPackage($packageB = $this->getPackage('B', '1.0'));
$this->repo->addPackage($packageBAlias = $this->getAliasPackage($packageB, '1.1'));
$packageBAlias->setRootPackageAlias(true);
// regular alias, not required, alias should not be installed
$this->repo->addPackage($packageC = $this->getPackage('C', '1.0'));
$this->repo->addPackage($packageCAlias = $this->getAliasPackage($packageC, '1.1'));
$this->reposComplete();
$this->request->requireName('A', $this->getVersionConstraint('==', '1.1'));
$this->request->requireName('B', $this->getVersionConstraint('==', '1.0'));
$this->request->requireName('C', $this->getVersionConstraint('==', '1.0'));
$this->checkSolverResult(array(
array('job' => 'install', 'package' => $packageA),
array('job' => 'markAliasInstalled', 'package' => $packageAAlias),
array('job' => 'install', 'package' => $packageB),
array('job' => 'markAliasInstalled', 'package' => $packageBAlias),
array('job' => 'install', 'package' => $packageC),
));
}
/** /**
* Tests for a bug introduced in commit 451bab1c2cd58e05af6e21639b829408ad023463 Solver.php line 554/523 * Tests for a bug introduced in commit 451bab1c2cd58e05af6e21639b829408ad023463 Solver.php line 554/523
* *
@ -915,6 +944,11 @@ class SolverTest extends TestCase
'from' => $operation->getInitialPackage(), 'from' => $operation->getInitialPackage(),
'to' => $operation->getTargetPackage(), 'to' => $operation->getTargetPackage(),
); );
} elseif (in_array($operation->getOperationType(), array('markAliasInstalled', 'markAliasUninstalled'))) {
$result[] = array(
'job' => $operation->getOperationType(),
'package' => $operation->getPackage(),
);
} else { } else {
$job = ('uninstall' === $operation->getOperationType() ? 'remove' : 'install'); $job = ('uninstall' === $operation->getOperationType() ? 'remove' : 'install');
$result[] = array( $result[] = array(
@ -924,6 +958,16 @@ class SolverTest extends TestCase
} }
} }
$expectedReadable = array();
foreach ($expected as $op) {
$expectedReadable[] = array_map('strval', $op);
}
$resultReadable = array();
foreach ($result as $op) {
$resultReadable[] = array_map('strval', $op);
}
$this->assertEquals($expectedReadable, $resultReadable);
$this->assertEquals($expected, $result); $this->assertEquals($expected, $result);
} }
} }

View File

@ -1,5 +1,5 @@
--TEST-- --TEST--
Root-defined aliases end up in lock file only if required to solve deps Root-defined aliases end up in lock file always on full update
--COMPOSER-- --COMPOSER--
{ {
"repositories": [ "repositories": [
@ -50,6 +50,11 @@ update
"version": "3.0.2.0", "version": "3.0.2.0",
"alias": "3.0.3", "alias": "3.0.3",
"alias_normalized": "3.0.3.0" "alias_normalized": "3.0.3.0"
},{
"package": "a/aliased2",
"version": "3.0.2.0",
"alias": "3.0.3",
"alias_normalized": "3.0.3.0"
}], }],
"minimum-stability": "stable", "minimum-stability": "stable",
"stability-flags": [], "stability-flags": [],
@ -60,6 +65,7 @@ update
} }
--EXPECT-- --EXPECT--
Installing a/aliased2 (3.0.2) Installing a/aliased2 (3.0.2)
Marking a/aliased2 (3.0.3) as installed, alias of a/aliased2 (3.0.2)
Installing a/aliased (3.0.2) Installing a/aliased (3.0.2)
Marking a/aliased (3.0.3) as installed, alias of a/aliased (3.0.2) Marking a/aliased (3.0.3) as installed, alias of a/aliased (3.0.2)
Installing b/requirer (1.0.0) Installing b/requirer (1.0.0)

View File

@ -0,0 +1,75 @@
--TEST--
Newly defined root aliases end up in lock file only if the package is updated
--COMPOSER--
{
"repositories": [
{
"type": "package",
"package": [
{
"name": "a/aliased", "version": "3.0.2"
},
{
"name": "a/aliased2", "version": "3.0.2"
}
]
}
],
"require": {
"a/aliased": "3.0.2 as 3.0.3",
"a/aliased2": "3.0.2 as 3.0.3"
}
}
--LOCK--
{
"packages": [
{
"name": "a/aliased", "version": "3.0.2",
"type": "library"
},
{
"name": "a/aliased2", "version": "3.0.2",
"type": "library"
}
],
"packages-dev": [],
"aliases": [],
"minimum-stability": "stable",
"stability-flags": [],
"prefer-stable": false,
"prefer-lowest": false,
"platform": [],
"platform-dev": []
}
--RUN--
update a/aliased
--EXPECT-LOCK--
{
"packages": [
{
"name": "a/aliased", "version": "3.0.2",
"type": "library"
},
{
"name": "a/aliased2", "version": "3.0.2",
"type": "library"
}
],
"packages-dev": [],
"aliases": [{
"package": "a/aliased",
"version": "3.0.2.0",
"alias": "3.0.3",
"alias_normalized": "3.0.3.0"
}],
"minimum-stability": "stable",
"stability-flags": [],
"prefer-stable": false,
"prefer-lowest": false,
"platform": [],
"platform-dev": []
}
--EXPECT--
Installing a/aliased (3.0.2)
Marking a/aliased (3.0.3) as installed, alias of a/aliased (3.0.2)
Installing a/aliased2 (3.0.2)

View File

@ -61,12 +61,12 @@ update
} }
], ],
"aliases": [{ "aliases": [{
"package": "a/aliased2", "package": "a/aliased",
"version": "dev-next", "version": "dev-next",
"alias": "4.1.0-RC2", "alias": "4.1.0-RC2",
"alias_normalized": "4.1.0.0-RC2" "alias_normalized": "4.1.0.0-RC2"
}, { }, {
"package": "a/aliased", "package": "a/aliased2",
"version": "dev-next", "version": "dev-next",
"alias": "4.1.0-RC2", "alias": "4.1.0-RC2",
"alias_normalized": "4.1.0.0-RC2" "alias_normalized": "4.1.0.0-RC2"

View File

@ -0,0 +1,54 @@
--TEST--
Newly defined root alias does not get loaded if package is loaded from lock file
--COMPOSER--
{
"repositories": [
{
"type": "package",
"package": [
{ "name": "some/dep", "version": "dev-main" },
{ "name": "foo/pkg", "version": "1.0.0", "require": {"some/dep": "^1"} }
]
}
],
"require": {
"some/dep": "dev-main as 1.0.0",
"foo/pkg": "^1.0"
}
}
--LOCK--
{
"packages": [
{ "name": "some/dep", "version": "dev-main" }
],
"packages-dev": [],
"aliases": [],
"minimum-stability": "stable",
"stability-flags": [],
"prefer-stable": false,
"prefer-lowest": false,
"platform": [],
"platform-dev": []
}
--INSTALLED--
[
{ "name": "some/dep", "version": "dev-main" }
]
--RUN--
update foo/pkg
--EXPECT-EXIT-CODE--
2
--EXPECT-OUTPUT--
Loading composer repositories with package information
Updating dependencies
Your requirements could not be resolved to an installable set of packages.
Problem 1
- Root composer.json requires foo/pkg ^1.0 -> satisfiable by foo/pkg[1.0.0].
- foo/pkg 1.0.0 requires some/dep ^1 -> found some/dep[dev-main] but it does not match the constraint.
Use the option --with-all-dependencies (-W) to allow upgrades, downgrades and removals for packages currently locked to specific versions.
--EXPECT--