D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
opt
/
saltstack
/
salt
/
lib
/
python3.10
/
site-packages
/
salt
/
modules
/
Filename :
zypperpkg.py
back
Copy
""" Package support for openSUSE via the zypper package manager :depends: - ``rpm`` Python module. Install with ``zypper install rpm-python`` .. important:: If you feel that Salt should be using this module to manage packages on a minion, and it is using a different module (or gives an error similar to *'pkg.install' is not available*), see :ref:`here <module-provider-override>`. """ import configparser import datetime import errno import fnmatch import logging import os import re import time import urllib.parse from xml.dom import minidom as dom from xml.parsers.expat import ExpatError import salt.utils.data import salt.utils.environment import salt.utils.event import salt.utils.files import salt.utils.functools import salt.utils.path import salt.utils.pkg import salt.utils.pkg.rpm import salt.utils.stringutils import salt.utils.systemd import salt.utils.versions from salt.exceptions import CommandExecutionError, MinionError, SaltInvocationError from salt.utils.versions import LooseVersion if salt.utils.files.is_fcntl_available(): import fcntl log = logging.getLogger(__name__) HAS_ZYPP = False ZYPP_HOME = "/etc/zypp" LOCKS = "{}/locks".format(ZYPP_HOME) REPOS = "{}/repos.d".format(ZYPP_HOME) DEFAULT_PRIORITY = 99 PKG_ARCH_SEPARATOR = "." # Define the module's virtual name __virtualname__ = "pkg" def __virtual__(): """ Set the virtual pkg module if the os is openSUSE """ if __grains__.get("os_family", "") != "Suse": return ( False, "Module zypper: non SUSE OS not supported by zypper package manager", ) # Not all versions of SUSE use zypper, check that it is available if not salt.utils.path.which("zypper"): return (False, "Module zypper: zypper package manager not found") return __virtualname__ class _Zypper: """ Zypper parallel caller. Validates the result and either raises an exception or reports an error. Allows serial zypper calls (first came, first won). """ SUCCESS_EXIT_CODES = { 0: "Successful run of zypper with no special info.", 100: "Patches are available for installation.", 101: "Security patches are available for installation.", 102: "Installation successful, reboot required.", 103: "Installation successful, restart of the package manager itself required.", } WARNING_EXIT_CODES = { 6: "No repositories are defined.", 7: "The ZYPP library is locked.", 106: ( "Some repository had to be disabled temporarily because it failed to" " refresh. You should check your repository configuration (e.g. zypper ref" " -f)." ), 107: ( "Installation basically succeeded, but some of the packages %post install" " scripts returned an error. These packages were successfully unpacked to" " disk and are registered in the rpm database, but due to the failed" " install script they may not work as expected. The failed scripts output" " might reveal what actually went wrong. Any scripts output is also logged" " to /var/log/zypp/history." ), } LOCK_EXIT_CODE = 7 XML_DIRECTIVES = ["-x", "--xmlout"] # ZYPPER_LOCK is not affected by --root ZYPPER_LOCK = "/var/run/zypp.pid" RPM_LOCK = "/var/lib/rpm/.rpm.lock" TAG_RELEASED = "zypper/released" TAG_BLOCKED = "zypper/blocked" def __init__(self): """ Constructor """ self._reset() def _reset(self): """ Resets values of the call setup. :return: """ self.__cmd = ["zypper", "--non-interactive"] self.__exit_code = 0 self.__call_result = dict() self.__error_msg = "" self.__env = salt.utils.environment.get_module_environment(globals()) # Call config self.__xml = False self.__no_lock = False self.__no_raise = False self.__refresh = False self.__ignore_repo_failure = False self.__systemd_scope = False self.__root = None # Call status self.__called = False def __call__(self, *args, **kwargs): """ :param args: :param kwargs: :return: """ # Reset after the call if self.__called: self._reset() # Ignore exit code for 106 (repo is not available) if "no_repo_failure" in kwargs: self.__ignore_repo_failure = kwargs["no_repo_failure"] if "systemd_scope" in kwargs: self.__systemd_scope = kwargs["systemd_scope"] if "root" in kwargs: self.__root = kwargs["root"] return self def __getattr__(self, item): """ Call configurator. :param item: :return: """ # Reset after the call if self.__called: self._reset() if item == "xml": self.__xml = True elif item == "nolock": self.__no_lock = True elif item == "noraise": self.__no_raise = True elif item == "refreshable": self.__refresh = True elif item == "call": return self.__call else: return self.__dict__[item] # Prevent the use of "refreshable" together with "nolock". if self.__no_lock: self.__no_lock = not self.__refresh return self @property def exit_code(self): return self.__exit_code @exit_code.setter def exit_code(self, exit_code): self.__exit_code = int(exit_code or "0") @property def error_msg(self): return self.__error_msg @error_msg.setter def error_msg(self, msg): if self._is_error(): self.__error_msg = msg and os.linesep.join(msg) or "Check Zypper's logs." @property def stdout(self): return self.__call_result.get("stdout", "") @property def stderr(self): return self.__call_result.get("stderr", "") @property def pid(self): return self.__call_result.get("pid", "") def _is_error(self): """ Is this is an error code? :return: """ if self.exit_code: msg = self.SUCCESS_EXIT_CODES.get(self.exit_code) if msg: log.info(msg) msg = self.WARNING_EXIT_CODES.get(self.exit_code) if msg: log.warning(msg) return ( self.exit_code not in self.SUCCESS_EXIT_CODES and self.exit_code not in self.WARNING_EXIT_CODES ) def _is_zypper_lock(self): """ Is this is a lock error code? :return: """ return self.exit_code == self.LOCK_EXIT_CODE def _is_rpm_lock(self): """ Is this an RPM lock error? """ if salt.utils.files.is_fcntl_available(): if self.exit_code > 0 and os.path.exists(self.RPM_LOCK): with salt.utils.files.fopen(self.RPM_LOCK, mode="w+") as rfh: try: fcntl.lockf(rfh, fcntl.LOCK_EX | fcntl.LOCK_NB) except OSError as err: if err.errno == errno.EAGAIN: return True else: fcntl.lockf(rfh, fcntl.LOCK_UN) return False def _is_xml_mode(self): """ Is Zypper's output is in XML format? :return: """ return ( [itm for itm in self.XML_DIRECTIVES if itm in self.__cmd] and True or False ) def _check_result(self): """ Check and set the result of a zypper command. In case of an error, either raise a CommandExecutionError or extract the error. result The result of a zypper command called with cmd.run_all """ if not self.__call_result: raise CommandExecutionError("No output result from Zypper?") self.exit_code = self.__call_result["retcode"] if self._is_zypper_lock() or self._is_rpm_lock(): return False if self._is_error(): _error_msg = list() if not self._is_xml_mode(): msg = ( self.__call_result["stderr"] and self.__call_result["stderr"].strip() or "" ) msg += ( self.__call_result["stdout"] and self.__call_result["stdout"].strip() or "" ) if msg: _error_msg.append(msg) else: try: doc = dom.parseString(self.__call_result["stdout"]) except ExpatError as err: log.error(err) doc = None if doc: msg_nodes = doc.getElementsByTagName("message") for node in msg_nodes: if node.getAttribute("type") == "error": _error_msg.append(node.childNodes[0].nodeValue) elif self.__call_result["stderr"].strip(): _error_msg.append(self.__call_result["stderr"].strip()) self.error_msg = _error_msg return True def __call(self, *args, **kwargs): """ Call Zypper. :param state: :return: """ self.__called = True if self.__xml: self.__cmd.append("--xmlout") if not self.__refresh and "--no-refresh" not in args: self.__cmd.append("--no-refresh") if self.__root: self.__cmd.extend(["--root", self.__root]) self.__cmd.extend(args) kwargs["output_loglevel"] = "trace" kwargs["python_shell"] = False kwargs["env"] = self.__env.copy() if self.__no_lock: kwargs["env"][ "ZYPP_READONLY_HACK" ] = ( # Disables locking for read-only operations. Do not try that at home! "1" ) # Zypper call will stuck here waiting, if another zypper hangs until forever. # However, Zypper lock needs to be always respected. was_blocked = False while True: cmd = [] if self.__systemd_scope: cmd.extend(["systemd-run", "--scope"]) cmd.extend(self.__cmd) log.debug("Calling Zypper: %s", " ".join(cmd)) self.__call_result = __salt__["cmd.run_all"](cmd, **kwargs) if self._check_result(): break if self._is_zypper_lock(): self._handle_zypper_lock_file() if self._is_rpm_lock(): self._handle_rpm_lock_file() was_blocked = True if was_blocked: __salt__["event.fire_master"]( { "success": not self.error_msg, "info": self.error_msg or "Zypper has been released", }, self.TAG_RELEASED, ) if self.error_msg and not self.__no_raise and not self.__ignore_repo_failure: raise CommandExecutionError( "Zypper command failure: {}".format(self.error_msg) ) return ( self._is_xml_mode() and dom.parseString( salt.utils.stringutils.to_str(self.__call_result["stdout"]) ) or self.__call_result["stdout"] ) def _handle_zypper_lock_file(self): if os.path.exists(self.ZYPPER_LOCK): try: with salt.utils.files.fopen(self.ZYPPER_LOCK) as rfh: data = __salt__["ps.proc_info"]( int(rfh.readline()), attrs=["pid", "name", "cmdline", "create_time"], ) data["cmdline"] = " ".join(data["cmdline"]) data["info"] = "Blocking process created at {}.".format( datetime.datetime.utcfromtimestamp( data["create_time"] ).isoformat() ) data["success"] = True except Exception as err: # pylint: disable=broad-except data = { "info": ( "Unable to retrieve information about " "blocking process: {}".format(err) ), "success": False, } else: data = { "info": "Zypper is locked, but no Zypper lock has been found.", "success": False, } if not data["success"]: log.debug("Unable to collect data about blocking process.") else: log.debug("Collected data about blocking process.") __salt__["event.fire_master"](data, self.TAG_BLOCKED) log.debug("Fired a Zypper blocked event to the master with the data: %s", data) log.debug("Waiting 5 seconds for Zypper gets released...") time.sleep(5) def _handle_rpm_lock_file(self): data = {"info": "RPM is temporarily locked.", "success": True} __salt__["event.fire_master"](data, self.TAG_BLOCKED) log.debug("Fired an RPM blocked event to the master with the data: %s", data) log.debug("Waiting 5 seconds for RPM to get released...") time.sleep(5) __zypper__ = _Zypper() class Wildcard: """ .. versionadded:: 2017.7.0 Converts string wildcard to a zypper query. Example: '1.2.3.4*' is '1.2.3.4.whatever.is.here' and is equal to: '1.2.3.4 >= and < 1.2.3.5' :param ptn: Pattern :return: Query range """ Z_OP = ["<", "<=", "=", ">=", ">"] def __init__(self, zypper): """ :type zypper: a reference to an instance of a _Zypper class. """ self.name = None self.version = None self.zypper = zypper self._attr_solvable_version = "edition" self._op = None def __call__(self, pkg_name, pkg_version): """ Convert a string wildcard to a zypper query. :param pkg_name: :param pkg_version: :return: """ if pkg_version: self.name = pkg_name self._set_version(pkg_version) # Dissects possible operator versions = sorted( LooseVersion(vrs) for vrs in self._get_scope_versions(self._get_available_versions()) ) return versions and "{}{}".format(self._op or "", versions[-1]) or None def _get_available_versions(self): """ Get available versions of the package. :return: """ solvables = self.zypper.nolock.xml.call( "se", "-xv", self.name ).getElementsByTagName("solvable") if not solvables: raise CommandExecutionError( "No packages found matching '{}'".format(self.name) ) return sorted( { slv.getAttribute(self._attr_solvable_version) for slv in solvables if slv.getAttribute(self._attr_solvable_version) } ) def _get_scope_versions(self, pkg_versions): """ Get available difference between next possible matches. :return: """ get_in_versions = [] for p_version in pkg_versions: if fnmatch.fnmatch(p_version, self.version): get_in_versions.append(p_version) return get_in_versions def _set_version(self, version): """ Stash operator from the version, if any. :return: """ if not version: return exact_version = re.sub(r"[<>=+]*", "", version) self._op = version.replace(exact_version, "") or None if self._op and self._op not in self.Z_OP: raise CommandExecutionError( 'Zypper do not supports operator "{}".'.format(self._op) ) self.version = exact_version def _systemd_scope(): return salt.utils.systemd.has_scope(__context__) and __salt__["config.get"]( "systemd.scope", True ) def _clean_cache(): """ Clean cached results """ keys = [] for cache_name in ["pkg.list_pkgs", "pkg.list_provides"]: for contextkey in __context__: if contextkey.startswith(cache_name): keys.append(contextkey) for key in keys: __context__.pop(key, None) def list_upgrades(refresh=True, root=None, **kwargs): """ List all available package upgrades on this system refresh force a refresh if set to True (default). If set to False it depends on zypper if a refresh is executed. root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.list_upgrades """ if refresh: refresh_db(root) ret = dict() cmd = ["list-updates"] if "fromrepo" in kwargs: repos = kwargs["fromrepo"] if isinstance(repos, str): repos = [repos] for repo in repos: cmd.extend(["--repo", repo if isinstance(repo, str) else str(repo)]) log.debug("Targeting repos: %s", repos) for update_node in ( __zypper__(root=root).nolock.xml.call(*cmd).getElementsByTagName("update") ): if update_node.getAttribute("kind") == "package": ret[update_node.getAttribute("name")] = update_node.getAttribute("edition") return ret # Provide a list_updates function for those used to using zypper list-updates list_updates = salt.utils.functools.alias_function(list_upgrades, "list_updates") def info_installed(*names, **kwargs): """ Return the information of the named package(s), installed on the system. :param names: Names of the packages to get information about. :param attr: Comma-separated package attributes. If no 'attr' is specified, all available attributes returned. Valid attributes are: version, vendor, release, build_date, build_date_time_t, install_date, install_date_time_t, build_host, group, source_rpm, arch, epoch, size, license, signature, packager, url, summary, description. :param errors: Handle RPM field errors. If 'ignore' is chosen, then various mistakes are simply ignored and omitted from the texts or strings. If 'report' is chonen, then a field with a mistake is not returned, instead a 'N/A (broken)' (not available, broken) text is placed. Valid attributes are: ignore, report :param all_versions: Include information for all versions of the packages installed on the minion. :param root: Operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.info_installed <package1> salt '*' pkg.info_installed <package1> <package2> <package3> ... salt '*' pkg.info_installed <package1> <package2> <package3> all_versions=True salt '*' pkg.info_installed <package1> attr=version,vendor all_versions=True salt '*' pkg.info_installed <package1> <package2> <package3> ... attr=version,vendor salt '*' pkg.info_installed <package1> <package2> <package3> ... attr=version,vendor errors=ignore salt '*' pkg.info_installed <package1> <package2> <package3> ... attr=version,vendor errors=report """ all_versions = kwargs.get("all_versions", False) ret = dict() for pkg_name, pkgs_nfo in __salt__["lowpkg.info"](*names, **kwargs).items(): pkg_nfo = pkgs_nfo if all_versions else [pkgs_nfo] for _nfo in pkg_nfo: t_nfo = dict() # Translate dpkg-specific keys to a common structure for key, value in _nfo.items(): if key == "source_rpm": t_nfo["source"] = value else: t_nfo[key] = value if not all_versions: ret[pkg_name] = t_nfo else: ret.setdefault(pkg_name, []).append(t_nfo) return ret def info_available(*names, **kwargs): """ Return the information of the named package available for the system. refresh force a refresh if set to True (default). If set to False it depends on zypper if a refresh is executed or not. root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.info_available <package1> salt '*' pkg.info_available <package1> <package2> <package3> ... """ ret = {} if not names: return ret else: names = sorted(list(set(names))) root = kwargs.get("root", None) # Refresh db before extracting the latest package if kwargs.get("refresh", True): refresh_db(root) pkg_info = [] batch = names[:] batch_size = 200 # Run in batches while batch: pkg_info.extend( re.split( r"Information for package*", __zypper__(root=root).nolock.call( "info", "-t", "package", *batch[:batch_size] ), ) ) batch = batch[batch_size:] for pkg_data in pkg_info: nfo = {} for line in [data for data in pkg_data.split("\n") if ":" in data]: if line.startswith("-----"): continue kw = [data.strip() for data in line.split(":", 1)] if len(kw) == 2 and kw[1]: nfo[kw[0].lower()] = kw[1] if nfo.get("name"): name = nfo.pop("name") ret[name] = nfo if nfo.get("status"): nfo["status"] = nfo.get("status") if nfo.get("installed"): nfo["installed"] = nfo.get("installed").lower().startswith("yes") return ret def parse_arch(name): """ Parse name and architecture from the specified package name. CLI Example: .. code-block:: bash salt '*' pkg.parse_arch zsh.x86_64 """ _name, _arch = None, None try: _name, _arch = name.rsplit(PKG_ARCH_SEPARATOR, 1) except ValueError: pass if _arch not in salt.utils.pkg.rpm.ARCHES + ("noarch",): _name = name _arch = None return {"name": _name, "arch": _arch} def latest_version(*names, **kwargs): """ Return the latest version of the named package available for upgrade or installation. If more than one package name is specified, a dict of name/version pairs is returned. If the latest version of a given package is already installed, an empty dict will be returned for that package. refresh force a refresh if set to True (default). If set to False it depends on zypper if a refresh is executed or not. root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.latest_version <package name> salt '*' pkg.latest_version <package1> <package2> <package3> ... """ ret = dict() if not names: return ret names = sorted(list(set(names))) package_info = info_available(*names, **kwargs) for name in names: pkg_info = package_info.get(name, {}) status = pkg_info.get("status", "").lower() if status.find("not installed") > -1 or status.find("out-of-date") > -1: ret[name] = pkg_info.get("version") else: ret[name] = "" # Return a string if only one package name passed if len(names) == 1 and ret: return ret[names[0]] return ret # available_version is being deprecated available_version = salt.utils.functools.alias_function( latest_version, "available_version" ) def upgrade_available(name, **kwargs): """ Check whether or not an upgrade is available for a given package refresh force a refresh if set to True (default). If set to False it depends on zypper if a refresh is executed or not. root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.upgrade_available <package name> """ # The "not not" tactic is intended here as it forces the return to be False. return not not latest_version(name, **kwargs) # pylint: disable=C0113 def version(*names, **kwargs): """ Returns a string representing the package version or an empty dict if not installed. If more than one package name is specified, a dict of name/version pairs is returned. root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.version <package name> salt '*' pkg.version <package1> <package2> <package3> ... """ return __salt__["pkg_resource.version"](*names, **kwargs) or {} def version_cmp(ver1, ver2, ignore_epoch=False, **kwargs): """ .. versionadded:: 2015.5.4 Do a cmp-style comparison on two packages. Return -1 if ver1 < ver2, 0 if ver1 == ver2, and 1 if ver1 > ver2. Return None if there was a problem making the comparison. ignore_epoch : False Set to ``True`` to ignore the epoch when comparing versions .. versionadded:: 2015.8.10,2016.3.2 CLI Example: .. code-block:: bash salt '*' pkg.version_cmp '0.2-001' '0.2.0.1-002' """ return __salt__["lowpkg.version_cmp"](ver1, ver2, ignore_epoch=ignore_epoch) def _list_pkgs_from_context(versions_as_list, contextkey, attr): """ Use pkg list from __context__ """ return __salt__["pkg_resource.format_pkg_list"]( __context__[contextkey], versions_as_list, attr ) def list_pkgs(versions_as_list=False, root=None, includes=None, **kwargs): """ List the packages currently installed as a dict. By default, the dict contains versions as a comma separated string:: {'<package_name>': '<version>[,<version>...]'} versions_as_list: If set to true, the versions are provided as a list {'<package_name>': ['<version>', '<version>']} root: operate on a different root directory. includes: List of types of packages to include (package, patch, pattern, product) By default packages are always included attr: If a list of package attributes is specified, returned value will contain them in addition to version, eg.:: {'<package_name>': [{'version' : 'version', 'arch' : 'arch'}]} Valid attributes are: ``epoch``, ``version``, ``release``, ``arch``, ``install_date``, ``install_date_time_t``. If ``all`` is specified, all valid attributes will be returned. .. versionadded:: 2018.3.0 removed: not supported purge_desired: not supported CLI Example: .. code-block:: bash salt '*' pkg.list_pkgs salt '*' pkg.list_pkgs attr=version,arch salt '*' pkg.list_pkgs attr='["version", "arch"]' """ versions_as_list = salt.utils.data.is_true(versions_as_list) # not yet implemented or not applicable if any( [salt.utils.data.is_true(kwargs.get(x)) for x in ("removed", "purge_desired")] ): return {} attr = kwargs.get("attr") if attr is not None and attr != "all": attr = salt.utils.args.split_input(attr) includes = includes if includes else [] # Results can be different if a different root or a different # inclusion types are passed contextkey = "pkg.list_pkgs_{}_{}".format(root, includes) if contextkey in __context__ and kwargs.get("use_context", True): return _list_pkgs_from_context(versions_as_list, contextkey, attr) ret = {} cmd = ["rpm"] if root: cmd.extend(["--root", root]) cmd.extend( [ "-qa", "--queryformat", salt.utils.pkg.rpm.QUERYFORMAT.replace("%{REPOID}", "(none)") + "\n", ] ) output = __salt__["cmd.run"](cmd, python_shell=False, output_loglevel="trace") for line in output.splitlines(): pkginfo = salt.utils.pkg.rpm.parse_pkginfo(line, osarch=__grains__["osarch"]) if pkginfo: # see rpm version string rules available at https://goo.gl/UGKPNd pkgver = pkginfo.version epoch = None release = None if ":" in pkgver: epoch, pkgver = pkgver.split(":", 1) if "-" in pkgver: pkgver, release = pkgver.split("-", 1) all_attr = { "epoch": epoch, "version": pkgver, "release": release, "arch": pkginfo.arch, "install_date": pkginfo.install_date, "install_date_time_t": pkginfo.install_date_time_t, } __salt__["pkg_resource.add_pkg"](ret, pkginfo.name, all_attr) _ret = {} for pkgname in ret: # Filter out GPG public keys packages if pkgname.startswith("gpg-pubkey"): continue _ret[pkgname] = sorted(ret[pkgname], key=lambda d: d["version"]) for include in includes: if include == "product": products = list_products(all=False, root=root) for product in products: extended_name = "{}:{}".format(include, product["name"]) _ret[extended_name] = [ { "epoch": product["epoch"], "version": product["version"], "release": product["release"], "arch": product["arch"], "install_date": None, "install_date_time_t": None, } ] if include in ("pattern", "patch"): if include == "pattern": elements = list_installed_patterns(root=root) elif include == "patch": elements = list_installed_patches(root=root) else: elements = [] for element in elements: extended_name = "{}:{}".format(include, element) info = info_available(extended_name, refresh=False, root=root) _ret[extended_name] = [ { "epoch": None, "version": info[element]["version"], "release": None, "arch": info[element]["arch"], "install_date": None, "install_date_time_t": None, } ] __context__[contextkey] = _ret return __salt__["pkg_resource.format_pkg_list"]( __context__[contextkey], versions_as_list, attr ) def list_repo_pkgs(*args, **kwargs): """ .. versionadded:: 2017.7.5,2018.3.1 Returns all available packages. Optionally, package names (and name globs) can be passed and the results will be filtered to packages matching those names. This is recommended as it speeds up the function considerably. This function can be helpful in discovering the version or repo to specify in a :mod:`pkg.installed <salt.states.pkg.installed>` state. The return data will be a dictionary mapping package names to a list of version numbers, ordered from newest to oldest. If ``byrepo`` is set to ``True``, then the return dictionary will contain repository names at the top level, and each repository will map packages to lists of version numbers. For example: .. code-block:: python # With byrepo=False (default) { 'bash': ['4.3-83.3.1', '4.3-82.6'], 'vim': ['7.4.326-12.1'] } { 'OSS': { 'bash': ['4.3-82.6'], 'vim': ['7.4.326-12.1'] }, 'OSS Update': { 'bash': ['4.3-83.3.1'] } } fromrepo : None Only include results from the specified repo(s). Multiple repos can be specified, comma-separated. byrepo : False When ``True``, the return data for each package will be organized by repository. root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.list_repo_pkgs salt '*' pkg.list_repo_pkgs foo bar baz salt '*' pkg.list_repo_pkgs 'python2-*' byrepo=True salt '*' pkg.list_repo_pkgs 'python2-*' fromrepo='OSS Updates' """ byrepo = kwargs.pop("byrepo", False) fromrepo = kwargs.pop("fromrepo", "") or "" ret = {} targets = [arg if isinstance(arg, str) else str(arg) for arg in args] def _is_match(pkgname): """ When package names are passed to a zypper search, they will be matched anywhere in the package name. This makes sure that only exact or fnmatch matches are identified. """ if not args: # No package names passed, everyone's a winner! return True for target in targets: if fnmatch.fnmatch(pkgname, target): return True return False root = kwargs.get("root") or None for node in ( __zypper__(root=root) .xml.call("se", "-s", *targets) .getElementsByTagName("solvable") ): pkginfo = dict(node.attributes.items()) try: if pkginfo["kind"] != "package": continue reponame = pkginfo["repository"] if fromrepo and reponame != fromrepo: continue pkgname = pkginfo["name"] pkgversion = pkginfo["edition"] except KeyError: continue else: if _is_match(pkgname): repo_dict = ret.setdefault(reponame, {}) version_list = repo_dict.setdefault(pkgname, set()) version_list.add(pkgversion) if byrepo: for reponame in ret: # Sort versions newest to oldest for pkgname in ret[reponame]: sorted_versions = sorted( (LooseVersion(x) for x in ret[reponame][pkgname]), reverse=True ) ret[reponame][pkgname] = [x.vstring for x in sorted_versions] return ret else: byrepo_ret = {} for reponame in ret: for pkgname in ret[reponame]: byrepo_ret.setdefault(pkgname, []).extend(ret[reponame][pkgname]) for pkgname in byrepo_ret: sorted_versions = sorted( (LooseVersion(x) for x in byrepo_ret[pkgname]), reverse=True ) byrepo_ret[pkgname] = [x.vstring for x in sorted_versions] return byrepo_ret def _get_configured_repos(root=None): """ Get all the info about repositories from the configurations. """ repos = os.path.join(root, os.path.relpath(REPOS, os.path.sep)) if root else REPOS repos_cfg = configparser.ConfigParser() if os.path.exists(repos): repos_cfg.read( [ repos + "/" + fname for fname in os.listdir(repos) if fname.endswith(".repo") ] ) else: log.warning("Repositories not found in %s", repos) return repos_cfg def _get_repo_info(alias, repos_cfg=None, root=None): """ Get one repo meta-data. """ try: meta = dict((repos_cfg or _get_configured_repos(root=root)).items(alias)) meta["alias"] = alias for key, val in meta.items(): if val in ["0", "1"]: meta[key] = int(meta[key]) == 1 elif val == "NONE": meta[key] = None return meta except (ValueError, configparser.NoSectionError): return {} def get_repo(repo, root=None, **kwargs): # pylint: disable=unused-argument """ Display a repo. root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.get_repo alias """ return _get_repo_info(repo, root=root) def list_repos(root=None, **kwargs): """ Lists all repos. root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.list_repos """ repos_cfg = _get_configured_repos(root=root) all_repos = {} for alias in repos_cfg.sections(): all_repos[alias] = _get_repo_info(alias, repos_cfg=repos_cfg, root=root) return all_repos def del_repo(repo, root=None): """ Delete a repo. root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.del_repo alias """ repos_cfg = _get_configured_repos(root=root) for alias in repos_cfg.sections(): if alias == repo: doc = __zypper__(root=root).xml.call( "rr", "--loose-auth", "--loose-query", alias ) msg = doc.getElementsByTagName("message") if doc.getElementsByTagName("progress") and msg: return { repo: True, "message": msg[0].childNodes[0].nodeValue, } raise CommandExecutionError("Repository '{}' not found.".format(repo)) def mod_repo(repo, **kwargs): """ Modify one or more values for a repo. If the repo does not exist, it will be created, so long as the following values are specified: repo or alias alias by which Zypper refers to the repo url, mirrorlist or baseurl the URL for Zypper to reference enabled Enable or disable (True or False) repository, but do not remove if disabled. name This is used as the descriptive name value in the repo file. refresh Enable or disable (True or False) auto-refresh of the repository. cache Enable or disable (True or False) RPM files caching. gpgcheck Enable or disable (True or False) GPG check for this repository. gpgautoimport : False If set to True, automatically trust and import public GPG key for the repository. root operate on a different root directory. Key/Value pairs may also be removed from a repo's configuration by setting a key to a blank value. Bear in mind that a name cannot be deleted, and a URL can only be deleted if a ``mirrorlist`` is specified (or vice versa). CLI Examples: .. code-block:: bash salt '*' pkg.mod_repo alias alias=new_alias salt '*' pkg.mod_repo alias url= mirrorlist=http://host.com/ """ root = kwargs.get("root") or None repos_cfg = _get_configured_repos(root=root) added = False # An attempt to add new one? if repo not in repos_cfg.sections(): url = kwargs.get("url", kwargs.get("mirrorlist", kwargs.get("baseurl"))) if not url: raise CommandExecutionError( "Repository '{}' not found, and neither 'baseurl' nor " "'mirrorlist' was specified".format(repo) ) if not urllib.parse.urlparse(url).scheme: raise CommandExecutionError( "Repository '{}' not found and URL for baseurl/mirrorlist " "is malformed".format(repo) ) # Is there already such repo under different alias? for alias in repos_cfg.sections(): repo_meta = _get_repo_info(alias, repos_cfg=repos_cfg, root=root) # Complete user URL, in case it is not new_url = urllib.parse.urlparse(url) if not new_url.path: new_url = urllib.parse.urlparse.ParseResult( scheme=new_url.scheme, # pylint: disable=E1123 netloc=new_url.netloc, path="/", params=new_url.params, query=new_url.query, fragment=new_url.fragment, ) base_url = urllib.parse.urlparse(repo_meta["baseurl"]) if new_url == base_url: raise CommandExecutionError( "Repository '{}' already exists as '{}'.".format(repo, alias) ) # Add new repo __zypper__(root=root).xml.call("ar", url, repo) # Verify the repository has been added repos_cfg = _get_configured_repos(root=root) if repo not in repos_cfg.sections(): raise CommandExecutionError( "Failed add new repository '{}' for unspecified reason. " "Please check zypper logs.".format(repo) ) added = True repo_info = _get_repo_info(repo, root=root) if ( not added and "baseurl" in kwargs and not (kwargs["baseurl"] == repo_info["baseurl"]) ): # Note: zypper does not support changing the baseurl # we need to remove the repository and add it again with the new baseurl repo_info.update(kwargs) repo_info.setdefault("cache", False) del_repo(repo, root=root) return mod_repo(repo, root=root, **repo_info) # Modify added or existing repo according to the options cmd_opt = [] global_cmd_opt = [] call_refresh = False if "enabled" in kwargs: cmd_opt.append(kwargs["enabled"] and "--enable" or "--disable") if "refresh" in kwargs: cmd_opt.append(kwargs["refresh"] and "--refresh" or "--no-refresh") if "cache" in kwargs: cmd_opt.append(kwargs["cache"] and "--keep-packages" or "--no-keep-packages") if "gpgcheck" in kwargs: cmd_opt.append(kwargs["gpgcheck"] and "--gpgcheck" or "--no-gpgcheck") if "priority" in kwargs: cmd_opt.append("--priority={}".format(kwargs.get("priority", DEFAULT_PRIORITY))) if "humanname" in kwargs: salt.utils.versions.warn_until( 3009, "Passing 'humanname' to 'mod_repo' is deprecated, slated " "for removal in {version}. Please use 'name' instead.", ) cmd_opt.append("--name='{}'".format(kwargs.get("humanname"))) if "name" in kwargs: cmd_opt.append("--name") cmd_opt.append(kwargs.get("name")) if kwargs.get("gpgautoimport") is True: global_cmd_opt.append("--gpg-auto-import-keys") call_refresh = True if cmd_opt: cmd_opt = global_cmd_opt + ["mr"] + cmd_opt + [repo] __zypper__(root=root).refreshable.xml.call(*cmd_opt) comment = None if call_refresh: # when used with "zypper ar --refresh" or "zypper mr --refresh" # --gpg-auto-import-keys is not doing anything # so we need to specifically refresh here with --gpg-auto-import-keys refresh_opts = global_cmd_opt + ["refresh"] + [repo] __zypper__(root=root).xml.call(*refresh_opts) elif not added and not cmd_opt: comment = "Specified arguments did not result in modification of repo" repo = get_repo(repo, root=root) if comment: repo["comment"] = comment return repo def refresh_db(force=None, root=None): """ Trigger a repository refresh by calling ``zypper refresh``. Refresh will run with ``--force`` if the "force=True" flag is passed on the CLI or ``refreshdb_force`` is set to ``true`` in the pillar. The CLI option overrides the pillar setting. It will return a dict:: {'<database name>': Bool} root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.refresh_db [force=true|false] Pillar Example: .. code-block:: yaml zypper: refreshdb_force: false """ # Remove rtag file to keep multiple refreshes from happening in pkg states salt.utils.pkg.clear_rtag(__opts__) ret = {} refresh_opts = ["refresh"] if force is None: force = __pillar__.get("zypper", {}).get("refreshdb_force", True) if force: refresh_opts.append("--force") out = __zypper__(root=root).refreshable.call(*refresh_opts) for line in out.splitlines(): if not line: continue if line.strip().startswith("Repository") and "'" in line: try: key = line.split("'")[1].strip() if "is up to date" in line: ret[key] = False except IndexError: continue elif line.strip().startswith("Building") and "'" in line: key = line.split("'")[1].strip() if "done" in line: ret[key] = True return ret def _find_types(pkgs): """Form a package names list, find prefixes of packages types.""" return sorted({pkg.split(":", 1)[0] for pkg in pkgs if len(pkg.split(":", 1)) == 2}) def install( name=None, refresh=False, fromrepo=None, pkgs=None, sources=None, downloadonly=None, skip_verify=False, version=None, ignore_repo_failure=False, no_recommends=False, root=None, **kwargs ): """ .. versionchanged:: 2015.8.12,2016.3.3,2016.11.0 On minions running systemd>=205, `systemd-run(1)`_ is now used to isolate commands which modify installed packages from the ``salt-minion`` daemon's control group. This is done to keep systemd from killing any zypper commands spawned by Salt when the ``salt-minion`` service is restarted. (see ``KillMode`` in the `systemd.kill(5)`_ manpage for more information). If desired, usage of `systemd-run(1)`_ can be suppressed by setting a :mod:`config option <salt.modules.config.get>` called ``systemd.scope``, with a value of ``False`` (no quotes). .. _`systemd-run(1)`: https://www.freedesktop.org/software/systemd/man/systemd-run.html .. _`systemd.kill(5)`: https://www.freedesktop.org/software/systemd/man/systemd.kill.html Install the passed package(s), add refresh=True to force a 'zypper refresh' before package is installed. name The name of the package to be installed. Note that this parameter is ignored if either ``pkgs`` or ``sources`` is passed. Additionally, please note that this option can only be used to install packages from a software repository. To install a package file manually, use the ``sources`` option. CLI Example: .. code-block:: bash salt '*' pkg.install <package name> refresh force a refresh if set to True. If set to False (default) it depends on zypper if a refresh is executed. fromrepo Specify a package repository to install from. downloadonly Only download the packages, do not install. skip_verify Skip the GPG verification check (e.g., ``--no-gpg-checks``) version Can be either a version number, or the combination of a comparison operator (<, >, <=, >=, =) and a version number (ex. '>1.2.3-4'). This parameter is ignored if ``pkgs`` or ``sources`` is passed. resolve_capabilities If this option is set to True zypper will take capabilities into account. In this case names which are just provided by a package will get installed. Default is False. Multiple Package Installation Options: pkgs A list of packages to install from a software repository. Must be passed as a python list. A specific version number can be specified by using a single-element dict representing the package and its version. As with the ``version`` parameter above, comparison operators can be used to target a specific version of a package. CLI Examples: .. code-block:: bash salt '*' pkg.install pkgs='["foo", "bar"]' salt '*' pkg.install pkgs='["foo", {"bar": "1.2.3-4"}]' salt '*' pkg.install pkgs='["foo", {"bar": "<1.2.3-4"}]' sources A list of RPM packages to install. Must be passed as a list of dicts, with the keys being package names, and the values being the source URI or local path to the package. CLI Example: .. code-block:: bash salt '*' pkg.install sources='[{"foo": "salt://foo.rpm"},{"bar": "salt://bar.rpm"}]' ignore_repo_failure Zypper returns error code 106 if one of the repositories are not available for various reasons. In case to set strict check, this parameter needs to be set to True. Default: False. no_recommends Do not install recommended packages, only required ones. root operate on a different root directory. diff_attr: If a list of package attributes is specified, returned value will contain them, eg.:: {'<package>': { 'old': { 'version': '<old-version>', 'arch': '<old-arch>'}, 'new': { 'version': '<new-version>', 'arch': '<new-arch>'}}} Valid attributes are: ``epoch``, ``version``, ``release``, ``arch``, ``install_date``, ``install_date_time_t``. If ``all`` is specified, all valid attributes will be returned. .. versionadded:: 2018.3.0 Returns a dict containing the new package names and versions:: {'<package>': {'old': '<old-version>', 'new': '<new-version>'}} If an attribute list is specified in ``diff_attr``, the dict will also contain any specified attribute, eg.:: {'<package>': { 'old': { 'version': '<old-version>', 'arch': '<old-arch>'}, 'new': { 'version': '<new-version>', 'arch': '<new-arch>'}}} """ if refresh: refresh_db(root) try: pkg_params, pkg_type = __salt__["pkg_resource.parse_targets"]( name, pkgs, sources, **kwargs ) except MinionError as exc: raise CommandExecutionError(exc) if not pkg_params: return {} version_num = Wildcard(__zypper__(root=root))(name, version) if version_num: if pkgs is None and sources is None: # Allow "version" to work for single package target pkg_params = {name: version_num} else: log.warning( '"version" parameter will be ignored for multiple package targets' ) if pkg_type == "repository": targets = [] for param, version_num in pkg_params.items(): if version_num is None: log.debug("targeting package: %s", param) targets.append(param) else: prefix, verstr = salt.utils.pkg.split_comparison(version_num) if not prefix: prefix = "=" target = "{}{}{}".format(param, prefix, verstr) log.debug("targeting package: %s", target) targets.append(target) elif pkg_type == "advisory": targets = [] cur_patches = list_patches(root=root) for advisory_id in pkg_params: if advisory_id not in cur_patches: raise CommandExecutionError( 'Advisory id "{}" not found'.format(advisory_id) ) else: # If we add here the `patch:` prefix, the # `_find_types` helper will take the patches into the # list of packages. Usually this is the correct thing # to do, but we can break software the depends on the # old behaviour. targets.append(advisory_id) else: targets = pkg_params diff_attr = kwargs.get("diff_attr") includes = _find_types(targets) old = ( list_pkgs(attr=diff_attr, root=root, includes=includes) if not downloadonly else list_downloaded(root) ) downgrades = [] if fromrepo: fromrepoopt = ["--force", "--force-resolution", "--from", fromrepo] log.info("Targeting repo '%s'", fromrepo) else: fromrepoopt = "" cmd_install = ["install", "--auto-agree-with-licenses"] cmd_install.append( kwargs.get("resolve_capabilities") and "--capability" or "--name" ) if not refresh: cmd_install.insert(0, "--no-refresh") if skip_verify: cmd_install.insert(0, "--no-gpg-checks") if downloadonly: cmd_install.append("--download-only") if fromrepo: cmd_install.extend(fromrepoopt) if no_recommends: cmd_install.append("--no-recommends") errors = [] # If the type is 'advisory', we manually add the 'patch:' # prefix. This kind of package will not appear in pkg_list in this # way. # # Note that this enable a different mechanism to install a patch; # if the name of the package is already prefixed with 'patch:' we # can avoid listing them in the `advisory_ids` field. if pkg_type == "advisory": targets = ["patch:{}".format(t) for t in targets] # Split the targets into batches of 500 packages each, so that # the maximal length of the command line is not broken systemd_scope = _systemd_scope() while targets: cmd = cmd_install + targets[:500] targets = targets[500:] for line in ( __zypper__( no_repo_failure=ignore_repo_failure, systemd_scope=systemd_scope, root=root, ) .call(*cmd) .splitlines() ): match = re.match( r"^The selected package '([^']+)'.+has lower version", line ) if match: downgrades.append(match.group(1)) while downgrades: cmd = cmd_install + ["--force"] + downgrades[:500] downgrades = downgrades[500:] __zypper__(no_repo_failure=ignore_repo_failure, root=root).call(*cmd) _clean_cache() new = ( list_pkgs(attr=diff_attr, root=root, includes=includes) if not downloadonly else list_downloaded(root) ) ret = salt.utils.data.compare_dicts(old, new) # If something else from packages are included in the search, # better clean the cache. if includes: _clean_cache() if errors: raise CommandExecutionError( "Problem encountered {} package(s)".format( "downloading" if downloadonly else "installing" ), info={"errors": errors, "changes": ret}, ) return ret def upgrade( name=None, pkgs=None, refresh=True, dryrun=False, dist_upgrade=False, fromrepo=None, novendorchange=False, skip_verify=False, no_recommends=False, root=None, diff_attr=None, **kwargs ): # pylint: disable=unused-argument """ .. versionchanged:: 2015.8.12,2016.3.3,2016.11.0 On minions running systemd>=205, `systemd-run(1)`_ is now used to isolate commands which modify installed packages from the ``salt-minion`` daemon's control group. This is done to keep systemd from killing any zypper commands spawned by Salt when the ``salt-minion`` service is restarted. (see ``KillMode`` in the `systemd.kill(5)`_ manpage for more information). If desired, usage of `systemd-run(1)`_ can be suppressed by setting a :mod:`config option <salt.modules.config.get>` called ``systemd.scope``, with a value of ``False`` (no quotes). .. _`systemd-run(1)`: https://www.freedesktop.org/software/systemd/man/systemd-run.html .. _`systemd.kill(5)`: https://www.freedesktop.org/software/systemd/man/systemd.kill.html Run a full system upgrade, a zypper upgrade name The name of the package to be installed. Note that this parameter is ignored if ``pkgs`` is passed or if ``dryrun`` is set to True. CLI Example: .. code-block:: bash salt '*' pkg.install name=<package name> pkgs A list of packages to install from a software repository. Must be passed as a python list. Note that this parameter is ignored if ``dryrun`` is set to True. CLI Examples: .. code-block:: bash salt '*' pkg.install pkgs='["foo", "bar"]' refresh force a refresh if set to True (default). If set to False it depends on zypper if a refresh is executed. dryrun If set to True, it creates a debug solver log file and then perform a dry-run upgrade (no changes are made). Default: False dist_upgrade Perform a system dist-upgrade. Default: False fromrepo Specify a list of package repositories to upgrade from. Default: None novendorchange If set to True, no allow vendor changes. Default: False skip_verify Skip the GPG verification check (e.g., ``--no-gpg-checks``) no_recommends Do not install recommended packages, only required ones. root Operate on a different root directory. diff_attr: If a list of package attributes is specified, returned value will contain them, eg.:: {'<package>': { 'old': { 'version': '<old-version>', 'arch': '<old-arch>'}, 'new': { 'version': '<new-version>', 'arch': '<new-arch>'}}} Valid attributes are: ``epoch``, ``version``, ``release``, ``arch``, ``install_date``, ``install_date_time_t``. If ``all`` is specified, all valid attributes will be returned. .. versionadded:: 3006.0 Returns a dictionary containing the changes: .. code-block:: python {'<package>': {'old': '<old-version>', 'new': '<new-version>'}} If an attribute list is specified in ``diff_attr``, the dict will also contain any specified attribute, eg.:: .. code-block:: python {'<package>': { 'old': { 'version': '<old-version>', 'arch': '<old-arch>'}, 'new': { 'version': '<new-version>', 'arch': '<new-arch>'}}} CLI Example: .. code-block:: bash salt '*' pkg.upgrade salt '*' pkg.upgrade name=mypackage salt '*' pkg.upgrade pkgs='["package1", "package2"]' salt '*' pkg.upgrade dist_upgrade=True fromrepo='["MyRepoName"]' novendorchange=True salt '*' pkg.upgrade dist_upgrade=True dryrun=True """ cmd_update = (["dist-upgrade"] if dist_upgrade else ["update"]) + [ "--auto-agree-with-licenses" ] if skip_verify: # The '--no-gpg-checks' needs to be placed before the Zypper command. cmd_update.insert(0, "--no-gpg-checks") if refresh: refresh_db(root) if dryrun: cmd_update.append("--dry-run") if fromrepo: if isinstance(fromrepo, str): fromrepo = [fromrepo] for repo in fromrepo: cmd_update.extend(["--from" if dist_upgrade else "--repo", repo]) log.info("Targeting repos: %s", fromrepo) if dist_upgrade: if novendorchange: # TODO: Grains validation should be moved to Zypper class if __grains__["osrelease_info"][0] > 11: cmd_update.append("--no-allow-vendor-change") log.info("Disabling vendor changes") else: log.warning( "Disabling vendor changes is not supported on this Zypper version" ) if no_recommends: cmd_update.append("--no-recommends") log.info("Disabling recommendations") if dryrun: # Creates a solver test case for debugging. log.info("Executing debugsolver and performing a dry-run dist-upgrade") __zypper__(systemd_scope=_systemd_scope(), root=root).noraise.call( *cmd_update + ["--debug-solver"] ) else: if name or pkgs: try: (pkg_params, _) = __salt__["pkg_resource.parse_targets"]( name=name, pkgs=pkgs, sources=None, **kwargs ) if pkg_params: cmd_update.extend(pkg_params.keys()) except MinionError as exc: raise CommandExecutionError(exc) old = list_pkgs(root=root, attr=diff_attr) __zypper__(systemd_scope=_systemd_scope(), root=root).noraise.call(*cmd_update) _clean_cache() new = list_pkgs(root=root, attr=diff_attr) ret = salt.utils.data.compare_dicts(old, new) if __zypper__.exit_code not in __zypper__.SUCCESS_EXIT_CODES: result = { "retcode": __zypper__.exit_code, "stdout": __zypper__.stdout, "stderr": __zypper__.stderr, "pid": __zypper__.pid, } raise CommandExecutionError( "Problem encountered upgrading packages", info={"changes": ret, "result": result}, ) if dryrun: ret = (__zypper__.stdout + os.linesep + __zypper__.stderr).strip() return ret def _uninstall(name=None, pkgs=None, root=None): """ Remove and purge do identical things but with different Zypper commands, this function performs the common logic. """ try: pkg_params = __salt__["pkg_resource.parse_targets"](name, pkgs)[0] except MinionError as exc: raise CommandExecutionError(exc) includes = _find_types(pkg_params.keys()) old = list_pkgs(root=root, includes=includes) targets = [] for target in pkg_params: # Check if package version set to be removed is actually installed: # old[target] contains a comma-separated list of installed versions if target in old and pkg_params[target] in old[target].split(","): targets.append(target + "-" + pkg_params[target]) elif target in old and not pkg_params[target]: targets.append(target) if not targets: return {} systemd_scope = _systemd_scope() errors = [] while targets: __zypper__(systemd_scope=systemd_scope, root=root).call( "remove", *targets[:500] ) targets = targets[500:] _clean_cache() new = list_pkgs(root=root, includes=includes) ret = salt.utils.data.compare_dicts(old, new) if errors: raise CommandExecutionError( "Problem encountered removing package(s)", info={"errors": errors, "changes": ret}, ) return ret def normalize_name(name): """ Strips the architecture from the specified package name, if necessary. Circumstances where this would be done include: * If the arch is 32 bit and the package name ends in a 32-bit arch. * If the arch matches the OS arch, or is ``noarch``. CLI Example: .. code-block:: bash salt '*' pkg.normalize_name zsh.x86_64 """ try: arch = name.rsplit(".", 1)[-1] if arch not in salt.utils.pkg.rpm.ARCHES + ("noarch",): return name except ValueError: return name if arch in (__grains__["osarch"], "noarch") or salt.utils.pkg.rpm.check_32( arch, osarch=__grains__["osarch"] ): return name[: -(len(arch) + 1)] return name def remove( name=None, pkgs=None, root=None, **kwargs ): # pylint: disable=unused-argument """ .. versionchanged:: 2015.8.12,2016.3.3,2016.11.0 On minions running systemd>=205, `systemd-run(1)`_ is now used to isolate commands which modify installed packages from the ``salt-minion`` daemon's control group. This is done to keep systemd from killing any zypper commands spawned by Salt when the ``salt-minion`` service is restarted. (see ``KillMode`` in the `systemd.kill(5)`_ manpage for more information). If desired, usage of `systemd-run(1)`_ can be suppressed by setting a :mod:`config option <salt.modules.config.get>` called ``systemd.scope``, with a value of ``False`` (no quotes). .. _`systemd-run(1)`: https://www.freedesktop.org/software/systemd/man/systemd-run.html .. _`systemd.kill(5)`: https://www.freedesktop.org/software/systemd/man/systemd.kill.html Remove packages with ``zypper -n remove`` name The name of the package to be deleted. Multiple Package Options: pkgs A list of packages to delete. Must be passed as a python list. The ``name`` parameter will be ignored if this option is passed. root Operate on a different root directory. .. versionadded:: 0.16.0 Returns a dict containing the changes. CLI Example: .. code-block:: bash salt '*' pkg.remove <package name> salt '*' pkg.remove <package1>,<package2>,<package3> salt '*' pkg.remove pkgs='["foo", "bar"]' """ return _uninstall(name=name, pkgs=pkgs, root=root) def purge(name=None, pkgs=None, root=None, **kwargs): # pylint: disable=unused-argument """ .. versionchanged:: 2015.8.12,2016.3.3,2016.11.0 On minions running systemd>=205, `systemd-run(1)`_ is now used to isolate commands which modify installed packages from the ``salt-minion`` daemon's control group. This is done to keep systemd from killing any zypper commands spawned by Salt when the ``salt-minion`` service is restarted. (see ``KillMode`` in the `systemd.kill(5)`_ manpage for more information). If desired, usage of `systemd-run(1)`_ can be suppressed by setting a :mod:`config option <salt.modules.config.get>` called ``systemd.scope``, with a value of ``False`` (no quotes). .. _`systemd-run(1)`: https://www.freedesktop.org/software/systemd/man/systemd-run.html .. _`systemd.kill(5)`: https://www.freedesktop.org/software/systemd/man/systemd.kill.html Recursively remove a package and all dependencies which were installed with it, this will call a ``zypper -n remove -u`` name The name of the package to be deleted. Multiple Package Options: pkgs A list of packages to delete. Must be passed as a python list. The ``name`` parameter will be ignored if this option is passed. root Operate on a different root directory. .. versionadded:: 0.16.0 Returns a dict containing the changes. CLI Example: .. code-block:: bash salt '*' pkg.purge <package name> salt '*' pkg.purge <package1>,<package2>,<package3> salt '*' pkg.purge pkgs='["foo", "bar"]' """ return _uninstall(name=name, pkgs=pkgs, root=root) def list_holds(pattern=None, full=True, root=None, **kwargs): """ .. versionadded:: 3005 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, ) 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. root operate on a different root directory. Return a dict containing the locked package with attributes:: {'<package>': {'case_sensitive': '<case_sensitive>', 'match_type': '<match_type>' 'type': '<type>'}} CLI Example: .. code-block:: bash salt '*' pkg.list_locks """ locks = {} _locks = os.path.join(root, os.path.relpath(LOCKS, os.path.sep)) if root else LOCKS try: with salt.utils.files.fopen(_locks) as fhr: items = salt.utils.stringutils.to_unicode(fhr.read()).split("\n\n") for meta in [item.split("\n") for item in items]: lock = {} for element in [el for el in meta if el]: if ":" in element: lock.update( dict([tuple(i.strip() for i in element.split(":", 1))]) ) if lock.get("solvable_name"): locks[lock.pop("solvable_name")] = lock except OSError: pass except Exception: # pylint: disable=broad-except log.warning("Detected a problem when accessing %s", _locks) return locks def clean_locks(root=None): """ Remove unused locks that do not currently (with regard to repositories used) lock any package. root Operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.clean_locks """ LCK = "removed" out = {LCK: 0} locks = os.path.join(root, os.path.relpath(LOCKS, os.path.sep)) if root else LOCKS if not os.path.exists(locks): return out for node in __zypper__(root=root).xml.call("cl").getElementsByTagName("message"): text = node.childNodes[0].nodeValue.lower() if text.startswith(LCK): out[LCK] = text.split(" ")[1] break return out def unhold(name=None, pkgs=None, root=None, **kwargs): """ .. versionadded:: 3003 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. CLI Example: .. code-block:: bash salt '*' pkg.unhold <package name> salt '*' pkg.unhold <package1>,<package2>,<package3> salt '*' pkg.unhold pkgs='["foo", "bar"]' """ ret = {} if not name and not pkgs: raise CommandExecutionError("Name or packages must be specified.") targets = [] if pkgs: targets.extend(pkgs) else: targets.append(name) locks = list_locks(root=root) removed = [] 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: ret[target]["comment"] = "Package {} was already unheld.".format(target) if removed: __zypper__(root=root).call("rl", *removed) return ret def hold(name=None, pkgs=None, root=None, **kwargs): """ .. versionadded:: 3003 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. CLI Example: .. code-block:: bash salt '*' pkg.hold <package name> salt '*' pkg.hold <package1>,<package2>,<package3> salt '*' pkg.hold pkgs='["foo", "bar"]' """ ret = {} if not name and not pkgs: raise CommandExecutionError("Name or packages must be specified.") targets = [] if pkgs: targets.extend(pkgs) else: targets.append(name) locks = list_locks(root=root) added = [] 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[target]["comment"] = "Package {} is already set to be held.".format( target ) if added: __zypper__(root=root).call("al", *added) return ret def verify(*names, **kwargs): """ Runs an rpm -Va on a system, and returns the results in a dict Files with an attribute of config, doc, ghost, license or readme in the package header can be ignored using the ``ignore_types`` keyword argument. The root parameter can also be passed via the keyword argument. CLI Example: .. code-block:: bash salt '*' pkg.verify salt '*' pkg.verify httpd salt '*' pkg.verify 'httpd postfix' salt '*' pkg.verify 'httpd postfix' ignore_types=['config','doc'] """ return __salt__["lowpkg.verify"](*names, **kwargs) def file_list(*packages, **kwargs): """ List the files that belong to a package. Not specifying any packages will return a list of *every* file on the system's rpm database (not generally recommended). The root parameter can also be passed via the keyword argument. CLI Examples: .. code-block:: bash salt '*' pkg.file_list httpd salt '*' pkg.file_list httpd postfix salt '*' pkg.file_list """ return __salt__["lowpkg.file_list"](*packages, **kwargs) def file_dict(*packages, **kwargs): """ List the files that belong to a package, grouped by package. Not specifying any packages will return a list of *every* file on the system's rpm database (not generally recommended). The root parameter can also be passed via the keyword argument. CLI Examples: .. code-block:: bash salt '*' pkg.file_list httpd salt '*' pkg.file_list httpd postfix salt '*' pkg.file_list """ return __salt__["lowpkg.file_dict"](*packages, **kwargs) def modified(*packages, **flags): """ List the modified files that belong to a package. Not specifying any packages will return a list of _all_ modified files on the system's RPM database. .. versionadded:: 2015.5.0 Filtering by flags (True or False): size Include only files where size changed. mode Include only files which file's mode has been changed. checksum Include only files which MD5 checksum has been changed. device Include only files which major and minor numbers has been changed. symlink Include only files which are symbolic link contents. owner Include only files where owner has been changed. group Include only files where group has been changed. time Include only files where modification time of the file has been changed. capabilities Include only files where capabilities differ or not. Note: supported only on newer RPM versions. root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.modified salt '*' pkg.modified httpd salt '*' pkg.modified httpd postfix salt '*' pkg.modified httpd owner=True group=False """ return __salt__["lowpkg.modified"](*packages, **flags) def owner(*paths, **kwargs): """ Return the name of the package that owns the file. Multiple file paths can be passed. If a single path is passed, a string will be returned, and if multiple paths are passed, a dictionary of file/package name pairs will be returned. If the file is not owned by a package, or is not present on the minion, then an empty string will be returned for that path. The root parameter can also be passed via the keyword argument. CLI Examples: .. code-block:: bash salt '*' pkg.owner /usr/bin/apachectl salt '*' pkg.owner /usr/bin/apachectl /etc/httpd/conf/httpd.conf """ return __salt__["lowpkg.owner"](*paths, **kwargs) def _get_visible_patterns(root=None): """Get all available patterns in the repo that are visible.""" patterns = {} search_patterns = __zypper__(root=root).nolock.xml.call("se", "-t", "pattern") for element in search_patterns.getElementsByTagName("solvable"): installed = element.getAttribute("status") == "installed" patterns[element.getAttribute("name")] = { "installed": installed, "summary": element.getAttribute("summary"), } return patterns def _get_installed_patterns(root=None): """ List all installed patterns. """ # Some patterns are non visible (`pattern-visible()` capability is # not set), so they cannot be found via a normal `zypper se -t # pattern`. # # Also patterns are not directly searchable in the local rpmdb. # # The proposed solution is, first search all the packages that # containst the 'pattern()' capability, and deduce the name of the # pattern from this capability. # # For example: # # 'pattern() = base' -> 'base' # 'pattern() = microos_defaults' -> 'microos_defaults' def _pattern_name(capability): """Return from a suitable capability the pattern name.""" return capability.split("=")[-1].strip() cmd = ["rpm"] if root: cmd.extend(["--root", root]) cmd.extend(["-q", "--provides", "--whatprovides", "pattern()"]) # If no `pattern()`s are found, RPM returns `1`, but for us is not # a real error. output = __salt__["cmd.run"](cmd, ignore_retcode=True) # On <= SLE12SP4 we have patterns that have multiple names (alias) # and that are duplicated. The alias start with ".", so we filter # them. installed_patterns = { _pattern_name(line) for line in output.splitlines() if line.startswith("pattern() = ") and not _pattern_name(line).startswith(".") } patterns = { k: v for k, v in _get_visible_patterns(root=root).items() if v["installed"] } for pattern in installed_patterns: if pattern not in patterns: patterns[pattern] = { "installed": True, "summary": "Non-visible pattern", } return patterns def list_patterns(refresh=False, root=None): """ List all known patterns from available repos. refresh force a refresh if set to True. If set to False (default) it depends on zypper if a refresh is executed. root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.list_patterns """ if refresh: refresh_db(root) return _get_visible_patterns(root=root) def list_installed_patterns(root=None): """ List installed patterns on the system. root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.list_installed_patterns """ return _get_installed_patterns(root=root) def search(criteria, refresh=False, **kwargs): """ List known packages, available to the system. refresh force a refresh if set to True. If set to False (default) it depends on zypper if a refresh is executed. match (str) One of `exact`, `words`, `substrings`. Search for an `exact` match or for the whole `words` only. Default to `substrings` to patch partial words. provides (bool) Search for packages which provide the search strings. recommends (bool) Search for packages which recommend the search strings. requires (bool) Search for packages which require the search strings. suggests (bool) Search for packages which suggest the search strings. conflicts (bool) Search packages conflicting with search strings. obsoletes (bool) Search for packages which obsolete the search strings. file_list (bool) Search for a match in the file list of packages. search_descriptions (bool) Search also in package summaries and descriptions. case_sensitive (bool) Perform case-sensitive search. installed_only (bool) Show only installed packages. not_installed_only (bool) Show only packages which are not installed. details (bool) Show version and repository root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.search <criteria> """ ALLOWED_SEARCH_OPTIONS = { "provides": "--provides", "recommends": "--recommends", "requires": "--requires", "suggests": "--suggests", "conflicts": "--conflicts", "obsoletes": "--obsoletes", "file_list": "--file-list", "search_descriptions": "--search-descriptions", "case_sensitive": "--case-sensitive", "installed_only": "--installed-only", "not_installed_only": "-u", "details": "--details", } root = kwargs.get("root", None) if refresh: refresh_db(root) cmd = ["search"] if kwargs.get("match") == "exact": cmd.append("--match-exact") elif kwargs.get("match") == "words": cmd.append("--match-words") elif kwargs.get("match") == "substrings": cmd.append("--match-substrings") for opt in kwargs: if opt in ALLOWED_SEARCH_OPTIONS: cmd.append(ALLOWED_SEARCH_OPTIONS.get(opt)) cmd.append(criteria) solvables = ( __zypper__(root=root) .nolock.noraise.xml.call(*cmd) .getElementsByTagName("solvable") ) if not solvables: raise CommandExecutionError("No packages found matching '{}'".format(criteria)) out = {} for solvable in solvables: out[solvable.getAttribute("name")] = dict() for k, v in solvable.attributes.items(): out[solvable.getAttribute("name")][k] = v return out def _get_first_aggregate_text(node_list): """ Extract text from the first occurred DOM aggregate. """ if not node_list: return "" out = [] for node in node_list[0].childNodes: if node.nodeType == dom.Document.TEXT_NODE: out.append(node.nodeValue) return "\n".join(out) def list_products(all=False, refresh=False, root=None): """ List all available or installed SUSE products. all List all products available or only installed. Default is False. refresh force a refresh if set to True. If set to False (default) it depends on zypper if a refresh is executed. root operate on a different root directory. Includes handling for OEM products, which read the OEM productline file and overwrite the release value. CLI Examples: .. code-block:: bash salt '*' pkg.list_products salt '*' pkg.list_products all=True """ if refresh: refresh_db(root) ret = list() OEM_PATH = "/var/lib/suseRegister/OEM" if root: OEM_PATH = os.path.join(root, os.path.relpath(OEM_PATH, os.path.sep)) cmd = list() if not all: cmd.append("--disable-repositories") cmd.append("products") if not all: cmd.append("-i") product_list = ( __zypper__(root=root).nolock.xml.call(*cmd).getElementsByTagName("product-list") ) if not product_list: return ret # No products found for prd in product_list[0].getElementsByTagName("product"): p_nfo = dict() for k_p_nfo, v_p_nfo in prd.attributes.items(): if k_p_nfo in ["isbase", "installed"]: p_nfo[k_p_nfo] = bool(v_p_nfo in ["true", "1"]) elif v_p_nfo: p_nfo[k_p_nfo] = v_p_nfo eol = prd.getElementsByTagName("endoflife") if eol: p_nfo["eol"] = eol[0].getAttribute("text") p_nfo["eol_t"] = int(eol[0].getAttribute("time_t") or 0) p_nfo["description"] = " ".join( [ line.strip() for line in _get_first_aggregate_text( prd.getElementsByTagName("description") ).split(os.linesep) ] ) if "productline" in p_nfo and p_nfo["productline"]: oem_file = os.path.join(OEM_PATH, p_nfo["productline"]) if os.path.isfile(oem_file): with salt.utils.files.fopen(oem_file, "r") as rfile: oem_release = salt.utils.stringutils.to_unicode( rfile.readline() ).strip() if oem_release: p_nfo["release"] = oem_release ret.append(p_nfo) return ret def download(*packages, **kwargs): """ Download packages to the local disk. refresh force a refresh if set to True. If set to False (default) it depends on zypper if a refresh is executed. root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.download httpd salt '*' pkg.download httpd postfix """ if not packages: raise SaltInvocationError("No packages specified") root = kwargs.get("root", None) refresh = kwargs.get("refresh", False) if refresh: refresh_db(root) pkg_ret = {} for dld_result in ( __zypper__(root=root) .xml.call("download", *packages) .getElementsByTagName("download-result") ): repo = dld_result.getElementsByTagName("repository")[0] path = dld_result.getElementsByTagName("localfile")[0].getAttribute("path") pkg_info = { "repository-name": repo.getAttribute("name"), "repository-alias": repo.getAttribute("alias"), "path": path, } key = _get_first_aggregate_text(dld_result.getElementsByTagName("name")) if __salt__["lowpkg.checksum"](pkg_info["path"], root=root): pkg_ret[key] = pkg_info if pkg_ret: failed = [pkg for pkg in packages if pkg not in pkg_ret] if failed: pkg_ret[ "_error" ] = "The following package(s) failed to download: {}".format( ", ".join(failed) ) return pkg_ret raise CommandExecutionError( "Unable to download packages: {}".format(", ".join(packages)) ) def list_downloaded(root=None, **kwargs): """ .. versionadded:: 2017.7.0 List prefetched packages downloaded by Zypper in the local disk. root operate on a different root directory. CLI Example: .. code-block:: bash salt '*' pkg.list_downloaded """ CACHE_DIR = "/var/cache/zypp/packages/" if root: CACHE_DIR = os.path.join(root, os.path.relpath(CACHE_DIR, os.path.sep)) ret = {} for root, dirnames, filenames in salt.utils.path.os_walk(CACHE_DIR): for filename in fnmatch.filter(filenames, "*.rpm"): package_path = os.path.join(root, filename) pkg_info = __salt__["lowpkg.bin_pkg_info"](package_path) pkg_timestamp = int(os.path.getctime(package_path)) ret.setdefault(pkg_info["name"], {})[pkg_info["version"]] = { "path": package_path, "size": os.path.getsize(package_path), "creation_date_time_t": pkg_timestamp, "creation_date_time": datetime.datetime.utcfromtimestamp( pkg_timestamp ).isoformat(), } return ret def diff(*paths, **kwargs): """ Return a formatted diff between current files and original in a package. NOTE: this function includes all files (configuration and not), but does not work on binary content. The root parameter can also be passed via the keyword argument. :param path: Full path to the installed file :return: Difference string or raises and exception if examined file is binary. CLI Example: .. code-block:: bash salt '*' pkg.diff /etc/apache2/httpd.conf /etc/sudoers """ ret = {} pkg_to_paths = {} for pth in paths: pth_pkg = __salt__["lowpkg.owner"](pth, **kwargs) if not pth_pkg: ret[pth] = os.path.exists(pth) and "Not managed" or "N/A" else: if pkg_to_paths.get(pth_pkg) is None: pkg_to_paths[pth_pkg] = [] pkg_to_paths[pth_pkg].append(pth) if pkg_to_paths: local_pkgs = __salt__["pkg.download"](*pkg_to_paths.keys(), **kwargs) for pkg, files in pkg_to_paths.items(): for path in files: ret[path] = ( __salt__["lowpkg.diff"](local_pkgs[pkg]["path"], path) or "Unchanged" ) return ret def _get_patches(installed_only=False, root=None): """ List all known patches in repos. """ patches = {} for element in ( __zypper__(root=root) .nolock.xml.call("se", "-t", "patch") .getElementsByTagName("solvable") ): installed = element.getAttribute("status") == "installed" if (installed_only and installed) or not installed_only: patches[element.getAttribute("name")] = { "installed": installed, "summary": element.getAttribute("summary"), } return patches def list_patches(refresh=False, root=None, **kwargs): """ .. versionadded:: 2017.7.0 List all known advisory patches from available repos. refresh force a refresh if set to True. If set to False (default) it depends on zypper if a refresh is executed. root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.list_patches """ if refresh: refresh_db(root) return _get_patches(root=root) def list_installed_patches(root=None, **kwargs): """ .. versionadded:: 2017.7.0 List installed advisory patches on the system. root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.list_installed_patches """ return _get_patches(installed_only=True, root=root) def list_provides(root=None, **kwargs): """ .. versionadded:: 2018.3.0 List package provides of installed packages as a dict. {'<provided_name>': ['<package_name>', '<package_name>', ...]} root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.list_provides """ ret = __context__.get("pkg.list_provides") if not ret: cmd = ["rpm"] if root: cmd.extend(["--root", root]) cmd.extend(["-qa", "--queryformat", "%{PROVIDES}_|-%{NAME}\n"]) ret = dict() for line in __salt__["cmd.run"]( cmd, output_loglevel="trace", python_shell=False ).splitlines(): provide, realname = line.split("_|-") if provide == realname: continue if provide not in ret: ret[provide] = list() ret[provide].append(realname) __context__["pkg.list_provides"] = ret return ret def resolve_capabilities(pkgs, refresh=False, root=None, **kwargs): """ .. versionadded:: 2018.3.0 Convert name provides in ``pkgs`` into real package names if ``resolve_capabilities`` parameter is set to True. In case of ``resolve_capabilities`` is set to False the package list is returned unchanged. refresh force a refresh if set to True. If set to False (default) it depends on zypper if a refresh is executed. root operate on a different root directory. resolve_capabilities If this option is set to True the input will be checked if a package with this name exists. If not, this function will search for a package which provides this name. If one is found the output is exchanged with the real package name. In case this option is set to False (Default) the input will be returned unchanged. CLI Examples: .. code-block:: bash salt '*' pkg.resolve_capabilities resolve_capabilities=True w3m_ssl """ if refresh: refresh_db(root) ret = list() for pkg in pkgs: if isinstance(pkg, dict): name = next(iter(pkg)) version = pkg[name] else: name = pkg version = None if kwargs.get("resolve_capabilities", False): try: search(name, root=root, match="exact") except CommandExecutionError: # no package this such a name found # search for a package which provides this name try: result = search(name, root=root, provides=True, match="exact") if len(result) == 1: name = next(iter(result.keys())) elif len(result) > 1: log.warning("Found ambiguous match for capability '%s'.", pkg) except CommandExecutionError as exc: # when search throws an exception stay with original name and version log.debug("Search failed with: %s", exc) if version: ret.append({name: version}) else: ret.append(name) return ret def services_need_restart(root=None, **kwargs): """ .. versionadded:: 3003 List services that use files which have been changed by the package manager. It might be needed to restart them. root operate on a different root directory. CLI Examples: .. code-block:: bash salt '*' pkg.services_need_restart """ cmd = ["ps", "-sss"] zypper_output = __zypper__(root=root).nolock.call(*cmd) services = zypper_output.split() return services