2012-11-02 15:43:33 +11:00
|
|
|
#!/usr/bin/python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
# (c) 2012, Boyd Adamson <boyd () boydadamson.com>
|
|
|
|
#
|
|
|
|
# This file is part of Ansible
|
|
|
|
#
|
|
|
|
# Ansible is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# Ansible is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#
|
|
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: svr4pkg
|
|
|
|
short_description: Manage Solaris SVR4 packages
|
|
|
|
description:
|
|
|
|
- Manages SVR4 packages on Solaris 10 and 11.
|
|
|
|
- These were the native packages on Solaris <= 10 and are available
|
|
|
|
as a legacy feature in Solaris 11.
|
|
|
|
- Note that this is a very basic packaging system. It will not enforce
|
|
|
|
dependencies on install or remove.
|
2013-03-30 20:44:34 +01:00
|
|
|
version_added: "0.9"
|
2015-05-14 00:35:16 -04:00
|
|
|
author: "Boyd Adamson (@brontitall)"
|
2012-11-02 15:43:33 +11:00
|
|
|
options:
|
|
|
|
name:
|
|
|
|
description:
|
|
|
|
- Package name, e.g. C(SUNWcsr)
|
|
|
|
required: true
|
|
|
|
|
|
|
|
state:
|
|
|
|
description:
|
|
|
|
- Whether to install (C(present)), or remove (C(absent)) a package.
|
2012-11-21 18:49:30 +01:00
|
|
|
- If the package is to be installed, then I(src) is required.
|
2012-11-02 15:43:33 +11:00
|
|
|
- The SVR4 package system doesn't provide an upgrade operation. You need to uninstall the old, then install the new package.
|
|
|
|
required: true
|
|
|
|
choices: ["present", "absent"]
|
|
|
|
|
|
|
|
src:
|
|
|
|
description:
|
|
|
|
- Specifies the location to install the package from. Required when C(state=present).
|
Documentation of svr4pkg module breaks 'make rpm'
Without this fix, generating documentation results in:
```
Traceback (most recent call last):
File "hacking/module_formatter.py", line 376, in <module>
main()
File "hacking/module_formatter.py", line 365, in main
text = template.render(doc)
File "/usr/lib64/python2.6/site-packages/jinja2/environment.py", line 669, in render
return self.environment.handle_exception(exc_info, True)
File "hacking/templates/man.j2", line 20, in top-level template code
{% for desc in v.description %}@{ desc | jpfunc }@{% endfor %}
File "hacking/module_formatter.py", line 94, in man_ify
t = _ITALIC.sub(r'\\fI' + r"\1" + r"\\fR", text)
TypeError: expected string or buffer
```
2012-11-09 02:19:22 +01:00
|
|
|
- "Can be any path acceptable to the C(pkgadd) command's C(-d) option. e.g.: C(somefile.pkg), C(/dir/with/pkgs), C(http:/server/mypkgs.pkg)."
|
2012-11-21 18:49:30 +01:00
|
|
|
- If using a file or directory, they must already be accessible by the host. See the M(copy) module for a way to get them there.
|
2012-11-02 15:43:33 +11:00
|
|
|
proxy:
|
|
|
|
description:
|
|
|
|
- HTTP[s] proxy to be used if C(src) is a URL.
|
2013-10-12 13:28:48 -04:00
|
|
|
response_file:
|
2013-10-09 19:56:47 -04:00
|
|
|
description:
|
2013-10-19 11:54:36 -04:00
|
|
|
- Specifies the location of a response file to be used if package expects input on install. (added in Ansible 1.4)
|
2013-11-13 20:04:54 -05:00
|
|
|
required: false
|
2014-03-25 09:03:48 +00:00
|
|
|
zone:
|
|
|
|
description:
|
|
|
|
- Whether to install the package only in the current zone, or install it into all zones.
|
|
|
|
- The installation into all zones works only if you are working with the global zone.
|
|
|
|
required: false
|
|
|
|
default: "all"
|
|
|
|
choices: ["current", "all"]
|
2014-03-26 10:25:42 +00:00
|
|
|
version_added: "1.6"
|
2014-03-28 15:35:52 -04:00
|
|
|
category:
|
|
|
|
description:
|
|
|
|
- Install/Remove category instead of a single package.
|
|
|
|
required: false
|
|
|
|
choices: ["true", "false"]
|
|
|
|
version_added: "1.6"
|
2013-06-14 11:53:43 +02:00
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = '''
|
|
|
|
# Install a package from an already copied file
|
|
|
|
- svr4pkg: name=CSWcommon src=/tmp/cswpkgs.pkg state=present
|
|
|
|
|
|
|
|
# Install a package directly from an http site
|
2014-03-25 09:03:48 +00:00
|
|
|
- svr4pkg: name=CSWpkgutil src=http://get.opencsw.org/now state=present zone=current
|
2012-11-02 15:43:33 +11:00
|
|
|
|
2013-10-09 19:56:47 -04:00
|
|
|
# Install a package with a response file
|
2013-10-12 13:28:48 -04:00
|
|
|
- svr4pkg: name=CSWggrep src=/tmp/third-party.pkg response_file=/tmp/ggrep.response state=present
|
2013-10-09 19:56:47 -04:00
|
|
|
|
2013-06-14 11:53:43 +02:00
|
|
|
# Ensure that a package is not installed.
|
|
|
|
- svr4pkg: name=SUNWgnome-sound-recorder state=absent
|
2014-03-28 15:35:52 -04:00
|
|
|
|
|
|
|
# Ensure that a category is not installed.
|
|
|
|
- svr4pkg: name=FIREFOX state=absent category=true
|
2012-11-02 15:43:33 +11:00
|
|
|
'''
|
2013-06-14 11:53:43 +02:00
|
|
|
|
|
|
|
|
2012-11-02 15:43:33 +11:00
|
|
|
import os
|
|
|
|
import tempfile
|
|
|
|
|
2014-03-28 15:35:52 -04:00
|
|
|
def package_installed(module, name, category):
|
2012-11-02 15:43:33 +11:00
|
|
|
cmd = [module.get_bin_path('pkginfo', True)]
|
|
|
|
cmd.append('-q')
|
2014-03-28 15:35:52 -04:00
|
|
|
if category:
|
|
|
|
cmd.append('-c')
|
2012-11-02 15:43:33 +11:00
|
|
|
cmd.append(name)
|
2013-07-16 09:17:57 +02:00
|
|
|
rc, out, err = module.run_command(' '.join(cmd))
|
Update modules to use run_command in module_common.py
This updates apt, apt_repository, command, cron, easy_install, facter,
fireball, git, group, mount, ohai, pip, service, setup, subversion,
supervisorctl, svr4pkg, user, and yum to take advantage of run_command
in module_common.py.
2013-01-11 22:10:21 -08:00
|
|
|
if rc == 0:
|
2012-11-02 15:43:33 +11:00
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
def create_admin_file():
|
|
|
|
(desc, filename) = tempfile.mkstemp(prefix='ansible_svr4pkg', text=True)
|
|
|
|
fullauto = '''
|
|
|
|
mail=
|
|
|
|
instance=unique
|
|
|
|
partial=nocheck
|
|
|
|
runlevel=quit
|
|
|
|
idepend=nocheck
|
|
|
|
rdepend=nocheck
|
|
|
|
space=quit
|
|
|
|
setuid=nocheck
|
|
|
|
conflict=nocheck
|
|
|
|
action=nocheck
|
|
|
|
networktimeout=60
|
|
|
|
networkretries=3
|
|
|
|
authentication=quit
|
|
|
|
keystore=/var/sadm/security
|
|
|
|
proxy=
|
|
|
|
basedir=default
|
|
|
|
'''
|
|
|
|
os.write(desc, fullauto)
|
|
|
|
os.close(desc)
|
|
|
|
return filename
|
|
|
|
|
|
|
|
def run_command(module, cmd):
|
|
|
|
progname = cmd[0]
|
|
|
|
cmd[0] = module.get_bin_path(progname, True)
|
Update modules to use run_command in module_common.py
This updates apt, apt_repository, command, cron, easy_install, facter,
fireball, git, group, mount, ohai, pip, service, setup, subversion,
supervisorctl, svr4pkg, user, and yum to take advantage of run_command
in module_common.py.
2013-01-11 22:10:21 -08:00
|
|
|
return module.run_command(cmd)
|
2012-11-02 15:43:33 +11:00
|
|
|
|
2014-03-28 15:35:52 -04:00
|
|
|
def package_install(module, name, src, proxy, response_file, zone, category):
|
2012-11-02 15:43:33 +11:00
|
|
|
adminfile = create_admin_file()
|
2014-03-25 09:03:48 +00:00
|
|
|
cmd = [ 'pkgadd', '-n']
|
|
|
|
if zone == 'current':
|
|
|
|
cmd += [ '-G' ]
|
|
|
|
cmd += [ '-a', adminfile, '-d', src ]
|
2012-11-02 15:43:33 +11:00
|
|
|
if proxy is not None:
|
|
|
|
cmd += [ '-x', proxy ]
|
2013-10-12 13:28:48 -04:00
|
|
|
if response_file is not None:
|
|
|
|
cmd += [ '-r', response_file ]
|
2014-03-28 15:35:52 -04:00
|
|
|
if category:
|
|
|
|
cmd += [ '-Y' ]
|
2012-11-02 15:43:33 +11:00
|
|
|
cmd.append(name)
|
|
|
|
(rc, out, err) = run_command(module, cmd)
|
|
|
|
os.unlink(adminfile)
|
|
|
|
return (rc, out, err)
|
|
|
|
|
2014-06-08 19:59:44 -04:00
|
|
|
def package_uninstall(module, name, src, category):
|
2012-11-02 15:43:33 +11:00
|
|
|
adminfile = create_admin_file()
|
2014-03-28 15:35:52 -04:00
|
|
|
if category:
|
|
|
|
cmd = [ 'pkgrm', '-na', adminfile, '-Y', name ]
|
|
|
|
else:
|
|
|
|
cmd = [ 'pkgrm', '-na', adminfile, name]
|
2012-11-02 15:43:33 +11:00
|
|
|
(rc, out, err) = run_command(module, cmd)
|
|
|
|
os.unlink(adminfile)
|
|
|
|
return (rc, out, err)
|
|
|
|
|
|
|
|
def main():
|
|
|
|
module = AnsibleModule(
|
|
|
|
argument_spec = dict(
|
|
|
|
name = dict(required = True),
|
|
|
|
state = dict(required = True, choices=['present', 'absent']),
|
|
|
|
src = dict(default = None),
|
2013-10-09 19:56:47 -04:00
|
|
|
proxy = dict(default = None),
|
2014-03-25 09:03:48 +00:00
|
|
|
response_file = dict(default = None),
|
2014-03-28 15:35:52 -04:00
|
|
|
zone = dict(required=False, default = 'all', choices=['current','all']),
|
|
|
|
category = dict(default=False, type='bool')
|
2013-02-27 12:23:35 -08:00
|
|
|
),
|
|
|
|
supports_check_mode=True
|
|
|
|
)
|
2012-11-02 15:43:33 +11:00
|
|
|
state = module.params['state']
|
|
|
|
name = module.params['name']
|
|
|
|
src = module.params['src']
|
|
|
|
proxy = module.params['proxy']
|
2013-10-12 13:28:48 -04:00
|
|
|
response_file = module.params['response_file']
|
2014-03-25 09:03:48 +00:00
|
|
|
zone = module.params['zone']
|
2014-03-28 15:35:52 -04:00
|
|
|
category = module.params['category']
|
2012-11-02 15:43:33 +11:00
|
|
|
rc = None
|
|
|
|
out = ''
|
|
|
|
err = ''
|
|
|
|
result = {}
|
|
|
|
result['name'] = name
|
|
|
|
result['state'] = state
|
|
|
|
|
|
|
|
if state == 'present':
|
|
|
|
if src is None:
|
|
|
|
module.fail_json(name=name,
|
|
|
|
msg="src is required when state=present")
|
2014-03-28 15:35:52 -04:00
|
|
|
if not package_installed(module, name, category):
|
2013-02-27 12:23:35 -08:00
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=True)
|
2014-03-28 15:35:52 -04:00
|
|
|
(rc, out, err) = package_install(module, name, src, proxy, response_file, zone, category)
|
2012-11-02 15:43:33 +11:00
|
|
|
# Stdout is normally empty but for some packages can be
|
|
|
|
# very long and is not often useful
|
|
|
|
if len(out) > 75:
|
|
|
|
out = out[:75] + '...'
|
|
|
|
|
|
|
|
elif state == 'absent':
|
2014-03-28 15:35:52 -04:00
|
|
|
if package_installed(module, name, category):
|
2013-02-27 12:23:35 -08:00
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=True)
|
2014-03-28 15:35:52 -04:00
|
|
|
(rc, out, err) = package_uninstall(module, name, src, category)
|
2012-11-02 15:43:33 +11:00
|
|
|
out = out[:75]
|
|
|
|
|
2015-05-27 13:35:34 +02:00
|
|
|
# Returncodes as per pkgadd(1m)
|
|
|
|
# 0 Successful completion
|
|
|
|
# 1 Fatal error.
|
|
|
|
# 2 Warning.
|
|
|
|
# 3 Interruption.
|
|
|
|
# 4 Administration.
|
|
|
|
# 5 Administration. Interaction is required. Do not use pkgadd -n.
|
|
|
|
# 10 Reboot after installation of all packages.
|
|
|
|
# 20 Reboot after installation of this package.
|
|
|
|
# 99 (observed) pkgadd: ERROR: could not process datastream from </tmp/pkgutil.pkg>
|
2014-06-09 10:48:06 -05:00
|
|
|
if rc in (0, 2, 3, 10, 20):
|
2014-06-08 19:59:44 -04:00
|
|
|
result['changed'] = True
|
2014-06-09 10:48:06 -05:00
|
|
|
# no install nor uninstall, or failed
|
2014-06-08 19:59:44 -04:00
|
|
|
else:
|
2012-11-02 15:43:33 +11:00
|
|
|
result['changed'] = False
|
2014-06-09 10:48:06 -05:00
|
|
|
|
2015-05-27 13:35:34 +02:00
|
|
|
# Only return failed=False when the returncode is known to be good as there may be more
|
|
|
|
# undocumented failure return codes
|
|
|
|
if rc not in (0, 2, 10, 20):
|
2014-06-09 10:48:06 -05:00
|
|
|
result['failed'] = True
|
2012-11-02 15:43:33 +11:00
|
|
|
else:
|
2014-06-09 10:48:06 -05:00
|
|
|
result['failed'] = False
|
2012-11-02 15:43:33 +11:00
|
|
|
|
|
|
|
if out:
|
|
|
|
result['stdout'] = out
|
|
|
|
if err:
|
|
|
|
result['stderr'] = err
|
|
|
|
|
|
|
|
module.exit_json(**result)
|
|
|
|
|
2013-12-02 15:11:23 -05:00
|
|
|
# import module snippets
|
|
|
|
from ansible.module_utils.basic import *
|
2012-11-02 15:43:33 +11:00
|
|
|
main()
|