2020-03-09 10:11:07 +01:00
|
|
|
#!/usr/bin/python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2022-08-05 22:12:10 +02:00
|
|
|
# Copyright (c) 2013, Andrew Dunham <andrew@du.nham.ca>
|
|
|
|
# Copyright (c) 2013, Daniel Jaouen <dcj24@cornell.edu>
|
|
|
|
# Copyright (c) 2015, Indrajit Raychaudhuri <irc+code@indrajit.com>
|
2020-03-09 10:11:07 +01:00
|
|
|
#
|
|
|
|
# Based on macports (Jimmy Tang <jcftang@gmail.com>)
|
|
|
|
#
|
2022-08-05 12:28:29 +02:00
|
|
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
2020-03-09 10:11:07 +01:00
|
|
|
|
|
|
|
from __future__ import absolute_import, division, print_function
|
|
|
|
__metaclass__ = type
|
|
|
|
|
|
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: homebrew
|
|
|
|
author:
|
|
|
|
- "Indrajit Raychaudhuri (@indrajitr)"
|
|
|
|
- "Daniel Jaouen (@danieljaouen)"
|
|
|
|
- "Andrew Dunham (@andrew-d)"
|
|
|
|
requirements:
|
2023-02-24 09:25:20 +01:00
|
|
|
- homebrew must already be installed on the target system
|
2020-03-09 10:11:07 +01:00
|
|
|
short_description: Package manager for Homebrew
|
|
|
|
description:
|
|
|
|
- Manages Homebrew packages
|
2023-02-24 09:25:20 +01:00
|
|
|
extends_documentation_fragment:
|
|
|
|
- community.general.attributes
|
|
|
|
attributes:
|
|
|
|
check_mode:
|
|
|
|
support: full
|
|
|
|
diff_mode:
|
|
|
|
support: none
|
2020-03-09 10:11:07 +01:00
|
|
|
options:
|
|
|
|
name:
|
|
|
|
description:
|
2020-03-24 09:29:03 +01:00
|
|
|
- A list of names of packages to install/remove.
|
|
|
|
aliases: [ 'formula', 'package', 'pkg' ]
|
2020-03-09 10:11:07 +01:00
|
|
|
type: list
|
|
|
|
elements: str
|
|
|
|
path:
|
|
|
|
description:
|
2023-06-15 15:47:00 +02:00
|
|
|
- "A V(:) separated list of paths to search for C(brew) executable.
|
|
|
|
Since a package (I(formula) in homebrew parlance) location is prefixed relative to the actual path of C(brew) command,
|
|
|
|
providing an alternative C(brew) path enables managing different set of packages in an alternative location in the system."
|
2022-09-08 07:46:10 +02:00
|
|
|
default: '/usr/local/bin:/opt/homebrew/bin:/home/linuxbrew/.linuxbrew/bin'
|
2020-03-24 09:29:03 +01:00
|
|
|
type: path
|
2020-03-09 10:11:07 +01:00
|
|
|
state:
|
|
|
|
description:
|
2020-03-24 09:29:03 +01:00
|
|
|
- state of the package.
|
|
|
|
choices: [ 'absent', 'head', 'installed', 'latest', 'linked', 'present', 'removed', 'uninstalled', 'unlinked', 'upgraded' ]
|
2020-03-09 10:11:07 +01:00
|
|
|
default: present
|
2020-03-24 09:29:03 +01:00
|
|
|
type: str
|
2020-03-09 10:11:07 +01:00
|
|
|
update_homebrew:
|
|
|
|
description:
|
2020-03-24 09:29:03 +01:00
|
|
|
- update homebrew itself first.
|
2020-03-09 10:11:07 +01:00
|
|
|
type: bool
|
2022-08-24 19:59:13 +02:00
|
|
|
default: false
|
2020-03-09 10:11:07 +01:00
|
|
|
upgrade_all:
|
|
|
|
description:
|
2020-03-24 09:29:03 +01:00
|
|
|
- upgrade all homebrew packages.
|
2020-03-09 10:11:07 +01:00
|
|
|
type: bool
|
2022-08-24 19:59:13 +02:00
|
|
|
default: false
|
2020-03-09 10:11:07 +01:00
|
|
|
aliases: ['upgrade']
|
|
|
|
install_options:
|
|
|
|
description:
|
2020-03-24 09:29:03 +01:00
|
|
|
- options flags to install a package.
|
2020-03-09 10:11:07 +01:00
|
|
|
aliases: ['options']
|
2020-03-24 09:29:03 +01:00
|
|
|
type: list
|
|
|
|
elements: str
|
2020-03-27 11:11:27 +01:00
|
|
|
upgrade_options:
|
|
|
|
description:
|
|
|
|
- Option flags to upgrade.
|
|
|
|
type: list
|
|
|
|
elements: str
|
2020-06-13 15:01:19 +02:00
|
|
|
version_added: '0.2.0'
|
2020-03-09 10:11:07 +01:00
|
|
|
notes:
|
2022-06-22 22:43:48 +02:00
|
|
|
- When used with a C(loop:) each package will be processed individually,
|
2023-06-15 15:47:00 +02:00
|
|
|
it is much more efficient to pass the list directly to the O(name) option.
|
2020-03-09 10:11:07 +01:00
|
|
|
'''
|
2020-04-01 14:02:25 +02:00
|
|
|
|
2020-03-09 10:11:07 +01:00
|
|
|
EXAMPLES = '''
|
2021-01-26 06:56:24 +01:00
|
|
|
# Install formula foo with 'brew' in default path
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2020-03-09 10:11:07 +01:00
|
|
|
name: foo
|
|
|
|
state: present
|
|
|
|
|
2023-06-15 15:47:00 +02:00
|
|
|
# Install formula foo with 'brew' in alternate path (/my/other/location/bin)
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2020-03-09 10:11:07 +01:00
|
|
|
name: foo
|
|
|
|
path: /my/other/location/bin
|
|
|
|
state: present
|
|
|
|
|
|
|
|
# Update homebrew first and install formula foo with 'brew' in default path
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2020-03-09 10:11:07 +01:00
|
|
|
name: foo
|
|
|
|
state: present
|
2022-08-24 19:59:13 +02:00
|
|
|
update_homebrew: true
|
2020-03-09 10:11:07 +01:00
|
|
|
|
|
|
|
# Update homebrew first and upgrade formula foo to latest available with 'brew' in default path
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2020-03-09 10:11:07 +01:00
|
|
|
name: foo
|
|
|
|
state: latest
|
2022-08-24 19:59:13 +02:00
|
|
|
update_homebrew: true
|
2020-03-09 10:11:07 +01:00
|
|
|
|
|
|
|
# Update homebrew and upgrade all packages
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2022-08-24 19:59:13 +02:00
|
|
|
update_homebrew: true
|
|
|
|
upgrade_all: true
|
2020-03-09 10:11:07 +01:00
|
|
|
|
|
|
|
# Miscellaneous other examples
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2020-03-09 10:11:07 +01:00
|
|
|
name: foo
|
|
|
|
state: head
|
|
|
|
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2020-03-09 10:11:07 +01:00
|
|
|
name: foo
|
|
|
|
state: linked
|
|
|
|
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2020-03-09 10:11:07 +01:00
|
|
|
name: foo
|
|
|
|
state: absent
|
|
|
|
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2020-03-09 10:11:07 +01:00
|
|
|
name: foo,bar
|
|
|
|
state: absent
|
|
|
|
|
2020-07-13 21:50:31 +02:00
|
|
|
- community.general.homebrew:
|
2020-03-09 10:11:07 +01:00
|
|
|
name: foo
|
|
|
|
state: present
|
|
|
|
install_options: with-baz,enable-debug
|
2020-03-27 11:11:27 +01:00
|
|
|
|
2021-05-08 11:58:55 +02:00
|
|
|
- name: Install formula foo with 'brew' from cask
|
|
|
|
community.general.homebrew:
|
|
|
|
name: homebrew/cask/foo
|
|
|
|
state: present
|
|
|
|
|
2021-10-18 22:43:57 +02:00
|
|
|
- name: Use ignore-pinned option while upgrading all
|
2020-07-13 21:50:31 +02:00
|
|
|
community.general.homebrew:
|
2022-08-24 19:59:13 +02:00
|
|
|
upgrade_all: true
|
2021-10-18 22:43:57 +02:00
|
|
|
upgrade_options: ignore-pinned
|
2020-03-09 10:11:07 +01:00
|
|
|
'''
|
|
|
|
|
2020-04-01 14:02:25 +02:00
|
|
|
RETURN = '''
|
|
|
|
msg:
|
|
|
|
description: if the cache was updated or not
|
|
|
|
returned: always
|
|
|
|
type: str
|
|
|
|
sample: "Changed: 0, Unchanged: 2"
|
|
|
|
unchanged_pkgs:
|
|
|
|
description:
|
|
|
|
- List of package names which are unchanged after module run
|
|
|
|
returned: success
|
|
|
|
type: list
|
|
|
|
sample: ["awscli", "ag"]
|
2020-06-13 15:01:19 +02:00
|
|
|
version_added: '0.2.0'
|
2020-04-01 14:02:25 +02:00
|
|
|
changed_pkgs:
|
|
|
|
description:
|
|
|
|
- List of package names which are changed after module run
|
|
|
|
returned: success
|
|
|
|
type: list
|
|
|
|
sample: ['git', 'git-cola']
|
2020-06-13 15:01:19 +02:00
|
|
|
version_added: '0.2.0'
|
2020-04-01 14:02:25 +02:00
|
|
|
'''
|
|
|
|
|
2024-01-27 10:37:13 +01:00
|
|
|
import json
|
2020-03-09 10:11:07 +01:00
|
|
|
import os.path
|
|
|
|
import re
|
|
|
|
|
|
|
|
from ansible.module_utils.basic import AnsibleModule
|
|
|
|
from ansible.module_utils.six import iteritems, string_types
|
|
|
|
|
|
|
|
|
|
|
|
# exceptions -------------------------------------------------------------- {{{
|
|
|
|
class HomebrewException(Exception):
|
|
|
|
pass
|
|
|
|
# /exceptions ------------------------------------------------------------- }}}
|
|
|
|
|
|
|
|
|
|
|
|
# utils ------------------------------------------------------------------- {{{
|
2020-10-27 22:46:49 +01:00
|
|
|
def _create_regex_group_complement(s):
|
2020-03-09 10:11:07 +01:00
|
|
|
lines = (line.strip() for line in s.split('\n') if line.strip())
|
|
|
|
chars = filter(None, (line.split('#')[0].strip() for line in lines))
|
|
|
|
group = r'[^' + r''.join(chars) + r']'
|
|
|
|
return re.compile(group)
|
2024-01-27 10:37:13 +01:00
|
|
|
|
|
|
|
|
|
|
|
def _check_package_in_json(json_output, package_type):
|
|
|
|
return bool(json_output.get(package_type, []) and json_output[package_type][0].get("installed"))
|
2020-03-09 10:11:07 +01:00
|
|
|
# /utils ------------------------------------------------------------------ }}}
|
|
|
|
|
|
|
|
|
|
|
|
class Homebrew(object):
|
|
|
|
'''A class to manage Homebrew packages.'''
|
|
|
|
|
|
|
|
# class regexes ------------------------------------------------ {{{
|
|
|
|
VALID_PATH_CHARS = r'''
|
|
|
|
\w # alphanumeric characters (i.e., [a-zA-Z0-9_])
|
|
|
|
\s # spaces
|
|
|
|
: # colons
|
|
|
|
{sep} # the OS-specific path separator
|
|
|
|
. # dots
|
2020-10-27 22:46:49 +01:00
|
|
|
\- # dashes
|
2020-03-09 10:11:07 +01:00
|
|
|
'''.format(sep=os.path.sep)
|
|
|
|
|
|
|
|
VALID_BREW_PATH_CHARS = r'''
|
|
|
|
\w # alphanumeric characters (i.e., [a-zA-Z0-9_])
|
|
|
|
\s # spaces
|
|
|
|
{sep} # the OS-specific path separator
|
|
|
|
. # dots
|
2020-10-27 22:46:49 +01:00
|
|
|
\- # dashes
|
2020-03-09 10:11:07 +01:00
|
|
|
'''.format(sep=os.path.sep)
|
|
|
|
|
|
|
|
VALID_PACKAGE_CHARS = r'''
|
|
|
|
\w # alphanumeric characters (i.e., [a-zA-Z0-9_])
|
|
|
|
. # dots
|
|
|
|
/ # slash (for taps)
|
|
|
|
\+ # plusses
|
2020-10-27 22:46:49 +01:00
|
|
|
\- # dashes
|
2020-03-09 10:11:07 +01:00
|
|
|
: # colons (for URLs)
|
|
|
|
@ # at-sign
|
|
|
|
'''
|
|
|
|
|
2020-10-27 22:46:49 +01:00
|
|
|
INVALID_PATH_REGEX = _create_regex_group_complement(VALID_PATH_CHARS)
|
|
|
|
INVALID_BREW_PATH_REGEX = _create_regex_group_complement(VALID_BREW_PATH_CHARS)
|
|
|
|
INVALID_PACKAGE_REGEX = _create_regex_group_complement(VALID_PACKAGE_CHARS)
|
2020-03-09 10:11:07 +01:00
|
|
|
# /class regexes ----------------------------------------------- }}}
|
|
|
|
|
|
|
|
# class validations -------------------------------------------- {{{
|
|
|
|
@classmethod
|
|
|
|
def valid_path(cls, path):
|
|
|
|
'''
|
|
|
|
`path` must be one of:
|
|
|
|
- list of paths
|
|
|
|
- a string containing only:
|
|
|
|
- alphanumeric characters
|
|
|
|
- dashes
|
|
|
|
- dots
|
|
|
|
- spaces
|
|
|
|
- colons
|
|
|
|
- os.path.sep
|
|
|
|
'''
|
|
|
|
|
|
|
|
if isinstance(path, string_types):
|
|
|
|
return not cls.INVALID_PATH_REGEX.search(path)
|
|
|
|
|
|
|
|
try:
|
|
|
|
iter(path)
|
|
|
|
except TypeError:
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
paths = path
|
|
|
|
return all(cls.valid_brew_path(path_) for path_ in paths)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def valid_brew_path(cls, brew_path):
|
|
|
|
'''
|
|
|
|
`brew_path` must be one of:
|
|
|
|
- None
|
|
|
|
- a string containing only:
|
|
|
|
- alphanumeric characters
|
|
|
|
- dashes
|
|
|
|
- dots
|
|
|
|
- spaces
|
|
|
|
- os.path.sep
|
|
|
|
'''
|
|
|
|
|
|
|
|
if brew_path is None:
|
|
|
|
return True
|
|
|
|
|
|
|
|
return (
|
|
|
|
isinstance(brew_path, string_types)
|
|
|
|
and not cls.INVALID_BREW_PATH_REGEX.search(brew_path)
|
|
|
|
)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def valid_package(cls, package):
|
|
|
|
'''A valid package is either None or alphanumeric.'''
|
|
|
|
|
|
|
|
if package is None:
|
|
|
|
return True
|
|
|
|
|
|
|
|
return (
|
|
|
|
isinstance(package, string_types)
|
|
|
|
and not cls.INVALID_PACKAGE_REGEX.search(package)
|
|
|
|
)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def valid_state(cls, state):
|
|
|
|
'''
|
|
|
|
A valid state is one of:
|
|
|
|
- None
|
|
|
|
- installed
|
|
|
|
- upgraded
|
|
|
|
- head
|
|
|
|
- linked
|
|
|
|
- unlinked
|
|
|
|
- absent
|
|
|
|
'''
|
|
|
|
|
|
|
|
if state is None:
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return (
|
|
|
|
isinstance(state, string_types)
|
|
|
|
and state.lower() in (
|
|
|
|
'installed',
|
|
|
|
'upgraded',
|
|
|
|
'head',
|
|
|
|
'linked',
|
|
|
|
'unlinked',
|
|
|
|
'absent',
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def valid_module(cls, module):
|
|
|
|
'''A valid module is an instance of AnsibleModule.'''
|
|
|
|
|
|
|
|
return isinstance(module, AnsibleModule)
|
|
|
|
|
|
|
|
# /class validations ------------------------------------------- }}}
|
|
|
|
|
|
|
|
# class properties --------------------------------------------- {{{
|
|
|
|
@property
|
|
|
|
def module(self):
|
|
|
|
return self._module
|
|
|
|
|
|
|
|
@module.setter
|
|
|
|
def module(self, module):
|
|
|
|
if not self.valid_module(module):
|
|
|
|
self._module = None
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid module: {0}.'.format(module)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
else:
|
|
|
|
self._module = module
|
|
|
|
return module
|
|
|
|
|
|
|
|
@property
|
|
|
|
def path(self):
|
|
|
|
return self._path
|
|
|
|
|
|
|
|
@path.setter
|
|
|
|
def path(self, path):
|
|
|
|
if not self.valid_path(path):
|
|
|
|
self._path = []
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid path: {0}.'.format(path)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
else:
|
|
|
|
if isinstance(path, string_types):
|
|
|
|
self._path = path.split(':')
|
|
|
|
else:
|
|
|
|
self._path = path
|
|
|
|
|
|
|
|
return path
|
|
|
|
|
|
|
|
@property
|
|
|
|
def brew_path(self):
|
|
|
|
return self._brew_path
|
|
|
|
|
|
|
|
@brew_path.setter
|
|
|
|
def brew_path(self, brew_path):
|
|
|
|
if not self.valid_brew_path(brew_path):
|
|
|
|
self._brew_path = None
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid brew_path: {0}.'.format(brew_path)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
else:
|
|
|
|
self._brew_path = brew_path
|
|
|
|
return brew_path
|
|
|
|
|
|
|
|
@property
|
|
|
|
def params(self):
|
|
|
|
return self._params
|
|
|
|
|
|
|
|
@params.setter
|
|
|
|
def params(self, params):
|
|
|
|
self._params = self.module.params
|
|
|
|
return self._params
|
|
|
|
|
|
|
|
@property
|
|
|
|
def current_package(self):
|
|
|
|
return self._current_package
|
|
|
|
|
|
|
|
@current_package.setter
|
|
|
|
def current_package(self, package):
|
|
|
|
if not self.valid_package(package):
|
|
|
|
self._current_package = None
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid package: {0}.'.format(package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
else:
|
|
|
|
self._current_package = package
|
|
|
|
return package
|
|
|
|
# /class properties -------------------------------------------- }}}
|
|
|
|
|
|
|
|
def __init__(self, module, path, packages=None, state=None,
|
|
|
|
update_homebrew=False, upgrade_all=False,
|
2020-03-27 11:11:27 +01:00
|
|
|
install_options=None, upgrade_options=None):
|
2020-03-09 10:11:07 +01:00
|
|
|
if not install_options:
|
|
|
|
install_options = list()
|
2020-03-27 11:11:27 +01:00
|
|
|
if not upgrade_options:
|
|
|
|
upgrade_options = list()
|
2020-03-09 10:11:07 +01:00
|
|
|
self._setup_status_vars()
|
|
|
|
self._setup_instance_vars(module=module, path=path, packages=packages,
|
|
|
|
state=state, update_homebrew=update_homebrew,
|
|
|
|
upgrade_all=upgrade_all,
|
2020-03-27 11:11:27 +01:00
|
|
|
install_options=install_options,
|
|
|
|
upgrade_options=upgrade_options,)
|
2020-03-09 10:11:07 +01:00
|
|
|
|
|
|
|
self._prep()
|
|
|
|
|
|
|
|
# prep --------------------------------------------------------- {{{
|
|
|
|
def _setup_status_vars(self):
|
|
|
|
self.failed = False
|
|
|
|
self.changed = False
|
|
|
|
self.changed_count = 0
|
|
|
|
self.unchanged_count = 0
|
2020-04-01 14:02:25 +02:00
|
|
|
self.changed_pkgs = []
|
|
|
|
self.unchanged_pkgs = []
|
2020-03-09 10:11:07 +01:00
|
|
|
self.message = ''
|
|
|
|
|
|
|
|
def _setup_instance_vars(self, **kwargs):
|
|
|
|
for key, val in iteritems(kwargs):
|
|
|
|
setattr(self, key, val)
|
|
|
|
|
|
|
|
def _prep(self):
|
|
|
|
self._prep_brew_path()
|
|
|
|
|
|
|
|
def _prep_brew_path(self):
|
|
|
|
if not self.module:
|
|
|
|
self.brew_path = None
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'AnsibleModule not set.'
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
self.brew_path = self.module.get_bin_path(
|
|
|
|
'brew',
|
|
|
|
required=True,
|
|
|
|
opt_dirs=self.path,
|
|
|
|
)
|
|
|
|
if not self.brew_path:
|
|
|
|
self.brew_path = None
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Unable to locate homebrew executable.'
|
|
|
|
raise HomebrewException('Unable to locate homebrew executable.')
|
|
|
|
|
|
|
|
return self.brew_path
|
|
|
|
|
|
|
|
def _status(self):
|
|
|
|
return (self.failed, self.changed, self.message)
|
|
|
|
# /prep -------------------------------------------------------- }}}
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
try:
|
|
|
|
self._run()
|
|
|
|
except HomebrewException:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if not self.failed and (self.changed_count + self.unchanged_count > 1):
|
|
|
|
self.message = "Changed: %d, Unchanged: %d" % (
|
|
|
|
self.changed_count,
|
|
|
|
self.unchanged_count,
|
|
|
|
)
|
|
|
|
(failed, changed, message) = self._status()
|
|
|
|
|
|
|
|
return (failed, changed, message)
|
|
|
|
|
|
|
|
# checks ------------------------------------------------------- {{{
|
|
|
|
def _current_package_is_installed(self):
|
|
|
|
if not self.valid_package(self.current_package):
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid package: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
cmd = [
|
|
|
|
"{brew_path}".format(brew_path=self.brew_path),
|
|
|
|
"info",
|
2024-01-27 10:37:13 +01:00
|
|
|
"--json=v2",
|
2020-03-09 10:11:07 +01:00
|
|
|
self.current_package,
|
|
|
|
]
|
|
|
|
rc, out, err = self.module.run_command(cmd)
|
2024-03-09 13:42:34 +01:00
|
|
|
if err:
|
|
|
|
self.failed = True
|
|
|
|
self.message = err.strip()
|
|
|
|
raise HomebrewException(self.message)
|
2024-01-27 10:37:13 +01:00
|
|
|
data = json.loads(out)
|
|
|
|
|
|
|
|
return _check_package_in_json(data, "formulae") or _check_package_in_json(data, "casks")
|
2020-03-09 10:11:07 +01:00
|
|
|
|
|
|
|
def _current_package_is_outdated(self):
|
|
|
|
if not self.valid_package(self.current_package):
|
|
|
|
return False
|
|
|
|
|
|
|
|
rc, out, err = self.module.run_command([
|
|
|
|
self.brew_path,
|
|
|
|
'outdated',
|
|
|
|
self.current_package,
|
|
|
|
])
|
|
|
|
|
|
|
|
return rc != 0
|
|
|
|
|
|
|
|
def _current_package_is_installed_from_head(self):
|
|
|
|
if not Homebrew.valid_package(self.current_package):
|
|
|
|
return False
|
|
|
|
elif not self._current_package_is_installed():
|
|
|
|
return False
|
|
|
|
|
|
|
|
rc, out, err = self.module.run_command([
|
|
|
|
self.brew_path,
|
|
|
|
'info',
|
|
|
|
self.current_package,
|
|
|
|
])
|
|
|
|
|
|
|
|
try:
|
|
|
|
version_info = [line for line in out.split('\n') if line][0]
|
|
|
|
except IndexError:
|
|
|
|
return False
|
|
|
|
|
|
|
|
return version_info.split(' ')[-1] == 'HEAD'
|
|
|
|
# /checks ------------------------------------------------------ }}}
|
|
|
|
|
|
|
|
# commands ----------------------------------------------------- {{{
|
|
|
|
def _run(self):
|
|
|
|
if self.update_homebrew:
|
|
|
|
self._update_homebrew()
|
|
|
|
|
|
|
|
if self.upgrade_all:
|
|
|
|
self._upgrade_all()
|
|
|
|
|
|
|
|
if self.packages:
|
|
|
|
if self.state == 'installed':
|
|
|
|
return self._install_packages()
|
|
|
|
elif self.state == 'upgraded':
|
|
|
|
return self._upgrade_packages()
|
|
|
|
elif self.state == 'head':
|
|
|
|
return self._install_packages()
|
|
|
|
elif self.state == 'linked':
|
|
|
|
return self._link_packages()
|
|
|
|
elif self.state == 'unlinked':
|
|
|
|
return self._unlink_packages()
|
|
|
|
elif self.state == 'absent':
|
|
|
|
return self._uninstall_packages()
|
|
|
|
|
|
|
|
# updated -------------------------------- {{{
|
|
|
|
def _update_homebrew(self):
|
|
|
|
if self.module.check_mode:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'Homebrew would be updated.'
|
|
|
|
raise HomebrewException(self.message)
|
2020-04-01 14:02:25 +02:00
|
|
|
|
2020-03-09 10:11:07 +01:00
|
|
|
rc, out, err = self.module.run_command([
|
|
|
|
self.brew_path,
|
|
|
|
'update',
|
|
|
|
])
|
|
|
|
if rc == 0:
|
|
|
|
if out and isinstance(out, string_types):
|
|
|
|
already_updated = any(
|
|
|
|
re.search(r'Already up-to-date.', s.strip(), re.IGNORECASE)
|
|
|
|
for s in out.split('\n')
|
|
|
|
if s
|
|
|
|
)
|
|
|
|
if not already_updated:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'Homebrew updated successfully.'
|
|
|
|
else:
|
|
|
|
self.message = 'Homebrew already up-to-date.'
|
|
|
|
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
self.failed = True
|
|
|
|
self.message = err.strip()
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
# /updated ------------------------------- }}}
|
|
|
|
|
|
|
|
# _upgrade_all --------------------------- {{{
|
|
|
|
def _upgrade_all(self):
|
|
|
|
if self.module.check_mode:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'Homebrew packages would be upgraded.'
|
|
|
|
raise HomebrewException(self.message)
|
2020-03-27 11:11:27 +01:00
|
|
|
cmd = [self.brew_path, 'upgrade'] + self.upgrade_options
|
|
|
|
|
|
|
|
rc, out, err = self.module.run_command(cmd)
|
2020-03-09 10:11:07 +01:00
|
|
|
if rc == 0:
|
|
|
|
if not out:
|
|
|
|
self.message = 'Homebrew packages already upgraded.'
|
|
|
|
|
|
|
|
else:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'Homebrew upgraded.'
|
|
|
|
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
self.failed = True
|
|
|
|
self.message = err.strip()
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
# /_upgrade_all -------------------------- }}}
|
|
|
|
|
|
|
|
# installed ------------------------------ {{{
|
|
|
|
def _install_current_package(self):
|
|
|
|
if not self.valid_package(self.current_package):
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid package: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
if self._current_package_is_installed():
|
|
|
|
self.unchanged_count += 1
|
2020-04-01 14:02:25 +02:00
|
|
|
self.unchanged_pkgs.append(self.current_package)
|
2020-03-09 10:11:07 +01:00
|
|
|
self.message = 'Package already installed: {0}'.format(
|
|
|
|
self.current_package,
|
|
|
|
)
|
|
|
|
return True
|
|
|
|
|
|
|
|
if self.module.check_mode:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package would be installed: {0}'.format(
|
|
|
|
self.current_package
|
|
|
|
)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
if self.state == 'head':
|
|
|
|
head = '--HEAD'
|
|
|
|
else:
|
|
|
|
head = None
|
|
|
|
|
|
|
|
opts = (
|
|
|
|
[self.brew_path, 'install']
|
|
|
|
+ self.install_options
|
|
|
|
+ [self.current_package, head]
|
|
|
|
)
|
|
|
|
cmd = [opt for opt in opts if opt]
|
|
|
|
rc, out, err = self.module.run_command(cmd)
|
|
|
|
|
|
|
|
if self._current_package_is_installed():
|
|
|
|
self.changed_count += 1
|
2020-04-01 14:02:25 +02:00
|
|
|
self.changed_pkgs.append(self.current_package)
|
2020-03-09 10:11:07 +01:00
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package installed: {0}'.format(self.current_package)
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
self.failed = True
|
|
|
|
self.message = err.strip()
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
def _install_packages(self):
|
|
|
|
for package in self.packages:
|
|
|
|
self.current_package = package
|
|
|
|
self._install_current_package()
|
|
|
|
|
|
|
|
return True
|
|
|
|
# /installed ----------------------------- }}}
|
|
|
|
|
|
|
|
# upgraded ------------------------------- {{{
|
|
|
|
def _upgrade_current_package(self):
|
|
|
|
command = 'upgrade'
|
|
|
|
|
|
|
|
if not self.valid_package(self.current_package):
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid package: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
if not self._current_package_is_installed():
|
|
|
|
command = 'install'
|
|
|
|
|
|
|
|
if self._current_package_is_installed() and not self._current_package_is_outdated():
|
|
|
|
self.message = 'Package is already upgraded: {0}'.format(
|
|
|
|
self.current_package,
|
|
|
|
)
|
|
|
|
self.unchanged_count += 1
|
2020-04-01 14:02:25 +02:00
|
|
|
self.unchanged_pkgs.append(self.current_package)
|
2020-03-09 10:11:07 +01:00
|
|
|
return True
|
|
|
|
|
|
|
|
if self.module.check_mode:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package would be upgraded: {0}'.format(
|
|
|
|
self.current_package
|
|
|
|
)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
opts = (
|
|
|
|
[self.brew_path, command]
|
|
|
|
+ self.install_options
|
|
|
|
+ [self.current_package]
|
|
|
|
)
|
|
|
|
cmd = [opt for opt in opts if opt]
|
|
|
|
rc, out, err = self.module.run_command(cmd)
|
|
|
|
|
|
|
|
if self._current_package_is_installed() and not self._current_package_is_outdated():
|
|
|
|
self.changed_count += 1
|
2020-04-01 14:02:25 +02:00
|
|
|
self.changed_pkgs.append(self.current_package)
|
2020-03-09 10:11:07 +01:00
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package upgraded: {0}'.format(self.current_package)
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
self.failed = True
|
|
|
|
self.message = err.strip()
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
def _upgrade_all_packages(self):
|
|
|
|
opts = (
|
|
|
|
[self.brew_path, 'upgrade']
|
|
|
|
+ self.install_options
|
|
|
|
)
|
|
|
|
cmd = [opt for opt in opts if opt]
|
|
|
|
rc, out, err = self.module.run_command(cmd)
|
|
|
|
|
|
|
|
if rc == 0:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'All packages upgraded.'
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
self.failed = True
|
|
|
|
self.message = err.strip()
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
def _upgrade_packages(self):
|
|
|
|
if not self.packages:
|
|
|
|
self._upgrade_all_packages()
|
|
|
|
else:
|
|
|
|
for package in self.packages:
|
|
|
|
self.current_package = package
|
|
|
|
self._upgrade_current_package()
|
|
|
|
return True
|
|
|
|
# /upgraded ------------------------------ }}}
|
|
|
|
|
|
|
|
# uninstalled ---------------------------- {{{
|
|
|
|
def _uninstall_current_package(self):
|
|
|
|
if not self.valid_package(self.current_package):
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid package: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
if not self._current_package_is_installed():
|
|
|
|
self.unchanged_count += 1
|
2020-04-01 14:02:25 +02:00
|
|
|
self.unchanged_pkgs.append(self.current_package)
|
2020-03-09 10:11:07 +01:00
|
|
|
self.message = 'Package already uninstalled: {0}'.format(
|
|
|
|
self.current_package,
|
|
|
|
)
|
|
|
|
return True
|
|
|
|
|
|
|
|
if self.module.check_mode:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package would be uninstalled: {0}'.format(
|
|
|
|
self.current_package
|
|
|
|
)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
opts = (
|
|
|
|
[self.brew_path, 'uninstall', '--force']
|
|
|
|
+ self.install_options
|
|
|
|
+ [self.current_package]
|
|
|
|
)
|
|
|
|
cmd = [opt for opt in opts if opt]
|
|
|
|
rc, out, err = self.module.run_command(cmd)
|
|
|
|
|
|
|
|
if not self._current_package_is_installed():
|
|
|
|
self.changed_count += 1
|
2020-04-01 14:02:25 +02:00
|
|
|
self.changed_pkgs.append(self.current_package)
|
2020-03-09 10:11:07 +01:00
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package uninstalled: {0}'.format(self.current_package)
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
self.failed = True
|
|
|
|
self.message = err.strip()
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
def _uninstall_packages(self):
|
|
|
|
for package in self.packages:
|
|
|
|
self.current_package = package
|
|
|
|
self._uninstall_current_package()
|
|
|
|
|
|
|
|
return True
|
|
|
|
# /uninstalled ----------------------------- }}}
|
|
|
|
|
|
|
|
# linked --------------------------------- {{{
|
|
|
|
def _link_current_package(self):
|
|
|
|
if not self.valid_package(self.current_package):
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid package: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
if not self._current_package_is_installed():
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Package not installed: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
if self.module.check_mode:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package would be linked: {0}'.format(
|
|
|
|
self.current_package
|
|
|
|
)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
opts = (
|
|
|
|
[self.brew_path, 'link']
|
|
|
|
+ self.install_options
|
|
|
|
+ [self.current_package]
|
|
|
|
)
|
|
|
|
cmd = [opt for opt in opts if opt]
|
|
|
|
rc, out, err = self.module.run_command(cmd)
|
|
|
|
|
|
|
|
if rc == 0:
|
|
|
|
self.changed_count += 1
|
2020-04-01 14:02:25 +02:00
|
|
|
self.changed_pkgs.append(self.current_package)
|
2020-03-09 10:11:07 +01:00
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package linked: {0}'.format(self.current_package)
|
|
|
|
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Package could not be linked: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
def _link_packages(self):
|
|
|
|
for package in self.packages:
|
|
|
|
self.current_package = package
|
|
|
|
self._link_current_package()
|
|
|
|
|
|
|
|
return True
|
|
|
|
# /linked -------------------------------- }}}
|
|
|
|
|
|
|
|
# unlinked ------------------------------- {{{
|
|
|
|
def _unlink_current_package(self):
|
|
|
|
if not self.valid_package(self.current_package):
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Invalid package: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
if not self._current_package_is_installed():
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Package not installed: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
if self.module.check_mode:
|
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package would be unlinked: {0}'.format(
|
|
|
|
self.current_package
|
|
|
|
)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
opts = (
|
|
|
|
[self.brew_path, 'unlink']
|
|
|
|
+ self.install_options
|
|
|
|
+ [self.current_package]
|
|
|
|
)
|
|
|
|
cmd = [opt for opt in opts if opt]
|
|
|
|
rc, out, err = self.module.run_command(cmd)
|
|
|
|
|
|
|
|
if rc == 0:
|
|
|
|
self.changed_count += 1
|
2020-04-01 14:02:25 +02:00
|
|
|
self.changed_pkgs.append(self.current_package)
|
2020-03-09 10:11:07 +01:00
|
|
|
self.changed = True
|
|
|
|
self.message = 'Package unlinked: {0}'.format(self.current_package)
|
|
|
|
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
self.failed = True
|
|
|
|
self.message = 'Package could not be unlinked: {0}.'.format(self.current_package)
|
|
|
|
raise HomebrewException(self.message)
|
|
|
|
|
|
|
|
def _unlink_packages(self):
|
|
|
|
for package in self.packages:
|
|
|
|
self.current_package = package
|
|
|
|
self._unlink_current_package()
|
|
|
|
|
|
|
|
return True
|
|
|
|
# /unlinked ------------------------------ }}}
|
|
|
|
# /commands ---------------------------------------------------- }}}
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
module = AnsibleModule(
|
|
|
|
argument_spec=dict(
|
|
|
|
name=dict(
|
|
|
|
aliases=["pkg", "package", "formula"],
|
|
|
|
required=False,
|
|
|
|
type='list',
|
|
|
|
elements='str',
|
|
|
|
),
|
|
|
|
path=dict(
|
2022-09-08 07:46:10 +02:00
|
|
|
default="/usr/local/bin:/opt/homebrew/bin:/home/linuxbrew/.linuxbrew/bin",
|
2020-03-09 10:11:07 +01:00
|
|
|
required=False,
|
|
|
|
type='path',
|
|
|
|
),
|
|
|
|
state=dict(
|
|
|
|
default="present",
|
|
|
|
choices=[
|
|
|
|
"present", "installed",
|
|
|
|
"latest", "upgraded", "head",
|
|
|
|
"linked", "unlinked",
|
|
|
|
"absent", "removed", "uninstalled",
|
|
|
|
],
|
|
|
|
),
|
|
|
|
update_homebrew=dict(
|
|
|
|
default=False,
|
|
|
|
type='bool',
|
|
|
|
),
|
|
|
|
upgrade_all=dict(
|
|
|
|
default=False,
|
|
|
|
aliases=["upgrade"],
|
|
|
|
type='bool',
|
|
|
|
),
|
|
|
|
install_options=dict(
|
|
|
|
default=None,
|
|
|
|
aliases=['options'],
|
|
|
|
type='list',
|
2020-03-24 09:29:03 +01:00
|
|
|
elements='str',
|
2020-03-27 11:11:27 +01:00
|
|
|
),
|
|
|
|
upgrade_options=dict(
|
|
|
|
default=None,
|
|
|
|
type='list',
|
|
|
|
elements='str',
|
2020-03-09 10:11:07 +01:00
|
|
|
)
|
|
|
|
),
|
|
|
|
supports_check_mode=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
module.run_command_environ_update = dict(LANG='C', LC_ALL='C', LC_MESSAGES='C', LC_CTYPE='C')
|
|
|
|
|
|
|
|
p = module.params
|
|
|
|
|
|
|
|
if p['name']:
|
|
|
|
packages = p['name']
|
|
|
|
else:
|
|
|
|
packages = None
|
|
|
|
|
|
|
|
path = p['path']
|
|
|
|
if path:
|
|
|
|
path = path.split(':')
|
|
|
|
|
|
|
|
state = p['state']
|
|
|
|
if state in ('present', 'installed'):
|
|
|
|
state = 'installed'
|
|
|
|
if state in ('head', ):
|
|
|
|
state = 'head'
|
|
|
|
if state in ('latest', 'upgraded'):
|
|
|
|
state = 'upgraded'
|
|
|
|
if state == 'linked':
|
|
|
|
state = 'linked'
|
|
|
|
if state == 'unlinked':
|
|
|
|
state = 'unlinked'
|
|
|
|
if state in ('absent', 'removed', 'uninstalled'):
|
|
|
|
state = 'absent'
|
|
|
|
|
|
|
|
update_homebrew = p['update_homebrew']
|
2020-04-05 10:23:56 +02:00
|
|
|
if not update_homebrew:
|
|
|
|
module.run_command_environ_update.update(
|
|
|
|
dict(HOMEBREW_NO_AUTO_UPDATE="True")
|
|
|
|
)
|
2020-03-09 10:11:07 +01:00
|
|
|
upgrade_all = p['upgrade_all']
|
|
|
|
p['install_options'] = p['install_options'] or []
|
|
|
|
install_options = ['--{0}'.format(install_option)
|
|
|
|
for install_option in p['install_options']]
|
|
|
|
|
2020-03-27 11:11:27 +01:00
|
|
|
p['upgrade_options'] = p['upgrade_options'] or []
|
|
|
|
upgrade_options = ['--{0}'.format(upgrade_option)
|
|
|
|
for upgrade_option in p['upgrade_options']]
|
2020-03-09 10:11:07 +01:00
|
|
|
brew = Homebrew(module=module, path=path, packages=packages,
|
|
|
|
state=state, update_homebrew=update_homebrew,
|
2020-03-27 11:11:27 +01:00
|
|
|
upgrade_all=upgrade_all, install_options=install_options,
|
|
|
|
upgrade_options=upgrade_options)
|
2020-03-09 10:11:07 +01:00
|
|
|
(failed, changed, message) = brew.run()
|
2020-04-01 14:02:25 +02:00
|
|
|
changed_pkgs = brew.changed_pkgs
|
|
|
|
unchanged_pkgs = brew.unchanged_pkgs
|
|
|
|
|
2020-03-09 10:11:07 +01:00
|
|
|
if failed:
|
|
|
|
module.fail_json(msg=message)
|
2020-04-01 14:02:25 +02:00
|
|
|
module.exit_json(
|
|
|
|
changed=changed,
|
|
|
|
msg=message,
|
|
|
|
unchanged_pkgs=unchanged_pkgs,
|
|
|
|
changed_pkgs=changed_pkgs
|
|
|
|
)
|
2020-03-09 10:11:07 +01:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|