2021-09-28 16:32:15 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright (c) 2020, FELDSAM s.r.o. - FeldHost™ <support@feldhost.cz>
|
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
|
2021-09-28 16:32:15 +02:00
|
|
|
|
|
|
|
from __future__ import (absolute_import, division, print_function)
|
|
|
|
|
|
|
|
__metaclass__ = type
|
|
|
|
|
|
|
|
DOCUMENTATION = r'''
|
|
|
|
name: opennebula
|
|
|
|
author:
|
|
|
|
- Kristian Feldsam (@feldsam)
|
|
|
|
short_description: OpenNebula inventory source
|
|
|
|
version_added: "3.8.0"
|
|
|
|
extends_documentation_fragment:
|
|
|
|
- constructed
|
|
|
|
description:
|
|
|
|
- Get inventory hosts from OpenNebula cloud.
|
2023-06-10 09:28:40 +02:00
|
|
|
- Uses an YAML configuration file ending with either C(opennebula.yml) or C(opennebula.yaml)
|
2021-09-28 16:32:15 +02:00
|
|
|
to set parameter values.
|
2023-06-10 09:28:40 +02:00
|
|
|
- Uses O(api_authfile), C(~/.one/one_auth), or E(ONE_AUTH) pointing to a OpenNebula credentials file.
|
2021-09-28 16:32:15 +02:00
|
|
|
options:
|
|
|
|
plugin:
|
|
|
|
description: Token that ensures this is a source file for the 'opennebula' plugin.
|
|
|
|
type: string
|
|
|
|
required: true
|
|
|
|
choices: [ community.general.opennebula ]
|
|
|
|
api_url:
|
|
|
|
description:
|
|
|
|
- URL of the OpenNebula RPC server.
|
|
|
|
- It is recommended to use HTTPS so that the username/password are not
|
|
|
|
transferred over the network unencrypted.
|
2023-06-10 09:28:40 +02:00
|
|
|
- If not set then the value of the E(ONE_URL) environment variable is used.
|
2021-09-28 16:32:15 +02:00
|
|
|
env:
|
|
|
|
- name: ONE_URL
|
2022-09-06 20:42:17 +02:00
|
|
|
required: true
|
2021-09-28 16:32:15 +02:00
|
|
|
type: string
|
|
|
|
api_username:
|
|
|
|
description:
|
|
|
|
- Name of the user to login into the OpenNebula RPC server. If not set
|
2023-06-10 09:28:40 +02:00
|
|
|
then the value of the E(ONE_USERNAME) environment variable is used.
|
2021-09-28 16:32:15 +02:00
|
|
|
env:
|
|
|
|
- name: ONE_USERNAME
|
|
|
|
type: string
|
|
|
|
api_password:
|
|
|
|
description:
|
|
|
|
- Password or a token of the user to login into OpenNebula RPC server.
|
2023-06-10 09:28:40 +02:00
|
|
|
- If not set, the value of the E(ONE_PASSWORD) environment variable is used.
|
2021-09-28 16:32:15 +02:00
|
|
|
env:
|
|
|
|
- name: ONE_PASSWORD
|
2022-09-06 20:42:17 +02:00
|
|
|
required: false
|
2021-09-28 16:32:15 +02:00
|
|
|
type: string
|
|
|
|
api_authfile:
|
|
|
|
description:
|
2023-06-10 09:28:40 +02:00
|
|
|
- If both O(api_username) or O(api_password) are not set, then it will try
|
2021-09-28 16:32:15 +02:00
|
|
|
authenticate with ONE auth file. Default path is C(~/.one/one_auth).
|
2023-06-10 09:28:40 +02:00
|
|
|
- Set environment variable E(ONE_AUTH) to override this path.
|
2021-09-28 16:32:15 +02:00
|
|
|
env:
|
|
|
|
- name: ONE_AUTH
|
2022-09-06 20:42:17 +02:00
|
|
|
required: false
|
2021-09-28 16:32:15 +02:00
|
|
|
type: string
|
|
|
|
hostname:
|
2023-06-10 09:28:40 +02:00
|
|
|
description: Field to match the hostname. Note V(v4_first_ip) corresponds to the first IPv4 found on VM.
|
2021-09-28 16:32:15 +02:00
|
|
|
type: string
|
|
|
|
default: v4_first_ip
|
|
|
|
choices:
|
|
|
|
- v4_first_ip
|
|
|
|
- v6_first_ip
|
|
|
|
- name
|
|
|
|
filter_by_label:
|
|
|
|
description: Only return servers filtered by this label.
|
|
|
|
type: string
|
|
|
|
group_by_labels:
|
|
|
|
description: Create host groups by vm labels
|
|
|
|
type: bool
|
2022-09-06 20:42:17 +02:00
|
|
|
default: true
|
2021-09-28 16:32:15 +02:00
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = r'''
|
|
|
|
# inventory_opennebula.yml file in YAML format
|
|
|
|
# Example command line: ansible-inventory --list -i inventory_opennebula.yml
|
|
|
|
|
|
|
|
# Pass a label filter to the API
|
|
|
|
plugin: community.general.opennebula
|
|
|
|
api_url: https://opennebula:2633/RPC2
|
|
|
|
filter_by_label: Cache
|
|
|
|
'''
|
|
|
|
|
|
|
|
try:
|
|
|
|
import pyone
|
|
|
|
|
|
|
|
HAS_PYONE = True
|
|
|
|
except ImportError:
|
|
|
|
HAS_PYONE = False
|
|
|
|
|
|
|
|
from ansible.errors import AnsibleError
|
|
|
|
from ansible.plugins.inventory import BaseInventoryPlugin, Constructable
|
2022-02-18 23:14:51 +01:00
|
|
|
from ansible.module_utils.common.text.converters import to_native
|
2024-04-20 09:39:42 +02:00
|
|
|
|
|
|
|
from ansible_collections.community.general.plugins.plugin_utils.unsafe import make_unsafe
|
2021-09-28 16:32:15 +02:00
|
|
|
|
|
|
|
from collections import namedtuple
|
|
|
|
import os
|
|
|
|
|
|
|
|
|
|
|
|
class InventoryModule(BaseInventoryPlugin, Constructable):
|
|
|
|
NAME = 'community.general.opennebula'
|
|
|
|
|
|
|
|
def verify_file(self, path):
|
|
|
|
valid = False
|
|
|
|
if super(InventoryModule, self).verify_file(path):
|
|
|
|
if path.endswith(('opennebula.yaml', 'opennebula.yml')):
|
|
|
|
valid = True
|
|
|
|
return valid
|
|
|
|
|
|
|
|
def _get_connection_info(self):
|
|
|
|
url = self.get_option('api_url')
|
|
|
|
username = self.get_option('api_username')
|
|
|
|
password = self.get_option('api_password')
|
|
|
|
authfile = self.get_option('api_authfile')
|
|
|
|
|
|
|
|
if not username and not password:
|
|
|
|
if authfile is None:
|
|
|
|
authfile = os.path.join(os.environ.get("HOME"), ".one", "one_auth")
|
|
|
|
try:
|
|
|
|
with open(authfile, "r") as fp:
|
|
|
|
authstring = fp.read().rstrip()
|
|
|
|
username, password = authstring.split(":")
|
|
|
|
except (OSError, IOError):
|
|
|
|
raise AnsibleError("Could not find or read ONE_AUTH file at '{e}'".format(e=authfile))
|
|
|
|
except Exception:
|
|
|
|
raise AnsibleError("Error occurs when reading ONE_AUTH file at '{e}'".format(e=authfile))
|
|
|
|
|
|
|
|
auth_params = namedtuple('auth', ('url', 'username', 'password'))
|
|
|
|
|
|
|
|
return auth_params(url=url, username=username, password=password)
|
|
|
|
|
|
|
|
def _get_vm_ipv4(self, vm):
|
|
|
|
nic = vm.TEMPLATE.get('NIC')
|
|
|
|
|
|
|
|
if isinstance(nic, dict):
|
|
|
|
nic = [nic]
|
|
|
|
|
|
|
|
for net in nic:
|
Fix OpenNebula inventory crash when NIC does not have IP (#8489)
* Fix OpenNebula inventory crash when NIC does not have IP
Match IPv6 behaviour.
When a NIC does not have an IP:
File "ansible/inventory/manager.py", line 292, in parse_source
plugin.parse(self._inventory, self._loader, source, cache=cache)
File "ansible-cm/plugins/inventory/opennebula.py", line 263, in parse
self._populate()
File "ansible-cm/plugins/inventory/opennebula.py", line 226, in _populate
servers = self._retrieve_servers(filter_by_label)
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "ansible-cm/plugins/inventory/opennebula.py", line 210, in _retrieve_servers
server['v4_first_ip'] = self._get_vm_ipv4(vm)
^^^^^^^^^^^^^^^^^^^^^
File "ansible-cm/plugins/inventory/opennebula.py", line 154, in _get_vm_ipv4
return net['IP']
* Update to call to match IPv6 and add changelog fragment
* Update changelog fragment.
---------
Co-authored-by: Felix Fontein <felix@fontein.de>
2024-06-16 22:14:18 +02:00
|
|
|
if net.get('IP'):
|
|
|
|
return net['IP']
|
2021-09-28 16:32:15 +02:00
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
def _get_vm_ipv6(self, vm):
|
|
|
|
nic = vm.TEMPLATE.get('NIC')
|
|
|
|
|
|
|
|
if isinstance(nic, dict):
|
|
|
|
nic = [nic]
|
|
|
|
|
|
|
|
for net in nic:
|
|
|
|
if net.get('IP6_GLOBAL'):
|
|
|
|
return net['IP6_GLOBAL']
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
def _get_vm_pool(self):
|
|
|
|
auth = self._get_connection_info()
|
|
|
|
|
|
|
|
if not (auth.username and auth.password):
|
|
|
|
raise AnsibleError('API Credentials missing. Check OpenNebula inventory file.')
|
|
|
|
else:
|
|
|
|
one_client = pyone.OneServer(auth.url, session=auth.username + ':' + auth.password)
|
|
|
|
|
|
|
|
# get hosts (VMs)
|
|
|
|
try:
|
|
|
|
vm_pool = one_client.vmpool.infoextended(-2, -1, -1, 3)
|
|
|
|
except Exception as e:
|
|
|
|
raise AnsibleError("Something happened during XML-RPC call: {e}".format(e=to_native(e)))
|
|
|
|
|
|
|
|
return vm_pool
|
|
|
|
|
|
|
|
def _retrieve_servers(self, label_filter=None):
|
|
|
|
vm_pool = self._get_vm_pool()
|
|
|
|
|
|
|
|
result = []
|
|
|
|
|
|
|
|
# iterate over hosts
|
|
|
|
for vm in vm_pool.VM:
|
|
|
|
server = vm.USER_TEMPLATE
|
|
|
|
|
|
|
|
labels = []
|
|
|
|
if vm.USER_TEMPLATE.get('LABELS'):
|
|
|
|
labels = [s for s in vm.USER_TEMPLATE.get('LABELS') if s == ',' or s == '-' or s.isalnum() or s.isspace()]
|
|
|
|
labels = ''.join(labels)
|
|
|
|
labels = labels.replace(' ', '_')
|
|
|
|
labels = labels.replace('-', '_')
|
|
|
|
labels = labels.split(',')
|
|
|
|
|
|
|
|
# filter by label
|
|
|
|
if label_filter is not None:
|
|
|
|
if label_filter not in labels:
|
|
|
|
continue
|
|
|
|
|
|
|
|
server['name'] = vm.NAME
|
2024-06-17 18:46:01 +02:00
|
|
|
server['id'] = vm.ID
|
2024-08-21 16:07:56 +02:00
|
|
|
if hasattr(vm.HISTORY_RECORDS, 'HISTORY'):
|
|
|
|
if vm.HISTORY_RECORDS.HISTORY:
|
|
|
|
if isinstance(vm.HISTORY_RECORDS.HISTORY, list) and len(vm.HISTORY_RECORDS.HISTORY) > 0:
|
|
|
|
if hasattr(vm.HISTORY_RECORDS.HISTORY[-1], 'HOSTNAME'):
|
|
|
|
server['host'] = vm.HISTORY_RECORDS.HISTORY[-1].HOSTNAME
|
2021-09-28 16:32:15 +02:00
|
|
|
server['LABELS'] = labels
|
|
|
|
server['v4_first_ip'] = self._get_vm_ipv4(vm)
|
|
|
|
server['v6_first_ip'] = self._get_vm_ipv6(vm)
|
|
|
|
|
|
|
|
result.append(server)
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
def _populate(self):
|
|
|
|
hostname_preference = self.get_option('hostname')
|
|
|
|
group_by_labels = self.get_option('group_by_labels')
|
2022-04-21 13:16:15 +02:00
|
|
|
strict = self.get_option('strict')
|
2021-09-28 16:32:15 +02:00
|
|
|
|
|
|
|
# Add a top group 'one'
|
|
|
|
self.inventory.add_group(group='all')
|
|
|
|
|
|
|
|
filter_by_label = self.get_option('filter_by_label')
|
2022-04-21 13:16:15 +02:00
|
|
|
servers = self._retrieve_servers(filter_by_label)
|
|
|
|
for server in servers:
|
2024-03-25 06:17:09 +01:00
|
|
|
server = make_unsafe(server)
|
2022-04-21 13:16:15 +02:00
|
|
|
hostname = server['name']
|
2021-09-28 16:32:15 +02:00
|
|
|
# check for labels
|
|
|
|
if group_by_labels and server['LABELS']:
|
|
|
|
for label in server['LABELS']:
|
|
|
|
self.inventory.add_group(group=label)
|
2022-04-21 13:16:15 +02:00
|
|
|
self.inventory.add_host(host=hostname, group=label)
|
2021-09-28 16:32:15 +02:00
|
|
|
|
2022-04-21 13:16:15 +02:00
|
|
|
self.inventory.add_host(host=hostname, group='all')
|
2021-09-28 16:32:15 +02:00
|
|
|
|
|
|
|
for attribute, value in server.items():
|
2022-04-21 13:16:15 +02:00
|
|
|
self.inventory.set_variable(hostname, attribute, value)
|
2021-09-28 16:32:15 +02:00
|
|
|
|
|
|
|
if hostname_preference != 'name':
|
2022-04-21 13:16:15 +02:00
|
|
|
self.inventory.set_variable(hostname, 'ansible_host', server[hostname_preference])
|
2021-09-28 16:32:15 +02:00
|
|
|
|
|
|
|
if server.get('SSH_PORT'):
|
2022-04-21 13:16:15 +02:00
|
|
|
self.inventory.set_variable(hostname, 'ansible_port', server['SSH_PORT'])
|
|
|
|
|
|
|
|
# handle construcable implementation: get composed variables if any
|
|
|
|
self._set_composite_vars(self.get_option('compose'), server, hostname, strict=strict)
|
|
|
|
|
|
|
|
# groups based on jinja conditionals get added to specific groups
|
|
|
|
self._add_host_to_composed_groups(self.get_option('groups'), server, hostname, strict=strict)
|
|
|
|
|
|
|
|
# groups based on variables associated with them in the inventory
|
|
|
|
self._add_host_to_keyed_groups(self.get_option('keyed_groups'), server, hostname, strict=strict)
|
2021-09-28 16:32:15 +02:00
|
|
|
|
|
|
|
def parse(self, inventory, loader, path, cache=True):
|
|
|
|
if not HAS_PYONE:
|
|
|
|
raise AnsibleError('OpenNebula Inventory plugin requires pyone to work!')
|
|
|
|
|
|
|
|
super(InventoryModule, self).parse(inventory, loader, path)
|
|
|
|
self._read_config_data(path=path)
|
|
|
|
|
|
|
|
self._populate()
|