mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
vcenter_license: New module to manage vCenter licenses (#25143)
This module supports check-mode and diff output. It also includes basic integration tests. This PR fixes #24971
This commit is contained in:
parent
71163d4d61
commit
3a86579b69
4 changed files with 287 additions and 0 deletions
|
@ -288,6 +288,7 @@ Ansible Changes By Release
|
||||||
* rundeck_project
|
* rundeck_project
|
||||||
- sensu_silence
|
- sensu_silence
|
||||||
- vmware
|
- vmware
|
||||||
|
* vcenter_license
|
||||||
* vmware_guest_find
|
* vmware_guest_find
|
||||||
- windows
|
- windows
|
||||||
* win_defrag
|
* win_defrag
|
||||||
|
|
208
lib/ansible/modules/cloud/vmware/vcenter_license.py
Normal file
208
lib/ansible/modules/cloud/vmware/vcenter_license.py
Normal file
|
@ -0,0 +1,208 @@
|
||||||
|
#!/usr/bin/python
|
||||||
|
|
||||||
|
# Copyright 2017, Dag Wieers <dag@wieers.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/>.
|
||||||
|
|
||||||
|
from __future__ import absolute_import, division, print_function
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
ANSIBLE_METADATA = {'metadata_version': '1.0',
|
||||||
|
'status': ['preview'],
|
||||||
|
'supported_by': 'community'}
|
||||||
|
|
||||||
|
DOCUMENTATION = r'''
|
||||||
|
module: vcenter_license
|
||||||
|
short_description: Manage VMware vCenter license keys
|
||||||
|
description:
|
||||||
|
- Add and delete vCenter license keys.
|
||||||
|
version_added: '2.4'
|
||||||
|
author: Dag Wieers (@dagwieers)
|
||||||
|
requirements:
|
||||||
|
- pyVmomi
|
||||||
|
options:
|
||||||
|
hostname:
|
||||||
|
description:
|
||||||
|
- The hostname or IP address of the vSphere vCenter.
|
||||||
|
required: yes
|
||||||
|
username:
|
||||||
|
description:
|
||||||
|
- The username to log into the vSphere vCenter.
|
||||||
|
required: yes
|
||||||
|
aliases: [admin, user]
|
||||||
|
password:
|
||||||
|
description:
|
||||||
|
- The password to log into to the vSphere vCenter.
|
||||||
|
required: yes
|
||||||
|
aliases: [pass, pwd]
|
||||||
|
labels:
|
||||||
|
description:
|
||||||
|
- The optional labels of the license key to manage in vSphere vCenter.
|
||||||
|
- This is dictionary with key/value pair.
|
||||||
|
license:
|
||||||
|
description:
|
||||||
|
- The license key to manage in vSphere vCenter.
|
||||||
|
required: yes
|
||||||
|
state:
|
||||||
|
description:
|
||||||
|
- Whether to add (C(present)) or remove (C(absent)) the license key.
|
||||||
|
choices: [absent, present]
|
||||||
|
default: present
|
||||||
|
notes:
|
||||||
|
- This module will also auto-assign the current vCenter to the license key
|
||||||
|
if the product matches the license key, and vCenter us currently assigned
|
||||||
|
an evaluation license only.
|
||||||
|
- The evaluation license (00000-00000-00000-00000-00000) is not listed
|
||||||
|
when unused.
|
||||||
|
'''
|
||||||
|
|
||||||
|
EXAMPLES = r'''
|
||||||
|
- name: Add a new vCenter license
|
||||||
|
vcenter_license:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
license: f600d-21ae3-5592b-249e0-cc341
|
||||||
|
state: present
|
||||||
|
delegate_to: localhost
|
||||||
|
|
||||||
|
- name: Remove an (unused) vCenter license
|
||||||
|
vmware_license:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
license: f600d-21ae3-5592b-249e0-cc341
|
||||||
|
state: absent
|
||||||
|
delegate_to: localhost
|
||||||
|
'''
|
||||||
|
|
||||||
|
RETURN = r'''
|
||||||
|
licenses:
|
||||||
|
description: list of license keys after module executed
|
||||||
|
returned: always
|
||||||
|
type: list
|
||||||
|
sample:
|
||||||
|
- f600d-21ae3-5592b-249e0-cc341
|
||||||
|
- 143cc-0e942-b2955-3ea12-d006f
|
||||||
|
'''
|
||||||
|
|
||||||
|
try:
|
||||||
|
from pyVmomi import vim, vmodl
|
||||||
|
HAS_PYVMOMI = True
|
||||||
|
except ImportError:
|
||||||
|
HAS_PYVMOMI = False
|
||||||
|
|
||||||
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
from ansible.module_utils.vmware import connect_to_api, vmware_argument_spec
|
||||||
|
|
||||||
|
|
||||||
|
def find_key(licenses, license):
|
||||||
|
for item in licenses:
|
||||||
|
if item.licenseKey == license:
|
||||||
|
return item
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
def list_keys(licenses):
|
||||||
|
keys = []
|
||||||
|
for item in licenses:
|
||||||
|
# Filter out evaluation license key
|
||||||
|
if item.used is None:
|
||||||
|
continue
|
||||||
|
keys.append(item.licenseKey)
|
||||||
|
return keys
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
argument_spec = vmware_argument_spec()
|
||||||
|
argument_spec.update(dict(
|
||||||
|
labels=dict(type='dict', default=dict(source='ansible')),
|
||||||
|
license=dict(type='str', required=True),
|
||||||
|
state=dict(type='str', default='present', choices=['absent', 'present']),
|
||||||
|
))
|
||||||
|
|
||||||
|
module = AnsibleModule(
|
||||||
|
argument_spec=argument_spec,
|
||||||
|
supports_check_mode=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
license = module.params['license']
|
||||||
|
state = module.params['state']
|
||||||
|
|
||||||
|
# FIXME: This does not seem to work on vCenter v6.0
|
||||||
|
labels = []
|
||||||
|
for k in module.params['labels']:
|
||||||
|
kv = vim.KeyValue()
|
||||||
|
kv.key = k
|
||||||
|
kv.value = module.params['labels'][k]
|
||||||
|
labels.append(kv)
|
||||||
|
|
||||||
|
result = dict(
|
||||||
|
changed=False,
|
||||||
|
diff=dict(),
|
||||||
|
)
|
||||||
|
|
||||||
|
if not HAS_PYVMOMI:
|
||||||
|
module.fail_json(msg='pyvmomi is required for this module')
|
||||||
|
|
||||||
|
content = connect_to_api(module)
|
||||||
|
lm = content.licenseManager
|
||||||
|
|
||||||
|
result['licenses'] = list_keys(lm.licenses)
|
||||||
|
if module._diff:
|
||||||
|
result['diff']['before'] = '\n'.join(result['licenses']) + '\n'
|
||||||
|
|
||||||
|
if state == 'present' and license not in result['licenses']:
|
||||||
|
|
||||||
|
result['changed'] = True
|
||||||
|
if module.check_mode:
|
||||||
|
result['licenses'].append(license)
|
||||||
|
else:
|
||||||
|
lm.AddLicense(license, labels)
|
||||||
|
|
||||||
|
# Automatically assign to current vCenter, if needed
|
||||||
|
key = find_key(lm.licenses, license)
|
||||||
|
if content.about.name in key.name:
|
||||||
|
try:
|
||||||
|
lam = lm.licenseAssignmentManager
|
||||||
|
lam.UpdateAssignedLicense(entity=content.about.instanceUuid, licenseKey=license)
|
||||||
|
except:
|
||||||
|
module.warn('Could not assign "%s" (%s) to vCenter.' % (license, key.name))
|
||||||
|
|
||||||
|
result['licenses'] = list_keys(lm.licenses)
|
||||||
|
if module._diff:
|
||||||
|
result['diff']['after'] = '\n'.join(result['licenses']) + '\n'
|
||||||
|
|
||||||
|
elif state == 'absent' and license in result['licenses']:
|
||||||
|
|
||||||
|
# Check if key is in use
|
||||||
|
key = find_key(lm.licenses, license)
|
||||||
|
if key.used > 0:
|
||||||
|
module.fail_json(msg='Cannot remove key "%s", still in use %s time(s).' % (license, key.used))
|
||||||
|
|
||||||
|
result['changed'] = True
|
||||||
|
if module.check_mode:
|
||||||
|
result['licenses'].remove(license)
|
||||||
|
else:
|
||||||
|
lm.RemoveLicense(license)
|
||||||
|
result['licenses'] = list_keys(lm.licenses)
|
||||||
|
if module._diff:
|
||||||
|
result['diff']['after'] = '\n'.join(result['licenses']) + '\n'
|
||||||
|
|
||||||
|
module.exit_json(**result)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
2
test/integration/targets/vcenter_license/aliases
Normal file
2
test/integration/targets/vcenter_license/aliases
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
posix/ci/cloud/vcenter
|
||||||
|
cloud/vcenter
|
76
test/integration/targets/vcenter_license/tasks/main.yml
Normal file
76
test/integration/targets/vcenter_license/tasks/main.yml
Normal file
|
@ -0,0 +1,76 @@
|
||||||
|
# Test code for the vcenter license module.
|
||||||
|
# (c) 2017, Dag Wieers <dag@wieers.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/>.
|
||||||
|
|
||||||
|
- name: Make sure pyvmomi is installed
|
||||||
|
pip:
|
||||||
|
name: pyvmomi
|
||||||
|
state: latest
|
||||||
|
|
||||||
|
- name: Store the vCenter container IP
|
||||||
|
set_fact:
|
||||||
|
vcenter_host: "{{ lookup('env', 'vcenter_host') }}"
|
||||||
|
|
||||||
|
- debug:
|
||||||
|
var: vcenter_host
|
||||||
|
|
||||||
|
- name: Wait for vcsim controller to come online {{ vcenter_host }}
|
||||||
|
wait_for:
|
||||||
|
host: "{{ vcenter_host }}"
|
||||||
|
port: 5000
|
||||||
|
state: started
|
||||||
|
|
||||||
|
- name: Kill vcsim
|
||||||
|
uri:
|
||||||
|
url: http://{{ vcenter_host }}:5000/killall
|
||||||
|
|
||||||
|
- name: Start vcsim
|
||||||
|
uri:
|
||||||
|
url: http://{{ vcenter_host }}:5000/spawn?cluster=2
|
||||||
|
register: vcsim_instance
|
||||||
|
|
||||||
|
- debug:
|
||||||
|
var: vcsim_instance
|
||||||
|
|
||||||
|
# FIXME: ServerFaultCode: LicenseManager:LicenseManager does not implement: AddLicense
|
||||||
|
#- name: Add a vCenter evaluation license
|
||||||
|
# vcenter_license:
|
||||||
|
# hostname: '{{ vcenter_host }}'
|
||||||
|
# username: '{{ vcsim_instance["json"]["username"] }}'
|
||||||
|
# password: '{{ vcsim_instance["json"]["password"] }}'
|
||||||
|
# validate_certs: no
|
||||||
|
# license: 00000-00000-00000-00000-00000
|
||||||
|
# state: present
|
||||||
|
|
||||||
|
#- name: Remove an (unused) vCenter evaluation license
|
||||||
|
# vcenter_license:
|
||||||
|
# hostname: '{{ vcenter_host }}'
|
||||||
|
# username: '{{ vcsim_instance["json"]["username"] }}'
|
||||||
|
# password: '{{ vcsim_instance["json"]["password"] }}'
|
||||||
|
# validate_certs: no
|
||||||
|
# license: 00000-00000-00000-00000-00000
|
||||||
|
# state: absent
|
||||||
|
|
||||||
|
#- name: Add an invalid vCenter license
|
||||||
|
# vcenter_license:
|
||||||
|
# hostname: '{{ vcenter_host }}'
|
||||||
|
# username: '{{ vcsim_instance["json"]["username"] }}'
|
||||||
|
# password: '{{ vcsim_instance["json"]["password"] }}'
|
||||||
|
# validate_certs: no
|
||||||
|
# license: 00000-00000-00000-00000-00001
|
||||||
|
# state: present
|
||||||
|
# ignore_errors: yes
|
Loading…
Add table
Reference in a new issue