From 2ee360753c8fa937d9c81bf7da24f457041650bc Mon Sep 17 00:00:00 2001 From: Victor Zhestkov <35733135+vzhestkov@users.noreply.github.com> Date: Mon, 5 Jul 2021 18:39:26 +0300 Subject: [PATCH] Implementation of held/unheld functions for state pkg (#387) * Implementation of held/unheld functions for state pkg --- salt/modules/zypperpkg.py | 201 +++++++++--- salt/states/pkg.py | 310 +++++++++++++++++++ tests/pytests/unit/modules/test_zypperpkg.py | 142 +++++++++ tests/pytests/unit/states/test_pkg.py | 155 ++++++++++ 4 files changed, 760 insertions(+), 48 deletions(-) create mode 100644 tests/pytests/unit/modules/test_zypperpkg.py create mode 100644 tests/pytests/unit/states/test_pkg.py diff --git a/salt/modules/zypperpkg.py b/salt/modules/zypperpkg.py index e064e2cb4e..932b30bac5 100644 --- a/salt/modules/zypperpkg.py +++ b/salt/modules/zypperpkg.py @@ -2071,6 +2071,76 @@ def purge( return _uninstall(inclusion_detection, name=name, pkgs=pkgs, root=root) +def list_holds(pattern=None, full=True, root=None, **kwargs): + """ + List information on locked packages. + + .. note:: + This function returns the computed output of ``list_locks`` + to show exact locked packages. + + pattern + Regular expression used to match the package name + + full : True + Show the full hold definition including version and epoch. Set to + ``False`` to return just the name of the package(s) being held. + + root + Operate on a different root directory. + + + CLI Example: + + .. code-block:: bash + + salt '*' pkg.list_holds + salt '*' pkg.list_holds full=False + """ + locks = list_locks(root=root) + ret = [] + inst_pkgs = {} + for solv_name, lock in locks.items(): + if lock.get("type", "package") != "package": + continue + try: + found_pkgs = search( + solv_name, + root=root, + match=None if "*" in solv_name else "exact", + case_sensitive=(lock.get("case_sensitive", "on") == "on"), + installed_only=True, + details=True, + all_versions=True, + ignore_no_matching_item=True, + ) + except CommandExecutionError: + continue + if found_pkgs: + for pkg in found_pkgs: + if pkg not in inst_pkgs: + inst_pkgs.update( + info_installed( + pkg, root=root, attr="edition,epoch", all_versions=True + ) + ) + + ptrn_re = re.compile(r"{}-\S+".format(pattern)) if pattern else None + for pkg_name, pkg_editions in inst_pkgs.items(): + for pkg_info in pkg_editions: + pkg_ret = ( + "{}-{}:{}.*".format( + pkg_name, pkg_info.get("epoch", 0), pkg_info.get("edition") + ) + if full + else pkg_name + ) + if pkg_ret not in ret and (not ptrn_re or ptrn_re.match(pkg_ret)): + ret.append(pkg_ret) + + return ret + + def list_locks(root=None): """ List current package locks. @@ -2141,43 +2211,68 @@ def clean_locks(root=None): return out -def unhold(name=None, pkgs=None, **kwargs): +def unhold(name=None, pkgs=None, root=None, **kwargs): """ - Remove specified package lock. + Remove a package hold. + + name + A package name to unhold, or a comma-separated list of package names to + unhold. + + pkgs + A list of packages to unhold. The ``name`` parameter will be ignored if + this option is passed. root - operate on a different root directory. + Operate on a different root directory. CLI Example: .. code-block:: bash - salt '*' pkg.remove_lock - salt '*' pkg.remove_lock ,, - salt '*' pkg.remove_lock pkgs='["foo", "bar"]' + salt '*' pkg.unhold + salt '*' pkg.unhold ,, + salt '*' pkg.unhold pkgs='["foo", "bar"]' """ ret = {} - root = kwargs.get("root") - if (not name and not pkgs) or (name and pkgs): + if not name and not pkgs: raise CommandExecutionError("Name or packages must be specified.") - elif name: - pkgs = [name] - locks = list_locks(root) - try: - pkgs = list(__salt__["pkg_resource.parse_targets"](pkgs)[0].keys()) - except MinionError as exc: - raise CommandExecutionError(exc) + targets = [] + if pkgs: + targets.extend(pkgs) + else: + targets.append(name) + locks = list_locks() removed = [] - missing = [] - for pkg in pkgs: - if locks.get(pkg): - removed.append(pkg) - ret[pkg]["comment"] = "Package {} is no longer held.".format(pkg) + + for target in targets: + version = None + if isinstance(target, dict): + (target, version) = next(iter(target.items())) + ret[target] = {"name": target, "changes": {}, "result": True, "comment": ""} + if locks.get(target): + lock_ver = None + if "version" in locks.get(target): + lock_ver = locks.get(target)["version"] + lock_ver = lock_ver.lstrip("= ") + if version and lock_ver != version: + ret[target]["result"] = False + ret[target][ + "comment" + ] = "Unable to unhold package {} as it is held with the other version.".format( + target + ) + else: + removed.append( + target if not lock_ver else "{}={}".format(target, lock_ver) + ) + ret[target]["changes"]["new"] = "" + ret[target]["changes"]["old"] = "hold" + ret[target]["comment"] = "Package {} is no longer held.".format(target) else: - missing.append(pkg) - ret[pkg]["comment"] = "Package {} unable to be unheld.".format(pkg) + ret[target]["comment"] = "Package {} was already unheld.".format(target) if removed: __zypper__(root=root).call("rl", *removed) @@ -2223,47 +2318,57 @@ def remove_lock(packages, root=None, **kwargs): # pylint: disable=unused-argume return {"removed": len(removed), "not_found": missing} -def hold(name=None, pkgs=None, **kwargs): +def hold(name=None, pkgs=None, root=None, **kwargs): """ - Add a package lock. Specify packages to lock by exact name. + Add a package hold. Specify one of ``name`` and ``pkgs``. + + name + A package name to hold, or a comma-separated list of package names to + hold. + + pkgs + A list of packages to hold. The ``name`` parameter will be ignored if + this option is passed. root - operate on a different root directory. + Operate on a different root directory. + CLI Example: .. code-block:: bash - salt '*' pkg.add_lock - salt '*' pkg.add_lock ,, - salt '*' pkg.add_lock pkgs='["foo", "bar"]' - - :param name: - :param pkgs: - :param kwargs: - :return: + salt '*' pkg.hold + salt '*' pkg.hold ,, + salt '*' pkg.hold pkgs='["foo", "bar"]' """ ret = {} - root = kwargs.get("root") - if (not name and not pkgs) or (name and pkgs): + if not name and not pkgs: raise CommandExecutionError("Name or packages must be specified.") - elif name: - pkgs = [name] - locks = list_locks(root=root) + targets = [] + if pkgs: + targets.extend(pkgs) + else: + targets.append(name) + + locks = list_locks() added = [] - try: - pkgs = list(__salt__["pkg_resource.parse_targets"](pkgs)[0].keys()) - except MinionError as exc: - raise CommandExecutionError(exc) - for pkg in pkgs: - ret[pkg] = {"name": pkg, "changes": {}, "result": False, "comment": ""} - if not locks.get(pkg): - added.append(pkg) - ret[pkg]["comment"] = "Package {} is now being held.".format(pkg) + for target in targets: + version = None + if isinstance(target, dict): + (target, version) = next(iter(target.items())) + ret[target] = {"name": target, "changes": {}, "result": True, "comment": ""} + if not locks.get(target): + added.append(target if not version else "{}={}".format(target, version)) + ret[target]["changes"]["new"] = "hold" + ret[target]["changes"]["old"] = "" + ret[target]["comment"] = "Package {} is now being held.".format(target) else: - ret[pkg]["comment"] = "Package {} is already set to be held.".format(pkg) + ret[target]["comment"] = "Package {} is already set to be held.".format( + target + ) if added: __zypper__(root=root).call("al", *added) diff --git a/salt/states/pkg.py b/salt/states/pkg.py index f7327a33e3..0ef3f056c5 100644 --- a/salt/states/pkg.py +++ b/salt/states/pkg.py @@ -3550,3 +3550,313 @@ def mod_watch(name, **kwargs): "comment": "pkg.{} does not work with the watch requisite".format(sfun), "result": False, } + + +def held(name, version=None, pkgs=None, replace=False, **kwargs): + """ + Set package in 'hold' state, meaning it will not be changed. + + :param str name: + The name of the package to be held. This parameter is ignored + if ``pkgs`` is used. + + :param str version: + Hold a specific version of a package. + Full description of this parameter is in `installed` function. + + .. note:: + + This parameter make sense for Zypper-based systems. + Ignored for YUM/DNF and APT + + :param list pkgs: + A list of packages to be held. All packages listed under ``pkgs`` + will be held. + + .. code-block:: yaml + + mypkgs: + pkg.held: + - pkgs: + - foo + - bar: 1.2.3-4 + - baz + + .. note:: + + For Zypper-based systems the package could be held for + the version specified. YUM/DNF and APT ingore it. + + :param bool replace: + Force replacement of existings holds with specified. + By default, this parameter is set to ``False``. + """ + + if isinstance(pkgs, list) and len(pkgs) == 0 and not replace: + return { + "name": name, + "changes": {}, + "result": True, + "comment": "No packages to be held provided", + } + + # If just a name (and optionally a version) is passed, just pack them into + # the pkgs argument. + if name and pkgs is None: + if version: + pkgs = [{name: version}] + version = None + else: + pkgs = [name] + + locks = {} + vr_lock = False + if "pkg.list_locks" in __salt__: + locks = __salt__["pkg.list_locks"]() + vr_lock = True + elif "pkg.list_holds" in __salt__: + _locks = __salt__["pkg.list_holds"](full=True) + lock_re = re.compile(r"^(.+)-(\d+):(.*)\.\*") + for lock in _locks: + match = lock_re.match(lock) + if match: + epoch = match.group(2) + if epoch == "0": + epoch = "" + else: + epoch = "{}:".format(epoch) + locks.update( + {match.group(1): {"version": "{}{}".format(epoch, match.group(3))}} + ) + else: + locks.update({lock: {}}) + elif "pkg.get_selections" in __salt__: + _locks = __salt__["pkg.get_selections"](state="hold") + for lock in _locks.get("hold", []): + locks.update({lock: {}}) + else: + return { + "name": name, + "changes": {}, + "result": False, + "comment": "No any function to get the list of held packages available.\n" + "Check if the package manager supports package locking.", + } + + if "pkg.hold" not in __salt__: + return { + "name": name, + "changes": {}, + "result": False, + "comment": "`hold` function is not implemented for the package manager.", + } + + ret = {"name": name, "changes": {}, "result": True, "comment": ""} + comments = [] + + held_pkgs = set() + for pkg in pkgs: + if isinstance(pkg, dict): + (pkg_name, pkg_ver) = next(iter(pkg.items())) + else: + pkg_name = pkg + pkg_ver = None + lock_ver = None + if pkg_name in locks and "version" in locks[pkg_name]: + lock_ver = locks[pkg_name]["version"] + lock_ver = lock_ver.lstrip("= ") + held_pkgs.add(pkg_name) + if pkg_name not in locks or (vr_lock and lock_ver != pkg_ver): + if __opts__["test"]: + if pkg_name in locks: + comments.append( + "The following package's hold rule would be updated: {}{}".format( + pkg_name, + "" if not pkg_ver else " (version = {})".format(pkg_ver), + ) + ) + else: + comments.append( + "The following package would be held: {}{}".format( + pkg_name, + "" if not pkg_ver else " (version = {})".format(pkg_ver), + ) + ) + else: + unhold_ret = None + if pkg_name in locks: + unhold_ret = __salt__["pkg.unhold"](name=name, pkgs=[pkg_name]) + hold_ret = __salt__["pkg.hold"](name=name, pkgs=[pkg]) + if not hold_ret.get(pkg_name, {}).get("result", False): + ret["result"] = False + if ( + unhold_ret + and unhold_ret.get(pkg_name, {}).get("result", False) + and hold_ret + and hold_ret.get(pkg_name, {}).get("result", False) + ): + comments.append( + "Package {} was updated with hold rule".format(pkg_name) + ) + elif hold_ret and hold_ret.get(pkg_name, {}).get("result", False): + comments.append("Package {} is now being held".format(pkg_name)) + else: + comments.append("Package {} was not held".format(pkg_name)) + ret["changes"].update(hold_ret) + + if replace: + for pkg_name in locks: + if locks[pkg_name].get("type", "package") != "package": + continue + if __opts__["test"]: + if pkg_name not in held_pkgs: + comments.append( + "The following package would be unheld: {}".format(pkg_name) + ) + else: + if pkg_name not in held_pkgs: + unhold_ret = __salt__["pkg.unhold"](name=name, pkgs=[pkg_name]) + if not unhold_ret.get(pkg_name, {}).get("result", False): + ret["result"] = False + if unhold_ret and unhold_ret.get(pkg_name, {}).get("comment"): + comments.append(unhold_ret.get(pkg_name).get("comment")) + ret["changes"].update(unhold_ret) + + ret["comment"] = "\n".join(comments) + if not (ret["changes"] or ret["comment"]): + ret["comment"] = "No changes made" + + return ret + + +def unheld(name, version=None, pkgs=None, all=False, **kwargs): + """ + Unset package from 'hold' state, to allow operations with the package. + + :param str name: + The name of the package to be unheld. This parameter is ignored if "pkgs" + is used. + + :param str version: + Unhold a specific version of a package. + Full description of this parameter is in `installed` function. + + .. note:: + + This parameter make sense for Zypper-based systems. + Ignored for YUM/DNF and APT. + + :param list pkgs: + A list of packages to be unheld. All packages listed under ``pkgs`` + will be unheld. + + .. code-block:: yaml + + mypkgs: + pkg.unheld: + - pkgs: + - foo + - bar: 1.2.3-4 + - baz + + .. note:: + + For Zypper-based systems the package could be held for + the version specified. YUM/DNF and APT ingore it. + For ``unheld`` there is no need to specify the exact version + to be unheld. + + :param bool all: + Force removing of all existings locks. + By default, this parameter is set to ``False``. + """ + + if isinstance(pkgs, list) and len(pkgs) == 0 and not all: + return { + "name": name, + "changes": {}, + "result": True, + "comment": "No packages to be unheld provided", + } + + # If just a name (and optionally a version) is passed, just pack them into + # the pkgs argument. + if name and pkgs is None: + pkgs = [{name: version}] + version = None + + locks = {} + vr_lock = False + if "pkg.list_locks" in __salt__: + locks = __salt__["pkg.list_locks"]() + vr_lock = True + elif "pkg.list_holds" in __salt__: + _locks = __salt__["pkg.list_holds"](full=True) + lock_re = re.compile(r"^(.+)-(\d+):(.*)\.\*") + for lock in _locks: + match = lock_re.match(lock) + if match: + epoch = match.group(2) + if epoch == "0": + epoch = "" + else: + epoch = "{}:".format(epoch) + locks.update( + {match.group(1): {"version": "{}{}".format(epoch, match.group(3))}} + ) + else: + locks.update({lock: {}}) + elif "pkg.get_selections" in __salt__: + _locks = __salt__["pkg.get_selections"](state="hold") + for lock in _locks.get("hold", []): + locks.update({lock: {}}) + else: + return { + "name": name, + "changes": {}, + "result": False, + "comment": "No any function to get the list of held packages available.\n" + "Check if the package manager supports package locking.", + } + + dpkgs = {} + for pkg in pkgs: + if isinstance(pkg, dict): + (pkg_name, pkg_ver) = next(iter(pkg.items())) + dpkgs.update({pkg_name: pkg_ver}) + else: + dpkgs.update({pkg: None}) + + ret = {"name": name, "changes": {}, "result": True, "comment": ""} + comments = [] + + for pkg_name in locks: + if locks[pkg_name].get("type", "package") != "package": + continue + lock_ver = None + if vr_lock and "version" in locks[pkg_name]: + lock_ver = locks[pkg_name]["version"] + lock_ver = lock_ver.lstrip("= ") + if all or (pkg_name in dpkgs and (not lock_ver or lock_ver == dpkgs[pkg_name])): + if __opts__["test"]: + comments.append( + "The following package would be unheld: {}{}".format( + pkg_name, + "" + if not dpkgs.get(pkg_name) + else " (version = {})".format(lock_ver), + ) + ) + else: + unhold_ret = __salt__["pkg.unhold"](name=name, pkgs=[pkg_name]) + if not unhold_ret.get(pkg_name, {}).get("result", False): + ret["result"] = False + if unhold_ret and unhold_ret.get(pkg_name, {}).get("comment"): + comments.append(unhold_ret.get(pkg_name).get("comment")) + ret["changes"].update(unhold_ret) + + ret["comment"] = "\n".join(comments) + if not (ret["changes"] or ret["comment"]): + ret["comment"] = "No changes made" + + return ret diff --git a/tests/pytests/unit/modules/test_zypperpkg.py b/tests/pytests/unit/modules/test_zypperpkg.py new file mode 100644 index 0000000000..464fae1f47 --- /dev/null +++ b/tests/pytests/unit/modules/test_zypperpkg.py @@ -0,0 +1,142 @@ +import pytest +import salt.modules.pkg_resource as pkg_resource +import salt.modules.zypperpkg as zypper +from tests.support.mock import MagicMock, patch + + +@pytest.fixture +def configure_loader_modules(): + return {zypper: {"rpm": None}, pkg_resource: {}} + + +def test_pkg_hold(): + """ + Tests holding packages with Zypper + """ + + # Test openSUSE 15.3 + list_locks_mock = { + "bar": {"type": "package", "match_type": "glob", "case_sensitive": "on"}, + "minimal_base": { + "type": "pattern", + "match_type": "glob", + "case_sensitive": "on", + }, + "baz": {"type": "package", "match_type": "glob", "case_sensitive": "on"}, + } + + cmd = MagicMock( + return_value={ + "pid": 1234, + "retcode": 0, + "stdout": "Specified lock has been successfully added.", + "stderr": "", + } + ) + with patch.object( + zypper, "list_locks", MagicMock(return_value=list_locks_mock) + ), patch.dict(zypper.__salt__, {"cmd.run_all": cmd}): + ret = zypper.hold("foo") + assert ret["foo"]["changes"]["old"] == "" + assert ret["foo"]["changes"]["new"] == "hold" + assert ret["foo"]["comment"] == "Package foo is now being held." + cmd.assert_called_once_with( + ["zypper", "--non-interactive", "--no-refresh", "al", "foo"], + env={}, + output_loglevel="trace", + python_shell=False, + ) + cmd.reset_mock() + ret = zypper.hold(pkgs=["foo", "bar"]) + assert ret["foo"]["changes"]["old"] == "" + assert ret["foo"]["changes"]["new"] == "hold" + assert ret["foo"]["comment"] == "Package foo is now being held." + assert ret["bar"]["changes"] == {} + assert ret["bar"]["comment"] == "Package bar is already set to be held." + cmd.assert_called_once_with( + ["zypper", "--non-interactive", "--no-refresh", "al", "foo"], + env={}, + output_loglevel="trace", + python_shell=False, + ) + + +def test_pkg_unhold(): + """ + Tests unholding packages with Zypper + """ + + # Test openSUSE 15.3 + list_locks_mock = { + "bar": {"type": "package", "match_type": "glob", "case_sensitive": "on"}, + "minimal_base": { + "type": "pattern", + "match_type": "glob", + "case_sensitive": "on", + }, + "baz": {"type": "package", "match_type": "glob", "case_sensitive": "on"}, + } + + cmd = MagicMock( + return_value={ + "pid": 1234, + "retcode": 0, + "stdout": "1 lock has been successfully removed.", + "stderr": "", + } + ) + with patch.object( + zypper, "list_locks", MagicMock(return_value=list_locks_mock) + ), patch.dict(zypper.__salt__, {"cmd.run_all": cmd}): + ret = zypper.unhold("foo") + assert ret["foo"]["comment"] == "Package foo was already unheld." + cmd.assert_not_called() + cmd.reset_mock() + ret = zypper.unhold(pkgs=["foo", "bar"]) + assert ret["foo"]["changes"] == {} + assert ret["foo"]["comment"] == "Package foo was already unheld." + assert ret["bar"]["changes"]["old"] == "hold" + assert ret["bar"]["changes"]["new"] == "" + assert ret["bar"]["comment"] == "Package bar is no longer held." + cmd.assert_called_once_with( + ["zypper", "--non-interactive", "--no-refresh", "rl", "bar"], + env={}, + output_loglevel="trace", + python_shell=False, + ) + + +def test_pkg_list_holds(): + """ + Tests listing of calculated held packages with Zypper + """ + + # Test openSUSE 15.3 + list_locks_mock = { + "bar": {"type": "package", "match_type": "glob", "case_sensitive": "on"}, + "minimal_base": { + "type": "pattern", + "match_type": "glob", + "case_sensitive": "on", + }, + "baz": {"type": "package", "match_type": "glob", "case_sensitive": "on"}, + } + installed_pkgs = { + "foo": [{"edition": "1.2.3-1.1"}], + "bar": [{"edition": "2.3.4-2.1", "epoch": "2"}], + } + + def zypper_search_mock(name, *_args, **_kwargs): + if name in installed_pkgs: + return {name: installed_pkgs.get(name)} + + with patch.object( + zypper, "list_locks", MagicMock(return_value=list_locks_mock) + ), patch.object( + zypper, "search", MagicMock(side_effect=zypper_search_mock) + ), patch.object( + zypper, "info_installed", MagicMock(side_effect=zypper_search_mock) + ): + ret = zypper.list_holds() + assert len(ret) == 1 + assert "bar-2:2.3.4-2.1.*" in ret diff --git a/tests/pytests/unit/states/test_pkg.py b/tests/pytests/unit/states/test_pkg.py new file mode 100644 index 0000000000..faf42c4681 --- /dev/null +++ b/tests/pytests/unit/states/test_pkg.py @@ -0,0 +1,155 @@ +import pytest +import salt.states.pkg as pkg +from tests.support.mock import MagicMock, patch + + +@pytest.fixture +def configure_loader_modules(): + return { + pkg: { + "__env__": "base", + "__salt__": {}, + "__grains__": {"os": "CentOS"}, + "__opts__": {"test": False, "cachedir": ""}, + "__instance_id__": "", + "__low__": {}, + "__utils__": {}, + }, + } + + +@pytest.mark.parametrize( + "package_manager", [("Zypper"), ("YUM/DNF"), ("APT")], +) +def test_held_unheld(package_manager): + """ + Test pkg.held and pkg.unheld with Zypper, YUM/DNF and APT + """ + + if package_manager == "Zypper": + list_holds_func = "pkg.list_locks" + list_holds_mock = MagicMock( + return_value={ + "bar": { + "type": "package", + "match_type": "glob", + "case_sensitive": "on", + }, + "minimal_base": { + "type": "pattern", + "match_type": "glob", + "case_sensitive": "on", + }, + "baz": { + "type": "package", + "match_type": "glob", + "case_sensitive": "on", + }, + } + ) + elif package_manager == "YUM/DNF": + list_holds_func = "pkg.list_holds" + list_holds_mock = MagicMock( + return_value=["bar-0:1.2.3-1.1.*", "baz-0:2.3.4-2.1.*"] + ) + elif package_manager == "APT": + list_holds_func = "pkg.get_selections" + list_holds_mock = MagicMock(return_value={"hold": ["bar", "baz"]}) + + def pkg_hold(name, pkgs=None, *_args, **__kwargs): + if name and pkgs is None: + pkgs = [name] + ret = {} + for pkg in pkgs: + ret.update( + { + pkg: { + "name": pkg, + "changes": {"new": "hold", "old": ""}, + "result": True, + "comment": "Package {} is now being held.".format(pkg), + } + } + ) + return ret + + def pkg_unhold(name, pkgs=None, *_args, **__kwargs): + if name and pkgs is None: + pkgs = [name] + ret = {} + for pkg in pkgs: + ret.update( + { + pkg: { + "name": pkg, + "changes": {"new": "", "old": "hold"}, + "result": True, + "comment": "Package {} is no longer held.".format(pkg), + } + } + ) + return ret + + hold_mock = MagicMock(side_effect=pkg_hold) + unhold_mock = MagicMock(side_effect=pkg_unhold) + + # Testing with Zypper + with patch.dict( + pkg.__salt__, + { + list_holds_func: list_holds_mock, + "pkg.hold": hold_mock, + "pkg.unhold": unhold_mock, + }, + ): + # Holding one of two packages + ret = pkg.held("held-test", pkgs=["foo", "bar"]) + assert "foo" in ret["changes"] + assert len(ret["changes"]) == 1 + hold_mock.assert_called_once_with(name="held-test", pkgs=["foo"]) + unhold_mock.assert_not_called() + + hold_mock.reset_mock() + unhold_mock.reset_mock() + + # Holding one of two packages and replacing all the rest held packages + ret = pkg.held("held-test", pkgs=["foo", "bar"], replace=True) + assert "foo" in ret["changes"] + assert "baz" in ret["changes"] + assert len(ret["changes"]) == 2 + hold_mock.assert_called_once_with(name="held-test", pkgs=["foo"]) + unhold_mock.assert_called_once_with(name="held-test", pkgs=["baz"]) + + hold_mock.reset_mock() + unhold_mock.reset_mock() + + # Remove all holds + ret = pkg.held("held-test", pkgs=[], replace=True) + assert "bar" in ret["changes"] + assert "baz" in ret["changes"] + assert len(ret["changes"]) == 2 + hold_mock.assert_not_called() + unhold_mock.assert_any_call(name="held-test", pkgs=["baz"]) + unhold_mock.assert_any_call(name="held-test", pkgs=["bar"]) + + hold_mock.reset_mock() + unhold_mock.reset_mock() + + # Unolding one of two packages + ret = pkg.unheld("held-test", pkgs=["foo", "bar"]) + assert "bar" in ret["changes"] + assert len(ret["changes"]) == 1 + unhold_mock.assert_called_once_with(name="held-test", pkgs=["bar"]) + hold_mock.assert_not_called() + + hold_mock.reset_mock() + unhold_mock.reset_mock() + + # Remove all holds + ret = pkg.unheld("held-test", all=True) + assert "bar" in ret["changes"] + assert "baz" in ret["changes"] + assert len(ret["changes"]) == 2 + hold_mock.assert_not_called() + unhold_mock.assert_any_call(name="held-test", pkgs=["baz"]) + unhold_mock.assert_any_call(name="held-test", pkgs=["bar"]) -- 2.32.0