allow to specify versions with zypper (#2328)

* fixes #2158
* handles version-specifiers (>,<,>=,<=,=) correctly
* adds option "oldpackage", which is passed to zypper
  * this is implied as soon as a version is specified
  * it can be used independently to allow downgrades coming from repos
* add __main__ check
* extend documentation on version specifier
pull/18777/head
Robin Roth 8 years ago committed by Matt Clay
parent adfd990232
commit 85b1bd5c90

@ -27,6 +27,7 @@
# along with Ansible. If not, see <http://www.gnu.org/licenses/>. # along with Ansible. If not, see <http://www.gnu.org/licenses/>.
from xml.dom.minidom import parseString as parseXML from xml.dom.minidom import parseString as parseXML
import re
DOCUMENTATION = ''' DOCUMENTATION = '''
--- ---
@ -44,7 +45,10 @@ description:
options: options:
name: name:
description: description:
- package name or package specifier with version C(name) or C(name-1.0). You can also pass a url or a local path to a rpm file. When using state=latest, this can be '*', which updates all installed packages. - Package name C(name) or package specifier.
- Can include a version like C(name=1.0), C(name>3.4) or C(name<=2.7). If a version is given, C(oldpackage) is implied and zypper is allowed to update the package within the version range given.
- You can also pass a url or a local path to a rpm file.
- When using state=latest, this can be '*', which updates all installed packages.
required: true required: true
aliases: [ 'pkg' ] aliases: [ 'pkg' ]
state: state:
@ -92,7 +96,13 @@ options:
default: "no" default: "no"
choices: [ "yes", "no" ] choices: [ "yes", "no" ]
aliases: [ "refresh" ] aliases: [ "refresh" ]
oldpackage:
version_added: "2.2"
description:
- Adds C(--oldpackage) option to I(zypper). Allows to downgrade packages with less side-effects than force. This is implied as soon as a version is specified as part of the package name.
required: false
default: "no"
choices: [ "yes", "no" ]
# informational: requirements for nodes # informational: requirements for nodes
requirements: requirements:
@ -127,25 +137,57 @@ EXAMPLES = '''
# Refresh repositories and update package "openssl" # Refresh repositories and update package "openssl"
- zypper: name=openssl state=present update_cache=yes - zypper: name=openssl state=present update_cache=yes
# Install specific version (possible comparisons: <, >, <=, >=, =)
- zypper: name=docker>=1.10 state=installed
''' '''
def split_name_version(name):
"""splits of the package name and desired version
example formats:
- docker>=1.10
- apache=2.4
Allowed version specifiers: <, >, <=, >=, =
Allowed version format: [0-9.-]*
Also allows a prefix indicating remove "-", "~" or install "+"
"""
prefix = ''
if name[0] in ['-', '~', '+']:
prefix = name[0]
name = name[1:]
version_check = re.compile('^(.*?)((?:<|>|<=|>=|=)[0-9.-]*)?$')
try:
reres = version_check.match(name)
name, version = reres.groups()
return prefix, name, version
except:
return prefix, name, None
def get_want_state(m, names, remove=False): def get_want_state(m, names, remove=False):
packages_install = [] packages_install = {}
packages_remove = [] packages_remove = {}
urls = [] urls = []
for name in names: for name in names:
if '://' in name or name.endswith('.rpm'): if '://' in name or name.endswith('.rpm'):
urls.append(name) urls.append(name)
elif name.startswith('-') or name.startswith('~'): else:
packages_remove.append(name[1:]) prefix, pname, version = split_name_version(name)
elif name.startswith('+'): if prefix in ['-', '~']:
packages_install.append(name[1:]) packages_remove[pname] = version
elif prefix == '+':
packages_install[pname] = version
else: else:
if remove: if remove:
packages_remove.append(name) packages_remove[pname] = version
else: else:
packages_install.append(name) packages_install[pname] = version
return packages_install, packages_remove, urls return packages_install, packages_remove, urls
@ -216,6 +258,8 @@ def get_cmd(m, subcommand):
cmd.append('--no-recommends') cmd.append('--no-recommends')
if m.params['force']: if m.params['force']:
cmd.append('--force') cmd.append('--force')
if m.params['oldpackage']:
cmd.append('--oldpackage')
return cmd return cmd
@ -249,13 +293,23 @@ def package_present(m, name, want_latest):
retvals = {'rc': 0, 'stdout': '', 'stderr': ''} retvals = {'rc': 0, 'stdout': '', 'stderr': ''}
name_install, name_remove, urls = get_want_state(m, name) name_install, name_remove, urls = get_want_state(m, name)
# if a version string is given, pass it to zypper
install_version = [p+name_install[p] for p in name_install if name_install[p]]
remove_version = [p+name_remove[p] for p in name_remove if name_remove[p]]
# add oldpackage flag when a version is given to allow downgrades
if install_version or remove_version:
m.params['oldpackage'] = True
if not want_latest: if not want_latest:
# for state=present: filter out already installed packages # for state=present: filter out already installed packages
prerun_state = get_installed_state(m, name_install + name_remove) install_and_remove = name_install.copy()
install_and_remove.update(name_remove)
prerun_state = get_installed_state(m, install_and_remove)
# generate lists of packages to install or remove # generate lists of packages to install or remove
name_install = [p for p in name_install if p not in prerun_state] name_install = [p for p in name_install if p not in prerun_state]
name_remove = [p for p in name_remove if p in prerun_state] name_remove = [p for p in name_remove if p in prerun_state]
if not name_install and not name_remove and not urls: if not any((name_install, name_remove, urls, install_version, remove_version)):
# nothing to install/remove and nothing to update # nothing to install/remove and nothing to update
return None, retvals return None, retvals
@ -264,6 +318,10 @@ def package_present(m, name, want_latest):
cmd.append('--') cmd.append('--')
cmd.extend(urls) cmd.extend(urls)
# pass packages with version information
cmd.extend(install_version)
cmd.extend(['-%s' % p for p in remove_version])
# allow for + or - prefixes in install/remove lists # allow for + or - prefixes in install/remove lists
# do this in one zypper run to allow for dependency-resolution # do this in one zypper run to allow for dependency-resolution
# for example "-exim postfix" runs without removing packages depending on mailserver # for example "-exim postfix" runs without removing packages depending on mailserver
@ -303,12 +361,14 @@ def package_absent(m, name):
if m.params['type'] == 'patch': if m.params['type'] == 'patch':
m.fail_json(msg="Can not remove patches.") m.fail_json(msg="Can not remove patches.")
prerun_state = get_installed_state(m, name_remove) prerun_state = get_installed_state(m, name_remove)
remove_version = [p+name_remove[p] for p in name_remove if name_remove[p]]
name_remove = [p for p in name_remove if p in prerun_state] name_remove = [p for p in name_remove if p in prerun_state]
if not name_remove: if not name_remove and not remove_version:
return None, retvals return None, retvals
cmd = get_cmd(m, 'remove') cmd = get_cmd(m, 'remove')
cmd.extend(name_remove) cmd.extend(name_remove)
cmd.extend(remove_version)
retvals['cmd'] = cmd retvals['cmd'] = cmd
result, retvals['rc'], retvals['stdout'], retvals['stderr'] = parse_zypper_xml(m, cmd) result, retvals['rc'], retvals['stdout'], retvals['stderr'] = parse_zypper_xml(m, cmd)
@ -339,6 +399,7 @@ def main():
disable_recommends = dict(required=False, default='yes', type='bool'), disable_recommends = dict(required=False, default='yes', type='bool'),
force = dict(required=False, default='no', type='bool'), force = dict(required=False, default='no', type='bool'),
update_cache = dict(required=False, aliases=['refresh'], default='no', type='bool'), update_cache = dict(required=False, aliases=['refresh'], default='no', type='bool'),
oldpackage = dict(required=False, default='no', type='bool'),
), ),
supports_check_mode = True supports_check_mode = True
) )
@ -347,6 +408,9 @@ def main():
state = module.params['state'] state = module.params['state']
update_cache = module.params['update_cache'] update_cache = module.params['update_cache']
# remove empty strings from package list
name = filter(None, name)
# Refresh repositories # Refresh repositories
if update_cache: if update_cache:
retvals = repo_refresh(module) retvals = repo_refresh(module)
@ -378,5 +442,6 @@ def main():
module.exit_json(name=name, state=state, update_cache=update_cache, **retvals) module.exit_json(name=name, state=state, update_cache=update_cache, **retvals)
# import module snippets # import module snippets
from ansible.module_utils.basic import * from ansible.module_utils.basic import AnsibleModule
if __name__ == "__main__":
main() main()

Loading…
Cancel
Save