2022-04-03 11:04:27 +02:00
|
|
|
#!/usr/bin/python
|
|
|
|
# -*- coding: utf-8 -*-
|
2023-10-19 11:10:04 +02:00
|
|
|
# Copyright (c) 2015, Mathew Davies <thepixeldeveloper@googlemail.com>
|
|
|
|
# Copyright (c) 2017, Sam Doran <sdoran@redhat.com>
|
|
|
|
# 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
|
2022-04-03 11:04:27 +02:00
|
|
|
|
|
|
|
from __future__ import absolute_import, division, print_function
|
|
|
|
__metaclass__ = type
|
|
|
|
|
|
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: elasticsearch_plugin
|
|
|
|
short_description: Manage Elasticsearch plugins
|
|
|
|
description:
|
|
|
|
- Manages Elasticsearch plugins.
|
|
|
|
author:
|
|
|
|
- Mathew Davies (@ThePixelDeveloper)
|
|
|
|
- Sam Doran (@samdoran)
|
2023-10-19 11:10:04 +02:00
|
|
|
extends_documentation_fragment:
|
|
|
|
- community.general.attributes
|
|
|
|
attributes:
|
|
|
|
check_mode:
|
|
|
|
support: full
|
|
|
|
diff_mode:
|
|
|
|
support: none
|
2022-04-03 11:04:27 +02:00
|
|
|
options:
|
|
|
|
name:
|
|
|
|
description:
|
|
|
|
- Name of the plugin to install.
|
2023-10-19 11:10:04 +02:00
|
|
|
required: true
|
2022-04-03 11:04:27 +02:00
|
|
|
type: str
|
|
|
|
state:
|
|
|
|
description:
|
|
|
|
- Desired state of a plugin.
|
|
|
|
choices: ["present", "absent"]
|
|
|
|
default: present
|
|
|
|
type: str
|
|
|
|
src:
|
|
|
|
description:
|
|
|
|
- Optionally set the source location to retrieve the plugin from. This can be a file://
|
|
|
|
URL to install from a local file, or a remote URL. If this is not set, the plugin
|
|
|
|
location is just based on the name.
|
|
|
|
- The name parameter must match the descriptor in the plugin ZIP specified.
|
|
|
|
- Is only used if the state would change, which is solely checked based on the name
|
|
|
|
parameter. If, for example, the plugin is already installed, changing this has no
|
|
|
|
effect.
|
|
|
|
- For ES 1.x use url.
|
2023-10-19 11:10:04 +02:00
|
|
|
required: false
|
2022-04-03 11:04:27 +02:00
|
|
|
type: str
|
|
|
|
url:
|
|
|
|
description:
|
|
|
|
- Set exact URL to download the plugin from (Only works for ES 1.x).
|
|
|
|
- For ES 2.x and higher, use src.
|
2023-10-19 11:10:04 +02:00
|
|
|
required: false
|
2022-04-03 11:04:27 +02:00
|
|
|
type: str
|
|
|
|
timeout:
|
|
|
|
description:
|
|
|
|
- "Timeout setting: 30s, 1m, 1h..."
|
|
|
|
- Only valid for Elasticsearch < 5.0. This option is ignored for Elasticsearch > 5.0.
|
|
|
|
default: 1m
|
|
|
|
type: str
|
|
|
|
force:
|
|
|
|
description:
|
|
|
|
- "Force batch mode when installing plugins. This is only necessary if a plugin requires additional permissions and console detection fails."
|
2023-10-19 11:10:04 +02:00
|
|
|
default: false
|
2022-04-03 11:04:27 +02:00
|
|
|
type: bool
|
|
|
|
plugin_bin:
|
|
|
|
description:
|
|
|
|
- Location of the plugin binary. If this file is not found, the default plugin binaries will be used.
|
|
|
|
- The default changed in Ansible 2.4 to None.
|
|
|
|
type: path
|
|
|
|
plugin_dir:
|
|
|
|
description:
|
|
|
|
- Your configured plugin directory specified in Elasticsearch
|
|
|
|
default: /usr/share/elasticsearch/plugins/
|
|
|
|
type: path
|
|
|
|
proxy_host:
|
|
|
|
description:
|
|
|
|
- Proxy host to use during plugin installation
|
|
|
|
type: str
|
|
|
|
proxy_port:
|
|
|
|
description:
|
|
|
|
- Proxy port to use during plugin installation
|
|
|
|
type: str
|
|
|
|
version:
|
|
|
|
description:
|
|
|
|
- Version of the plugin to be installed.
|
|
|
|
If plugin exists with previous version, it will NOT be updated
|
|
|
|
type: str
|
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = '''
|
|
|
|
- name: Install Elasticsearch Head plugin in Elasticsearch 2.x
|
|
|
|
community.general.elasticsearch_plugin:
|
|
|
|
name: mobz/elasticsearch-head
|
|
|
|
state: present
|
|
|
|
|
|
|
|
- name: Install a specific version of Elasticsearch Head in Elasticsearch 2.x
|
|
|
|
community.general.elasticsearch_plugin:
|
|
|
|
name: mobz/elasticsearch-head
|
|
|
|
version: 2.0.0
|
|
|
|
|
|
|
|
- name: Uninstall Elasticsearch head plugin in Elasticsearch 2.x
|
|
|
|
community.general.elasticsearch_plugin:
|
|
|
|
name: mobz/elasticsearch-head
|
|
|
|
state: absent
|
|
|
|
|
|
|
|
- name: Install a specific plugin in Elasticsearch >= 5.0
|
|
|
|
community.general.elasticsearch_plugin:
|
|
|
|
name: analysis-icu
|
|
|
|
state: present
|
|
|
|
|
|
|
|
- name: Install the ingest-geoip plugin with a forced installation
|
|
|
|
community.general.elasticsearch_plugin:
|
|
|
|
name: ingest-geoip
|
|
|
|
state: present
|
2023-10-19 11:10:04 +02:00
|
|
|
force: true
|
2022-04-03 11:04:27 +02:00
|
|
|
'''
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
|
|
|
from ansible.module_utils.basic import AnsibleModule
|
|
|
|
|
|
|
|
|
|
|
|
PACKAGE_STATE_MAP = dict(
|
|
|
|
present="install",
|
|
|
|
absent="remove"
|
|
|
|
)
|
|
|
|
|
|
|
|
PLUGIN_BIN_PATHS = tuple([
|
|
|
|
'/usr/share/elasticsearch/bin/elasticsearch-plugin',
|
|
|
|
'/usr/share/elasticsearch/bin/plugin'
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
|
|
def parse_plugin_repo(string):
|
|
|
|
elements = string.split("/")
|
|
|
|
|
|
|
|
# We first consider the simplest form: pluginname
|
|
|
|
repo = elements[0]
|
|
|
|
|
|
|
|
# We consider the form: username/pluginname
|
|
|
|
if len(elements) > 1:
|
|
|
|
repo = elements[1]
|
|
|
|
|
|
|
|
# remove elasticsearch- prefix
|
|
|
|
# remove es- prefix
|
|
|
|
for string in ("elasticsearch-", "es-"):
|
|
|
|
if repo.startswith(string):
|
|
|
|
return repo[len(string):]
|
|
|
|
|
|
|
|
return repo
|
|
|
|
|
|
|
|
|
|
|
|
def is_plugin_present(plugin_name, plugin_dir):
|
|
|
|
return os.path.isdir(os.path.join(plugin_dir, plugin_name))
|
|
|
|
|
|
|
|
|
|
|
|
def parse_error(string):
|
|
|
|
reason = "ERROR: "
|
|
|
|
try:
|
|
|
|
return string[string.index(reason) + len(reason):].strip()
|
|
|
|
except ValueError:
|
|
|
|
return string
|
|
|
|
|
|
|
|
|
|
|
|
def install_plugin(module, plugin_bin, plugin_name, version, src, url, proxy_host, proxy_port, timeout, force):
|
|
|
|
cmd_args = [plugin_bin, PACKAGE_STATE_MAP["present"]]
|
|
|
|
is_old_command = (os.path.basename(plugin_bin) == 'plugin')
|
|
|
|
|
|
|
|
# Timeout and version are only valid for plugin, not elasticsearch-plugin
|
|
|
|
if is_old_command:
|
|
|
|
if timeout:
|
|
|
|
cmd_args.append("--timeout %s" % timeout)
|
|
|
|
|
|
|
|
if version:
|
|
|
|
plugin_name = plugin_name + '/' + version
|
|
|
|
cmd_args[2] = plugin_name
|
|
|
|
|
|
|
|
if proxy_host and proxy_port:
|
|
|
|
cmd_args.append("-DproxyHost=%s -DproxyPort=%s" % (proxy_host, proxy_port))
|
|
|
|
|
|
|
|
# Legacy ES 1.x
|
|
|
|
if url:
|
|
|
|
cmd_args.append("--url %s" % url)
|
|
|
|
|
|
|
|
if force:
|
|
|
|
cmd_args.append("--batch")
|
|
|
|
if src:
|
|
|
|
cmd_args.append(src)
|
|
|
|
else:
|
|
|
|
cmd_args.append(plugin_name)
|
|
|
|
|
|
|
|
cmd = " ".join(cmd_args)
|
|
|
|
|
|
|
|
if module.check_mode:
|
|
|
|
rc, out, err = 0, "check mode", ""
|
|
|
|
else:
|
|
|
|
rc, out, err = module.run_command(cmd)
|
|
|
|
|
|
|
|
if rc != 0:
|
|
|
|
reason = parse_error(out)
|
|
|
|
module.fail_json(msg="Installing plugin '%s' failed: %s" % (plugin_name, reason), err=err)
|
|
|
|
|
|
|
|
return True, cmd, out, err
|
|
|
|
|
|
|
|
|
|
|
|
def remove_plugin(module, plugin_bin, plugin_name):
|
|
|
|
cmd_args = [plugin_bin, PACKAGE_STATE_MAP["absent"], parse_plugin_repo(plugin_name)]
|
|
|
|
|
|
|
|
cmd = " ".join(cmd_args)
|
|
|
|
|
|
|
|
if module.check_mode:
|
|
|
|
rc, out, err = 0, "check mode", ""
|
|
|
|
else:
|
|
|
|
rc, out, err = module.run_command(cmd)
|
|
|
|
|
|
|
|
if rc != 0:
|
|
|
|
reason = parse_error(out)
|
|
|
|
module.fail_json(msg="Removing plugin '%s' failed: %s" % (plugin_name, reason), err=err)
|
|
|
|
|
|
|
|
return True, cmd, out, err
|
|
|
|
|
|
|
|
|
|
|
|
def get_plugin_bin(module, plugin_bin=None):
|
|
|
|
# Use the plugin_bin that was supplied first before trying other options
|
|
|
|
valid_plugin_bin = None
|
|
|
|
if plugin_bin and os.path.isfile(plugin_bin):
|
|
|
|
valid_plugin_bin = plugin_bin
|
|
|
|
|
|
|
|
else:
|
|
|
|
# Add the plugin_bin passed into the module to the top of the list of paths to test,
|
|
|
|
# testing for that binary name first before falling back to the default paths.
|
|
|
|
bin_paths = list(PLUGIN_BIN_PATHS)
|
|
|
|
if plugin_bin and plugin_bin not in bin_paths:
|
|
|
|
bin_paths.insert(0, plugin_bin)
|
|
|
|
|
|
|
|
# Get separate lists of dirs and binary names from the full paths to the
|
|
|
|
# plugin binaries.
|
|
|
|
plugin_dirs = list(set([os.path.dirname(x) for x in bin_paths]))
|
|
|
|
plugin_bins = list(set([os.path.basename(x) for x in bin_paths]))
|
|
|
|
|
|
|
|
# Check for the binary names in the default system paths as well as the path
|
|
|
|
# specified in the module arguments.
|
|
|
|
for bin_file in plugin_bins:
|
|
|
|
valid_plugin_bin = module.get_bin_path(bin_file, opt_dirs=plugin_dirs)
|
|
|
|
if valid_plugin_bin:
|
|
|
|
break
|
|
|
|
|
|
|
|
if not valid_plugin_bin:
|
|
|
|
module.fail_json(msg='%s does not exist and no other valid plugin installers were found. Make sure Elasticsearch is installed.' % plugin_bin)
|
|
|
|
|
|
|
|
return valid_plugin_bin
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
module = AnsibleModule(
|
|
|
|
argument_spec=dict(
|
|
|
|
name=dict(required=True),
|
|
|
|
state=dict(default="present", choices=list(PACKAGE_STATE_MAP.keys())),
|
|
|
|
src=dict(default=None),
|
|
|
|
url=dict(default=None),
|
|
|
|
timeout=dict(default="1m"),
|
|
|
|
force=dict(type='bool', default=False),
|
|
|
|
plugin_bin=dict(type="path"),
|
|
|
|
plugin_dir=dict(default="/usr/share/elasticsearch/plugins/", type="path"),
|
|
|
|
proxy_host=dict(default=None),
|
|
|
|
proxy_port=dict(default=None),
|
|
|
|
version=dict(default=None)
|
|
|
|
),
|
|
|
|
mutually_exclusive=[("src", "url")],
|
|
|
|
supports_check_mode=True
|
|
|
|
)
|
|
|
|
|
|
|
|
name = module.params["name"]
|
|
|
|
state = module.params["state"]
|
|
|
|
url = module.params["url"]
|
|
|
|
src = module.params["src"]
|
|
|
|
timeout = module.params["timeout"]
|
|
|
|
force = module.params["force"]
|
|
|
|
plugin_bin = module.params["plugin_bin"]
|
|
|
|
plugin_dir = module.params["plugin_dir"]
|
|
|
|
proxy_host = module.params["proxy_host"]
|
|
|
|
proxy_port = module.params["proxy_port"]
|
|
|
|
version = module.params["version"]
|
|
|
|
|
|
|
|
# Search provided path and system paths for valid binary
|
|
|
|
plugin_bin = get_plugin_bin(module, plugin_bin)
|
|
|
|
|
|
|
|
repo = parse_plugin_repo(name)
|
|
|
|
present = is_plugin_present(repo, plugin_dir)
|
|
|
|
|
|
|
|
# skip if the state is correct
|
|
|
|
if (present and state == "present") or (state == "absent" and not present):
|
|
|
|
module.exit_json(changed=False, name=name, state=state)
|
|
|
|
|
|
|
|
if state == "present":
|
|
|
|
changed, cmd, out, err = install_plugin(module, plugin_bin, name, version, src, url, proxy_host, proxy_port, timeout, force)
|
|
|
|
|
|
|
|
elif state == "absent":
|
|
|
|
changed, cmd, out, err = remove_plugin(module, plugin_bin, name)
|
|
|
|
|
|
|
|
module.exit_json(changed=changed, cmd=cmd, name=name, state=state, url=url, timeout=timeout, stdout=out, stderr=err)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|