Blame SOURCES/0002-Fix-debug-restore-command-RhBug-1844533.patch

c97985
From b94763c7f52dbbcc9920b4216d53fd8109e434c9 Mon Sep 17 00:00:00 2001
c97985
From: Marek Blaha <mblaha@redhat.com>
c97985
Date: Wed, 17 Jun 2020 15:49:50 +0200
c97985
Subject: [PATCH] Fix debug-restore command
c97985
c97985
- correctly work with install-only packages (BZ#1844533)
c97985
- do not remove current versions of packages that are supposed to be
c97985
  replaced (downgraded / upgraded)
c97985
---
c97985
 plugins/debug.py | 108 ++++++++++++++++++++++++-----------------------
c97985
 1 file changed, 56 insertions(+), 52 deletions(-)
c97985
c97985
diff --git a/plugins/debug.py b/plugins/debug.py
c97985
index 6d00613d..29c5bf78 100644
c97985
--- a/plugins/debug.py
c97985
+++ b/plugins/debug.py
c97985
@@ -201,10 +201,9 @@ def run(self):
c97985
             self.opts.filter_types = set(
c97985
                 self.opts.filter_types.replace(",", " ").split())
c97985
 
c97985
-        installed = self.base.sack.query().installed()
c97985
         dump_pkgs = self.read_dump_file(self.opts.filename[0])
c97985
 
c97985
-        self.process_installed(installed, dump_pkgs, self.opts)
c97985
+        self.process_installed(dump_pkgs, self.opts)
c97985
 
c97985
         self.process_dump(dump_pkgs, self.opts)
c97985
 
c97985
@@ -212,56 +211,63 @@ def run(self):
c97985
             self.base.resolve()
c97985
             self.base.do_transaction()
c97985
 
c97985
-    def process_installed(self, installed, dump_pkgs, opts):
c97985
-        for pkg in sorted(installed):
c97985
-            filtered = False
c97985
+    def process_installed(self, dump_pkgs, opts):
c97985
+        installed = self.base.sack.query().installed()
c97985
+        installonly_pkgs = self.base._get_installonly_query(installed)
c97985
+        for pkg in installed:
c97985
+            pkg_remove = False
c97985
             spec = pkgspec(pkg)
c97985
-            action, dn, da, de, dv, dr = dump_pkgs.get((pkg.name, pkg.arch),
c97985
-                                                       [None, None, None,
c97985
-                                                        None, None, None])
c97985
-            dump_naevr = (dn, da, de, dv, dr)
c97985
-            if pkg.pkgtup == dump_naevr:
c97985
-                # package unchanged
c97985
-                del dump_pkgs[(pkg.name, pkg.arch)]
c97985
-            else:
c97985
-                if action == "install":
c97985
-                    # already have some version
c97985
-                    dump_pkgs[(pkg.name, pkg.arch)][0] = "replace"
c97985
-                    if "replace" not in opts.filter_types:
c97985
-                        filtered = True
c97985
+            dumped_versions = dump_pkgs.get((pkg.name, pkg.arch), None)
c97985
+            if dumped_versions is not None:
c97985
+                evr = (pkg.epoch, pkg.version, pkg.release)
c97985
+                if evr in dumped_versions:
c97985
+                    # the correct version is already installed
c97985
+                    dumped_versions[evr] = 'skip'
c97985
                 else:
c97985
-                    if "remove" not in opts.filter_types:
c97985
-                        filtered = True
c97985
-                if not filtered:
c97985
-                    if opts.output:
c97985
-                        print("remove    %s" % spec)
c97985
+                    # other version is currently installed
c97985
+                    if pkg in installonly_pkgs:
c97985
+                        # package is install-only, should be removed
c97985
+                        pkg_remove = True
c97985
                     else:
c97985
-                        self.base.package_remove(pkg)
c97985
-
c97985
-    def process_dump(self, dump_pkgs, opts):
c97985
-        for (action, n, a, e, v, r) in sorted(dump_pkgs.values()):
c97985
-            filtered = False
c97985
-            if opts.ignore_arch:
c97985
-                arch = ""
c97985
-            else:
c97985
-                arch = "." + a
c97985
-            if opts.install_latest and action == "install":
c97985
-                pkg_spec = "%s%s" % (n, arch)
c97985
-                if "install" not in opts.filter_types:
c97985
-                    filtered = True
c97985
+                        # package should be upgraded / downgraded
c97985
+                        if "replace" in opts.filter_types:
c97985
+                            action = 'replace'
c97985
+                        else:
c97985
+                            action = 'skip'
c97985
+                        for d_evr in dumped_versions.keys():
c97985
+                            dumped_versions[d_evr] = action
c97985
             else:
c97985
-                pkg_spec = pkgtup2spec(n, arch, e, v, r)
c97985
-                if (action == "replace" and
c97985
-                        "replace" not in opts.filter_types):
c97985
-                    filtered = True
c97985
-            if not filtered:
c97985
+                # package should not be installed
c97985
+                pkg_remove = True
c97985
+            if pkg_remove and "remove" in opts.filter_types:
c97985
                 if opts.output:
c97985
-                    print("install   %s" % pkg_spec)
c97985
+                    print("remove    %s" % spec)
c97985
                 else:
c97985
-                    try:
c97985
-                        self.base.install(pkg_spec)
c97985
-                    except dnf.exceptions.MarkingError:
c97985
-                        logger.error(_("Package %s is not available"), pkg_spec)
c97985
+                    self.base.package_remove(pkg)
c97985
+
c97985
+    def process_dump(self, dump_pkgs, opts):
c97985
+        for (n, a) in sorted(dump_pkgs.keys()):
c97985
+            dumped_versions = dump_pkgs[(n, a)]
c97985
+            for (e, v, r) in sorted(dumped_versions.keys()):
c97985
+                action = dumped_versions[(e, v, r)]
c97985
+                if action == 'skip':
c97985
+                    continue
c97985
+                if opts.ignore_arch:
c97985
+                    arch = ""
c97985
+                else:
c97985
+                    arch = "." + a
c97985
+                if opts.install_latest and action == "install":
c97985
+                    pkg_spec = "%s%s" % (n, arch)
c97985
+                else:
c97985
+                    pkg_spec = pkgtup2spec(n, arch, e, v, r)
c97985
+                if action in opts.filter_types:
c97985
+                    if opts.output:
c97985
+                        print("%s   %s" % (action, pkg_spec))
c97985
+                    else:
c97985
+                        try:
c97985
+                            self.base.install(pkg_spec)
c97985
+                        except dnf.exceptions.MarkingError:
c97985
+                            logger.error(_("Package %s is not available"), pkg_spec)
c97985
 
c97985
     @staticmethod
c97985
     def read_dump_file(filename):
c97985
@@ -288,11 +294,9 @@ def read_dump_file(filename):
c97985
 
c97985
             pkg_spec = line.strip()
c97985
             nevra = hawkey.split_nevra(pkg_spec)
c97985
-            pkgs[(nevra.name, nevra.arch)] = ["install", ucd(nevra.name),
c97985
-                                              ucd(nevra.arch),
c97985
-                                              ucd(nevra.epoch),
c97985
-                                              ucd(nevra.version),
c97985
-                                              ucd(nevra.release)]
c97985
+            # {(name, arch): {(epoch, version, release): action}}
c97985
+            pkgs.setdefault((nevra.name, nevra.arch), {})[
c97985
+                (nevra.epoch, nevra.version, nevra.release)] = "install"
c97985
 
c97985
         return pkgs
c97985
 
c97985
@@ -321,6 +325,6 @@ def pkgspec(pkg):
c97985
 
c97985
 
c97985
 def pkgtup2spec(name, arch, epoch, version, release):
c97985
-    a = "" if not arch else ".%s" % arch
c97985
+    a = "" if not arch else ".%s" % arch.lstrip('.')
c97985
     e = "" if epoch in (None, "") else "%s:" % epoch
c97985
     return "%s-%s%s-%s%s" % (name, e, version, release, a)