Mini Shell
"""
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 = f"{ZYPP_HOME}/locks"
REPOS = f"{ZYPP_HOME}/repos.d"
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(f"Zypper command failure: {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(f"No packages found matching '{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(
f'Zypper do not supports operator "{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, **kwargs)
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, **kwargs)
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 = f"pkg.list_pkgs_{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 = f"{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(f"Repository '{repo}' not found.")
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(
f"Repository '{repo}' already exists as '{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:
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
kwargs.update({"repos": repo})
refresh_db(root=root, **kwargs)
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, gpgautoimport=False, **kwargs):
"""
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}
gpgautoimport : False
If set to True, automatically trust and import public GPG key for
the repository.
.. versionadded:: 3007.0
repos
Refresh just the specified repos
.. versionadded:: 3007.0
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"]
global_opts = []
if force is None:
force = __pillar__.get("zypper", {}).get("refreshdb_force", True)
if force:
refresh_opts.append("--force")
repos = kwargs.get("repos", [])
refresh_opts.extend([repos] if not isinstance(repos, list) else repos)
if gpgautoimport:
global_opts.append("--gpg-auto-import-keys")
# We do the actual call to zypper refresh.
# We ignore retcode 6 which is returned when there are no repositories defined.
out = __zypper__(root=root).refreshable.call(
*global_opts, *refresh_opts, success_retcodes=[0, 6]
)
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, **kwargs)
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 = f"{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(f'Advisory id "{advisory_id}" not found')
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 = [f"patch:{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, **kwargs)
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)
ptfpackages = _find_ptf_packages(pkg_params.keys(), root=root)
includes = _find_types(pkg_params.keys())
old = list_pkgs(root=root, includes=includes)
targets = []
for target in pkg_params:
if target in ptfpackages:
# ptfpackages needs special handling
continue
# 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 and not ptfpackages:
return {}
systemd_scope = _systemd_scope()
errors = []
while targets:
__zypper__(systemd_scope=systemd_scope, root=root).call(
"remove", *targets[:500]
)
targets = targets[500:]
# handle ptf packages
while ptfpackages:
__zypper__(systemd_scope=systemd_scope, root=root).call(
"removeptf", "--allow-downgrade", *ptfpackages[:500]
)
ptfpackages = ptfpackages[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"]'
.. versionchanged:: 3007.0
Can now remove also PTF packages which require a different handling in the backend.
Can now remove also PTF packages which require a different handling in the backend.
"""
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(rf"{pattern}-\S+") 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 f"{target}={lock_ver}")
ret[target]["changes"]["new"] = ""
ret[target]["changes"]["old"] = "hold"
ret[target]["comment"] = f"Package {target} is no longer held."
else:
ret[target]["comment"] = f"Package {target} was already unheld."
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 f"{target}={version}")
ret[target]["changes"]["new"] = "hold"
ret[target]["changes"]["old"] = ""
ret[target]["comment"] = f"Package {target} is now being held."
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 _find_ptf_packages(pkgs, root=None):
"""
Find ptf packages in "pkgs" and return them as list
"""
ptfs = []
cmd = ["rpm"]
if root:
cmd.extend(["--root", root])
cmd.extend(["-q", "--qf", "%{NAME}: [%{PROVIDES} ]\n"])
cmd.extend(pkgs)
output = __salt__["cmd.run"](cmd)
for line in output.splitlines():
if not line.strip():
continue
if ":" not in line:
continue
pkg, provides = line.split(":", 1)
if "ptf()" in provides:
ptfs.append(pkg)
return ptfs
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, **kwargs)
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(f"No packages found matching '{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, **kwargs)
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, **kwargs)
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, **kwargs)
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
Zerion Mini Shell 1.0