|
|
5e9bef |
commit 25a1cf0a8f0002b16d6ef9c24d530019fd88fd64
|
|
|
5e9bef |
Author: Michal Domonkos <mdomonko@redhat.com>
|
|
|
5e9bef |
Date: Wed May 16 17:40:22 2018 +0200
|
|
|
5e9bef |
|
|
|
5e9bef |
update(): honor multilib_policy for obsoletes. BZ 1477574
|
|
|
5e9bef |
|
|
|
5e9bef |
When a package migrates from noarch to arch during an update, we usually
|
|
|
5e9bef |
don't want to install all the newly available multilib arches.
|
|
|
5e9bef |
|
|
|
5e9bef |
For "yum install", this is already controlled by multilib_policy, even
|
|
|
5e9bef |
if the noarch package is already installed (since yum will try to update
|
|
|
5e9bef |
it). However, with "yum update", multilib_policy isn't checked.
|
|
|
5e9bef |
|
|
|
5e9bef |
This commit adds the check to update() so that "yum update" works the
|
|
|
5e9bef |
same as "yum install" in the noarch->arch scenario.
|
|
|
5e9bef |
|
|
|
5e9bef |
Tests calling "update" without arguments now need to specify
|
|
|
5e9bef |
multilib_policy='all'.
|
|
|
5e9bef |
|
|
|
5e9bef |
diff --git a/test/simpleobsoletestests.py b/test/simpleobsoletestests.py
|
|
|
5e9bef |
index 6cede1e0..a26a5d42 100644
|
|
|
5e9bef |
--- a/test/simpleobsoletestests.py
|
|
|
5e9bef |
+++ b/test/simpleobsoletestests.py
|
|
|
5e9bef |
@@ -38,7 +38,8 @@ class SimpleObsoletesTests(OperationsTests):
|
|
|
5e9bef |
|
|
|
5e9bef |
def testObsoletenoarchToi386(self):
|
|
|
5e9bef |
p = self.pkgs
|
|
|
5e9bef |
- res, msg = self.runOperation(['update'], [p.installed_noarch], [p.obsoletes_i386])
|
|
|
5e9bef |
+ res, msg = self.runOperation(['update'], [p.installed_noarch], [p.obsoletes_i386],
|
|
|
5e9bef |
+ {'multilib_policy': 'all'})
|
|
|
5e9bef |
self.assert_(res=='ok', msg)
|
|
|
5e9bef |
self.assertResult((p.obsoletes_i386,))
|
|
|
5e9bef |
def testObsoletenoarchToi386ForDependency(self):
|
|
|
5e9bef |
@@ -50,7 +51,8 @@ class SimpleObsoletesTests(OperationsTests):
|
|
|
5e9bef |
|
|
|
5e9bef |
def testObsoletenoarchTox86_64(self):
|
|
|
5e9bef |
p = self.pkgs
|
|
|
5e9bef |
- res, msg = self.runOperation(['update'], [p.installed_noarch], [p.obsoletes_x86_64])
|
|
|
5e9bef |
+ res, msg = self.runOperation(['update'], [p.installed_noarch], [p.obsoletes_x86_64],
|
|
|
5e9bef |
+ {'multilib_policy': 'all'})
|
|
|
5e9bef |
self.assert_(res=='ok', msg)
|
|
|
5e9bef |
self.assertResult((p.obsoletes_x86_64,))
|
|
|
5e9bef |
def testObsoletenoarchTox86_64ForDependency(self):
|
|
|
5e9bef |
@@ -62,7 +64,8 @@ class SimpleObsoletesTests(OperationsTests):
|
|
|
5e9bef |
|
|
|
5e9bef |
def testObsoletenoarchToMultiarch(self):
|
|
|
5e9bef |
p = self.pkgs
|
|
|
5e9bef |
- res, msg = self.runOperation(['update'], [p.installed_noarch], [p.obsoletes_i386, p.obsoletes_x86_64])
|
|
|
5e9bef |
+ res, msg = self.runOperation(['update'], [p.installed_noarch], [p.obsoletes_i386, p.obsoletes_x86_64],
|
|
|
5e9bef |
+ {'multilib_policy': 'all'})
|
|
|
5e9bef |
self.assert_(res=='ok', msg)
|
|
|
5e9bef |
if new_behavior:
|
|
|
5e9bef |
self.assertResult((p.obsoletes_x86_64,), (p.obsoletes_i386,))
|
|
|
5e9bef |
diff --git a/yum/__init__.py b/yum/__init__.py
|
|
|
5e9bef |
index a156a6a6..7b7293f3 100644
|
|
|
5e9bef |
--- a/yum/__init__.py
|
|
|
5e9bef |
+++ b/yum/__init__.py
|
|
|
5e9bef |
@@ -4779,6 +4779,21 @@ much more problems).
|
|
|
5e9bef |
return False
|
|
|
5e9bef |
return True
|
|
|
5e9bef |
|
|
|
5e9bef |
+ def _valid_obsoleter_arch(self, obsoleter, obsoletee):
|
|
|
5e9bef |
+ """Return whether this obsoleter meets multilib_policy in case we are
|
|
|
5e9bef |
+ dealing with the noarch->arch obsoletion case."""
|
|
|
5e9bef |
+ if not self.arch.multilib or self.conf.multilib_policy != 'best':
|
|
|
5e9bef |
+ # Install everything
|
|
|
5e9bef |
+ return True
|
|
|
5e9bef |
+ if obsoletee.arch != 'noarch' or obsoleter.arch == 'noarch':
|
|
|
5e9bef |
+ # We do respect any arch->(no)arch obsoletions (having
|
|
|
5e9bef |
+ # obsoletee.i386 installed on x86_64, you'd still expect
|
|
|
5e9bef |
+ # obsoleter.i386 to replace it, even if you have
|
|
|
5e9bef |
+ # multilib_policy=best).
|
|
|
5e9bef |
+ return True
|
|
|
5e9bef |
+ # noarch->arch case
|
|
|
5e9bef |
+ return obsoleter.arch in self.arch.legit_multi_arches
|
|
|
5e9bef |
+
|
|
|
5e9bef |
def install(self, po=None, **kwargs):
|
|
|
5e9bef |
"""Mark the specified item for installation. If a package
|
|
|
5e9bef |
object is given, mark it for installation. Otherwise, mark
|
|
|
5e9bef |
@@ -5146,10 +5161,12 @@ much more problems).
|
|
|
5e9bef |
allow_missing=True)
|
|
|
5e9bef |
if obsoleting_pkg is None:
|
|
|
5e9bef |
continue
|
|
|
5e9bef |
+ installed_pkg = self.getInstalledPackageObject(installed)
|
|
|
5e9bef |
+ if not self._valid_obsoleter_arch(obsoleting_pkg, installed_pkg):
|
|
|
5e9bef |
+ continue
|
|
|
5e9bef |
topkg = self._test_loop(obsoleting_pkg, self._pkg2obspkg)
|
|
|
5e9bef |
if topkg is not None:
|
|
|
5e9bef |
obsoleting_pkg = topkg
|
|
|
5e9bef |
- installed_pkg = self.getInstalledPackageObject(installed)
|
|
|
5e9bef |
txmbr = self.tsInfo.addObsoleting(obsoleting_pkg, installed_pkg)
|
|
|
5e9bef |
self.tsInfo.addObsoleted(installed_pkg, obsoleting_pkg)
|
|
|
5e9bef |
if requiringPo:
|
|
|
5e9bef |
@@ -5183,6 +5200,7 @@ much more problems).
|
|
|
5e9bef |
|
|
|
5e9bef |
instpkgs = []
|
|
|
5e9bef |
availpkgs = []
|
|
|
5e9bef |
+ arch_specified = True
|
|
|
5e9bef |
if po: # just a po
|
|
|
5e9bef |
if po.repoid == 'installed':
|
|
|
5e9bef |
instpkgs.append(po)
|
|
|
5e9bef |
@@ -5244,6 +5262,8 @@ much more problems).
|
|
|
5e9bef |
self.logger.critical(_('No Match for argument: %s') % to_unicode(arg))
|
|
|
5e9bef |
if not self.conf.skip_missing_names_on_update:
|
|
|
5e9bef |
raise Errors.UpdateMissingNameError, _('Not tolerating missing names on update, stopping.')
|
|
|
5e9bef |
+
|
|
|
5e9bef |
+ arch_specified = '.' in kwargs['pattern']
|
|
|
5e9bef |
|
|
|
5e9bef |
else: # we have kwargs, sort them out.
|
|
|
5e9bef |
nevra_dict = self._nevra_kwarg_parse(kwargs)
|
|
|
5e9bef |
@@ -5296,12 +5316,16 @@ much more problems).
|
|
|
5e9bef |
allow_missing=True)
|
|
|
5e9bef |
if obsoleting_pkg is None:
|
|
|
5e9bef |
continue
|
|
|
5e9bef |
+ if not arch_specified and not self._valid_obsoleter_arch(obsoleting_pkg, installed_pkg):
|
|
|
5e9bef |
+ continue
|
|
|
5e9bef |
obs_pkgs.append(obsoleting_pkg)
|
|
|
5e9bef |
# NOTE: Broekn wrt. repoid
|
|
|
5e9bef |
for obsoleting_pkg in packagesNewestByName(obs_pkgs):
|
|
|
5e9bef |
tx_return.extend(self.install(po=obsoleting_pkg))
|
|
|
5e9bef |
for available_pkg in availpkgs:
|
|
|
5e9bef |
for obsoleted_pkg in self._find_obsoletees(available_pkg):
|
|
|
5e9bef |
+ if not arch_specified and not self._valid_obsoleter_arch(available_pkg, obsoleted_pkg):
|
|
|
5e9bef |
+ continue
|
|
|
5e9bef |
obsoleted = obsoleted_pkg.pkgtup
|
|
|
5e9bef |
txmbr = self.tsInfo.addObsoleting(available_pkg, obsoleted_pkg)
|
|
|
5e9bef |
if requiringPo:
|