Merge pull request #9475 from naderman/fix-alias-alt
Make sure the root aliases always get installed when a package is updatedpull/9478/head
commit
d6794217d9
|
@ -133,6 +133,10 @@ class LockTransaction extends Transaction
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
usort($usedAliases, function ($a, $b) {
|
||||||
|
return strcmp($a['package'], $b['package']);
|
||||||
|
});
|
||||||
|
|
||||||
return $usedAliases;
|
return $usedAliases;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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.
|
||||||
*
|
*
|
||||||
|
|
|
@ -35,6 +35,7 @@ abstract class Rule
|
||||||
const RULE_PACKAGE_SAME_NAME = 10;
|
const RULE_PACKAGE_SAME_NAME = 10;
|
||||||
const RULE_LEARNED = 12;
|
const RULE_LEARNED = 12;
|
||||||
const RULE_PACKAGE_ALIAS = 13;
|
const RULE_PACKAGE_ALIAS = 13;
|
||||||
|
const RULE_PACKAGE_ROOT_ALIAS = 14;
|
||||||
|
|
||||||
// bitfield defs
|
// bitfield defs
|
||||||
const BITFIELD_TYPE = 0;
|
const BITFIELD_TYPE = 0;
|
||||||
|
@ -312,12 +313,20 @@ abstract class Rule
|
||||||
|
|
||||||
return 'Conclusion: '.$ruleText.$learnedString;
|
return 'Conclusion: '.$ruleText.$learnedString;
|
||||||
case self::RULE_PACKAGE_ALIAS:
|
case self::RULE_PACKAGE_ALIAS:
|
||||||
$aliasPackage = $pool->literalToPackage($literals[0]);
|
case self::RULE_PACKAGE_ROOT_ALIAS:
|
||||||
|
if ($this->getReason() === self::RULE_PACKAGE_ALIAS) {
|
||||||
|
$aliasPackage = $pool->literalToPackage($literals[0]);
|
||||||
|
$otherLiteral = 1;
|
||||||
|
} else {
|
||||||
|
// root alias rules work the other way around
|
||||||
|
$aliasPackage = $pool->literalToPackage($literals[1]);
|
||||||
|
$otherLiteral = 0;
|
||||||
|
}
|
||||||
// avoid returning content like "9999999-dev is an alias of dev-master" as it is useless
|
// avoid returning content like "9999999-dev is an alias of dev-master" as it is useless
|
||||||
if ($aliasPackage->getVersion() === VersionParser::DEFAULT_BRANCH_ALIAS) {
|
if ($aliasPackage->getVersion() === VersionParser::DEFAULT_BRANCH_ALIAS) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
$package = $this->deduplicateDefaultBranchAlias($pool->literalToPackage($literals[1]));
|
$package = $this->deduplicateDefaultBranchAlias($pool->literalToPackage($literals[$otherLiteral]));
|
||||||
|
|
||||||
return $aliasPackage->getPrettyString() .' is an alias of '.$package->getPrettyString().' and thus requires it to be installed too.';
|
return $aliasPackage->getPrettyString() .' is an alias of '.$package->getPrettyString().' and thus requires it to be installed too.';
|
||||||
default:
|
default:
|
||||||
|
|
|
@ -28,7 +28,6 @@ class RuleSetGenerator
|
||||||
protected $rules;
|
protected $rules;
|
||||||
protected $addedMap;
|
protected $addedMap;
|
||||||
protected $conflictAddedMap;
|
protected $conflictAddedMap;
|
||||||
protected $addedPackages;
|
|
||||||
protected $addedPackagesByNames;
|
protected $addedPackagesByNames;
|
||||||
protected $conflictsForName;
|
protected $conflictsForName;
|
||||||
|
|
||||||
|
@ -157,9 +156,8 @@ class RuleSetGenerator
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->addedMap[$package->id] = true;
|
$this->addedMap[$package->id] = $package;
|
||||||
|
|
||||||
$this->addedPackages[] = $package;
|
|
||||||
if (!$package instanceof AliasPackage) {
|
if (!$package instanceof AliasPackage) {
|
||||||
foreach ($package->getNames(false) as $name) {
|
foreach ($package->getNames(false) as $name) {
|
||||||
$this->addedPackagesByNames[$name][] = $package;
|
$this->addedPackagesByNames[$name][] = $package;
|
||||||
|
@ -168,6 +166,11 @@ class RuleSetGenerator
|
||||||
$workQueue->enqueue($package->getAliasOf());
|
$workQueue->enqueue($package->getAliasOf());
|
||||||
$this->addRule(RuleSet::TYPE_PACKAGE, $this->createRequireRule($package, array($package->getAliasOf()), Rule::RULE_PACKAGE_ALIAS, $package));
|
$this->addRule(RuleSet::TYPE_PACKAGE, $this->createRequireRule($package, array($package->getAliasOf()), Rule::RULE_PACKAGE_ALIAS, $package));
|
||||||
|
|
||||||
|
// root aliases must be installed with their main package, so create a rule the other way around as well
|
||||||
|
if ($package->isRootPackageAlias()) {
|
||||||
|
$this->addRule(RuleSet::TYPE_PACKAGE, $this->createRequireRule($package->getAliasOf(), array($package), Rule::RULE_PACKAGE_ROOT_ALIAS, $package->getAliasOf()));
|
||||||
|
}
|
||||||
|
|
||||||
// if alias package has no self.version requires, its requirements do not
|
// if alias package has no self.version requires, its requirements do not
|
||||||
// need to be added as the aliased package processing will take care of it
|
// need to be added as the aliased package processing will take care of it
|
||||||
if (!$package->hasSelfVersionRequires()) {
|
if (!$package->hasSelfVersionRequires()) {
|
||||||
|
@ -194,7 +197,7 @@ class RuleSetGenerator
|
||||||
protected function addConflictRules($ignorePlatformReqs = false)
|
protected function addConflictRules($ignorePlatformReqs = false)
|
||||||
{
|
{
|
||||||
/** @var PackageInterface $package */
|
/** @var PackageInterface $package */
|
||||||
foreach ($this->addedPackages as $package) {
|
foreach ($this->addedMap as $package) {
|
||||||
foreach ($package->getConflicts() as $link) {
|
foreach ($package->getConflicts() as $link) {
|
||||||
if (!isset($this->addedPackagesByNames[$link->getTarget()])) {
|
if (!isset($this->addedPackagesByNames[$link->getTarget()])) {
|
||||||
continue;
|
continue;
|
||||||
|
@ -231,7 +234,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 +265,17 @@ class RuleSetGenerator
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected function addRulesForRootAliases($ignorePlatformReqs)
|
||||||
|
{
|
||||||
|
foreach ($this->pool->getPackages() as $package) {
|
||||||
|
// ensure that rules for root alias packages get loaded even if the root alias itself isn't required
|
||||||
|
// otherwise a package could be installed without its root alias which leads to unexpected behavior
|
||||||
|
if ($package instanceof AliasPackage && $package->isRootPackageAlias()) {
|
||||||
|
$this->addRulesForPackage($package, $ignorePlatformReqs);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param bool|array $ignorePlatformReqs
|
* @param bool|array $ignorePlatformReqs
|
||||||
*/
|
*/
|
||||||
|
@ -271,16 +285,17 @@ class RuleSetGenerator
|
||||||
|
|
||||||
$this->addedMap = array();
|
$this->addedMap = array();
|
||||||
$this->conflictAddedMap = array();
|
$this->conflictAddedMap = array();
|
||||||
$this->addedPackages = array();
|
|
||||||
$this->addedPackagesByNames = array();
|
$this->addedPackagesByNames = array();
|
||||||
$this->conflictsForName = array();
|
$this->conflictsForName = array();
|
||||||
|
|
||||||
$this->addRulesForRequest($request, $ignorePlatformReqs);
|
$this->addRulesForRequest($request, $ignorePlatformReqs);
|
||||||
|
|
||||||
|
$this->addRulesForRootAliases($ignorePlatformReqs);
|
||||||
|
|
||||||
$this->addConflictRules($ignorePlatformReqs);
|
$this->addConflictRules($ignorePlatformReqs);
|
||||||
|
|
||||||
// Remove references to packages
|
// Remove references to packages
|
||||||
$this->addedPackages = $this->addedPackagesByNames = null;
|
$this->addedMap = $this->addedPackagesByNames = null;
|
||||||
|
|
||||||
return $this->rules;
|
return $this->rules;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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)
|
|
@ -0,0 +1,30 @@
|
||||||
|
--TEST--
|
||||||
|
A root alias for a package which cannot be found in an acceptable version does not lead to different error.
|
||||||
|
--COMPOSER--
|
||||||
|
{
|
||||||
|
"repositories": [
|
||||||
|
{
|
||||||
|
"type": "package",
|
||||||
|
"package": [
|
||||||
|
{ "name": "a/aliased", "version": "1.2.3" }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"require": {
|
||||||
|
"a/aliased": "3.0.2 as 3.0.3"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
--RUN--
|
||||||
|
update
|
||||||
|
--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 a/aliased 3.0.2 as 3.0.3, found a/aliased[1.2.3] but it does not match the constraint.
|
||||||
|
|
||||||
|
--EXPECT--
|
||||||
|
|
|
@ -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"
|
||||||
|
|
|
@ -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--
|
Loading…
Reference in New Issue