mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
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
This commit is contained in:
parent
adfd990232
commit
85b1bd5c90
1 changed files with 81 additions and 16 deletions
|
@ -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…
Reference in a new issue