pkgin: support check mode, make other improvements
* Add support for check mode * Use "pkgin search" to guarantee 0 or 1 result * Edit documentation for style, new feature, etc. * General refactoring * Lay some groundwork for future support of "state=latest"
This commit is contained in:
parent
229e5ad80f
commit
3425828795
1 changed files with 88 additions and 46 deletions
134
packaging/os/pkgin.py
Executable file → Normal file
134
packaging/os/pkgin.py
Executable file → Normal file
|
@ -1,8 +1,10 @@
|
||||||
#!/usr/bin/python
|
#!/usr/bin/python
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
# (c) 2013, Shaun Zinck
|
# Copyright (c) 2013 Shaun Zinck <shaun.zinck at gmail.com>
|
||||||
# Written by Shaun Zinck <shaun.zinck at gmail.com>
|
# Copyright (c) 2015 Lawrence Leonard Gilbert <larry@L2G.to>
|
||||||
|
#
|
||||||
|
# Written by Shaun Zinck
|
||||||
# Based on pacman module written by Afterburn <http://github.com/afterburn>
|
# Based on pacman module written by Afterburn <http://github.com/afterburn>
|
||||||
# that was based on apt module written by Matthew Williams <matthew@flowroute.com>
|
# that was based on apt module written by Matthew Williams <matthew@flowroute.com>
|
||||||
#
|
#
|
||||||
|
@ -23,27 +25,32 @@
|
||||||
DOCUMENTATION = '''
|
DOCUMENTATION = '''
|
||||||
---
|
---
|
||||||
module: pkgin
|
module: pkgin
|
||||||
short_description: Package manager for SmartOS
|
short_description: Package manager for SmartOS, NetBSD, et al.
|
||||||
description:
|
description:
|
||||||
- Manages SmartOS packages
|
- "The standard package manager for SmartOS, but also usable on NetBSD
|
||||||
|
or any OS that uses C(pkgsrc). (Home: U(http://pkgin.net/))"
|
||||||
version_added: "1.0"
|
version_added: "1.0"
|
||||||
|
author: Shaun Zinck, Larry Gilbert
|
||||||
|
notes:
|
||||||
|
- "Known bug with pkgin < 0.8.0: if a package is removed and another
|
||||||
|
package depends on it, the other package will be silently removed as
|
||||||
|
well. New to Ansible 1.9: check-mode support."
|
||||||
options:
|
options:
|
||||||
name:
|
name:
|
||||||
description:
|
description:
|
||||||
- name of package to install/remove
|
- Name of package to install/remove;
|
||||||
|
- multiple names may be given, separated by commas
|
||||||
required: true
|
required: true
|
||||||
state:
|
state:
|
||||||
description:
|
description:
|
||||||
- state of the package
|
- Intended state of the package
|
||||||
choices: [ 'present', 'absent' ]
|
choices: [ 'present', 'absent' ]
|
||||||
required: false
|
required: false
|
||||||
default: present
|
default: present
|
||||||
author: Shaun Zinck
|
|
||||||
notes: []
|
|
||||||
'''
|
'''
|
||||||
|
|
||||||
EXAMPLES = '''
|
EXAMPLES = '''
|
||||||
# install package foo"
|
# install package foo
|
||||||
- pkgin: name=foo state=present
|
- pkgin: name=foo state=present
|
||||||
|
|
||||||
# remove package foo
|
# remove package foo
|
||||||
|
@ -60,64 +67,97 @@ import os
|
||||||
import sys
|
import sys
|
||||||
import pipes
|
import pipes
|
||||||
|
|
||||||
def query_package(module, pkgin_path, name, state="present"):
|
def query_package(module, pkgin_path, name):
|
||||||
|
"""Search for the package by name.
|
||||||
|
|
||||||
if state == "present":
|
Possible return values:
|
||||||
|
* "present" - installed, no upgrade needed
|
||||||
|
* "outdated" - installed, but can be upgraded
|
||||||
|
* False - not installed or not found
|
||||||
|
"""
|
||||||
|
|
||||||
rc, out, err = module.run_command("%s -y list | grep ^%s" % (pipes.quote(pkgin_path), pipes.quote(name)), use_unsafe_shell=True)
|
# Use "pkgin search" to find the package. The regular expression will
|
||||||
|
# only match on the complete name.
|
||||||
|
rc, out, err = module.run_command("%s search \"^%s$\"" % (pkgin_path, name))
|
||||||
|
|
||||||
if rc == 0:
|
# rc will not be 0 unless the search was a success
|
||||||
# At least one package with a package name that starts with ``name``
|
if rc == 0:
|
||||||
# is installed. For some cases this is not sufficient to determine
|
|
||||||
# wether the queried package is installed.
|
|
||||||
#
|
|
||||||
# E.g. for ``name='gcc47'``, ``gcc47`` not being installed, but
|
|
||||||
# ``gcc47-libs`` being installed, ``out`` would be:
|
|
||||||
#
|
|
||||||
# gcc47-libs-4.7.2nb4 The GNU Compiler Collection (GCC) support shared libraries.
|
|
||||||
#
|
|
||||||
# Multiline output is also possible, for example with the same query
|
|
||||||
# and bot ``gcc47`` and ``gcc47-libs`` being installed:
|
|
||||||
#
|
|
||||||
# gcc47-libs-4.7.2nb4 The GNU Compiler Collection (GCC) support shared libraries.
|
|
||||||
# gcc47-4.7.2nb3 The GNU Compiler Collection (GCC) - 4.7 Release Series
|
|
||||||
|
|
||||||
# Loop over lines in ``out``
|
# Get first line
|
||||||
for line in out.split('\n'):
|
line = out.split('\n')[0]
|
||||||
|
|
||||||
# Strip description
|
# Break up line at spaces. The first part will be the package with its
|
||||||
# (results in sth. like 'gcc47-libs-4.7.2nb4')
|
# version (e.g. 'gcc47-libs-4.7.2nb4'), and the second will be the state
|
||||||
pkgname_with_version = out.split(' ')[0]
|
# of the package:
|
||||||
|
# '' - not installed
|
||||||
|
# '<' - installed but out of date
|
||||||
|
# '=' - installed and up to date
|
||||||
|
# '>' - installed but newer than the repository version
|
||||||
|
pkgname_with_version, raw_state = out.split(' ')[0:2]
|
||||||
|
|
||||||
# Strip version
|
# Strip version
|
||||||
# (results in sth like 'gcc47-libs')
|
# (results in sth like 'gcc47-libs')
|
||||||
pkgname_without_version = '-'.join(pkgname_with_version.split('-')[:-1])
|
pkgname_without_version = '-'.join(pkgname_with_version.split('-')[:-1])
|
||||||
|
|
||||||
if name == pkgname_without_version:
|
if name != pkgname_without_version:
|
||||||
return True
|
return False
|
||||||
|
# no fall-through
|
||||||
|
|
||||||
return False
|
# The package was found; now return its state
|
||||||
|
if raw_state == '<':
|
||||||
|
return 'outdated'
|
||||||
|
elif raw_state == '=' or raw_state == '>':
|
||||||
|
return 'present'
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
|
def format_action_message(module, action, count):
|
||||||
|
vars = { "actioned": action,
|
||||||
|
"count": count }
|
||||||
|
|
||||||
|
if module.check_mode:
|
||||||
|
message = "would have %(actioned)s %(count)d package" % vars
|
||||||
|
else:
|
||||||
|
message = "%(actioned)s %(count)d package" % vars
|
||||||
|
|
||||||
|
if count == 1:
|
||||||
|
return message
|
||||||
|
else:
|
||||||
|
return message + "s"
|
||||||
|
|
||||||
|
|
||||||
|
def format_pkgin_command(module, pkgin_path, command, package):
|
||||||
|
vars = { "pkgin": pkgin_path,
|
||||||
|
"command": command,
|
||||||
|
"package": package }
|
||||||
|
|
||||||
|
if module.check_mode:
|
||||||
|
return "%(pkgin)s -n %(command)s %(package)s" % vars
|
||||||
|
else:
|
||||||
|
return "%(pkgin)s -y %(command)s %(package)s" % vars
|
||||||
|
|
||||||
|
|
||||||
def remove_packages(module, pkgin_path, packages):
|
def remove_packages(module, pkgin_path, packages):
|
||||||
|
|
||||||
remove_c = 0
|
remove_c = 0
|
||||||
|
|
||||||
# Using a for loop incase of error, we can report the package that failed
|
# Using a for loop incase of error, we can report the package that failed
|
||||||
for package in packages:
|
for package in packages:
|
||||||
# Query the package first, to see if we even need to remove
|
# Query the package first, to see if we even need to remove
|
||||||
if not query_package(module, pkgin_path, package):
|
if not query_package(module, pkgin_path, package):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
rc, out, err = module.run_command("%s -y remove %s" % (pkgin_path, package))
|
rc, out, err = module.run_command(
|
||||||
|
format_pkgin_command(module, pkgin_path, "remove", package))
|
||||||
|
|
||||||
if query_package(module, pkgin_path, package):
|
if not module.check_mode and query_package(module, pkgin_path, package):
|
||||||
module.fail_json(msg="failed to remove %s: %s" % (package, out))
|
module.fail_json(msg="failed to remove %s: %s" % (package, out))
|
||||||
|
|
||||||
remove_c += 1
|
remove_c += 1
|
||||||
|
|
||||||
if remove_c > 0:
|
if remove_c > 0:
|
||||||
|
module.exit_json(changed=True, msg=format_action_message(module, "removed", remove_c))
|
||||||
module.exit_json(changed=True, msg="removed %s package(s)" % remove_c)
|
|
||||||
|
|
||||||
module.exit_json(changed=False, msg="package(s) already absent")
|
module.exit_json(changed=False, msg="package(s) already absent")
|
||||||
|
|
||||||
|
@ -130,15 +170,16 @@ def install_packages(module, pkgin_path, packages):
|
||||||
if query_package(module, pkgin_path, package):
|
if query_package(module, pkgin_path, package):
|
||||||
continue
|
continue
|
||||||
|
|
||||||
rc, out, err = module.run_command("%s -y install %s" % (pkgin_path, package))
|
rc, out, err = module.run_command(
|
||||||
|
format_pkgin_command(module, pkgin_path, "install", package))
|
||||||
|
|
||||||
if not query_package(module, pkgin_path, package):
|
if not module.check_mode and not query_package(module, pkgin_path, package):
|
||||||
module.fail_json(msg="failed to install %s: %s" % (package, out))
|
module.fail_json(msg="failed to install %s: %s" % (package, out))
|
||||||
|
|
||||||
install_c += 1
|
install_c += 1
|
||||||
|
|
||||||
if install_c > 0:
|
if install_c > 0:
|
||||||
module.exit_json(changed=True, msg="present %s package(s)" % (install_c))
|
module.exit_json(changed=True, msg=format_action_message(module, "installed", install_c))
|
||||||
|
|
||||||
module.exit_json(changed=False, msg="package(s) already present")
|
module.exit_json(changed=False, msg="package(s) already present")
|
||||||
|
|
||||||
|
@ -148,7 +189,8 @@ def main():
|
||||||
module = AnsibleModule(
|
module = AnsibleModule(
|
||||||
argument_spec = dict(
|
argument_spec = dict(
|
||||||
state = dict(default="present", choices=["present","absent"]),
|
state = dict(default="present", choices=["present","absent"]),
|
||||||
name = dict(aliases=["pkg"], required=True)))
|
name = dict(aliases=["pkg"], required=True)),
|
||||||
|
supports_check_mode = True)
|
||||||
|
|
||||||
pkgin_path = module.get_bin_path('pkgin', True, ['/opt/local/bin'])
|
pkgin_path = module.get_bin_path('pkgin', True, ['/opt/local/bin'])
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue