mirror of https://github.com/ansible/ansible.git
VMWare: New Module for content library CRUD operations (#58716)
parent
3f2b766d10
commit
7b8edbf9dd
@ -0,0 +1,290 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright: (c) 2019, Ansible Project
|
||||||
|
# Copyright: (c) 2019, Pavan Bidkar <pbidkar@vmware.com>
|
||||||
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
|
||||||
|
from __future__ import absolute_import, division, print_function
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
ANSIBLE_METADATA = {
|
||||||
|
'metadata_version': '1.1',
|
||||||
|
'status': ['preview'],
|
||||||
|
'supported_by': 'community'
|
||||||
|
}
|
||||||
|
|
||||||
|
DOCUMENTATION = r'''
|
||||||
|
---
|
||||||
|
module: vmware_content_library_manager
|
||||||
|
short_description: Create, update and delete VMware content library
|
||||||
|
description:
|
||||||
|
- Module to manage VMware content Library
|
||||||
|
- Content Library feature is introduced in vSphere 6.0 version, so this module is not supported in the earlier versions of vSphere.
|
||||||
|
- All variables and VMware object names are case sensitive.
|
||||||
|
version_added: '2.9'
|
||||||
|
author:
|
||||||
|
- Pavan Bidkar (@pgbidkar)
|
||||||
|
notes:
|
||||||
|
- Tested on vSphere 6.5, 6.7
|
||||||
|
requirements:
|
||||||
|
- python >= 2.6
|
||||||
|
- PyVmomi
|
||||||
|
- vSphere Automation SDK
|
||||||
|
options:
|
||||||
|
library_name:
|
||||||
|
description:
|
||||||
|
- The name of VMware content library to manage.
|
||||||
|
type: str
|
||||||
|
required: True
|
||||||
|
library_description:
|
||||||
|
description:
|
||||||
|
- The content library description.
|
||||||
|
- This is required only if C(state) is set to C(present).
|
||||||
|
- This parameter is ignored, when C(state) is set to C(absent).
|
||||||
|
- Process of updating content library only allows description change.
|
||||||
|
type: str
|
||||||
|
required: False
|
||||||
|
default: ''
|
||||||
|
library_type:
|
||||||
|
description:
|
||||||
|
- The content library type.
|
||||||
|
- This is required only if C(state) is set to C(present).
|
||||||
|
- This parameter is ignored, when C(state) is set to C(absent).
|
||||||
|
type: str
|
||||||
|
required: False
|
||||||
|
default: 'local'
|
||||||
|
choices: [ 'local', 'subscribed' ]
|
||||||
|
datastore_name:
|
||||||
|
description:
|
||||||
|
- Name of the datastore on which backing content library is created.
|
||||||
|
- This is required only if C(state) is set to C(present).
|
||||||
|
- This parameter is ignored, when C(state) is set to C(absent).
|
||||||
|
- Currently only datastore backing creation is supported.
|
||||||
|
type: str
|
||||||
|
required: False
|
||||||
|
aliases: ['datastore']
|
||||||
|
state:
|
||||||
|
description:
|
||||||
|
- The state of content library.
|
||||||
|
- If set to C(present) and library does not exists, then content library is created.
|
||||||
|
- If set to C(present) and library exists, then content library is updated.
|
||||||
|
- If set to C(absent) and library exists, then content library is deleted.
|
||||||
|
- If set to C(absent) and library does not exists, no action is taken.
|
||||||
|
type: str
|
||||||
|
required: False
|
||||||
|
default: 'present'
|
||||||
|
choices: [ 'present', 'absent' ]
|
||||||
|
extends_documentation_fragment: vmware_rest_client.documentation
|
||||||
|
'''
|
||||||
|
|
||||||
|
EXAMPLES = r'''
|
||||||
|
- name: Create Content Library
|
||||||
|
vmware_content_library_manager:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
library_name: test-content-lib
|
||||||
|
library_description: 'Library with Datastore Backing'
|
||||||
|
library_type: local
|
||||||
|
datastore_name: datastore
|
||||||
|
validate_certs: False
|
||||||
|
state: present
|
||||||
|
delegate_to: localhost
|
||||||
|
|
||||||
|
- name: Update Content Library
|
||||||
|
vmware_content_library_manager:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
library_name: test-content-lib
|
||||||
|
library_description: 'Library with Datastore Backing'
|
||||||
|
validate_certs: no
|
||||||
|
state: present
|
||||||
|
delegate_to: localhost
|
||||||
|
|
||||||
|
- name: Delete Content Library
|
||||||
|
vmware_content_library_manager:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
library_name: test-content-lib
|
||||||
|
validate_certs: no
|
||||||
|
state: absent
|
||||||
|
delegate_to: localhost
|
||||||
|
'''
|
||||||
|
|
||||||
|
RETURN = r'''
|
||||||
|
content_library_info:
|
||||||
|
description: library creation success and library_id
|
||||||
|
returned: on success
|
||||||
|
type: dict
|
||||||
|
sample: {
|
||||||
|
"library_id": "d0b92fa9-7039-4f29-8e9c-0debfcb22b72",
|
||||||
|
"library_description": 'Test description',
|
||||||
|
"library_type": 'LOCAL',
|
||||||
|
"msg": "Content Library 'demo-local-lib-4' created.",
|
||||||
|
}
|
||||||
|
'''
|
||||||
|
|
||||||
|
import uuid
|
||||||
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
from ansible.module_utils.vmware_rest_client import VmwareRestClient
|
||||||
|
from ansible.module_utils.vmware import PyVmomi
|
||||||
|
|
||||||
|
HAS_VAUTOMATION_PYTHON_SDK = False
|
||||||
|
try:
|
||||||
|
from com.vmware.content_client import LibraryModel
|
||||||
|
from com.vmware.content.library_client import StorageBacking
|
||||||
|
HAS_VAUTOMATION_PYTHON_SDK = True
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class VmwareContentLibCreate(VmwareRestClient):
|
||||||
|
def __init__(self, module):
|
||||||
|
"""Constructor."""
|
||||||
|
super(VmwareContentLibCreate, self).__init__(module)
|
||||||
|
self.content_service = self.api_client
|
||||||
|
self.local_libraries = dict()
|
||||||
|
self.library_name = self.params.get('library_name')
|
||||||
|
self.library_description = self.params.get('library_description')
|
||||||
|
self.library_type = self.params.get('library_type')
|
||||||
|
self.library_types = dict()
|
||||||
|
self.datastore_name = self.params.get('datastore_name')
|
||||||
|
self.get_all_libraries()
|
||||||
|
self.pyv = PyVmomi(module=module)
|
||||||
|
|
||||||
|
def process_state(self):
|
||||||
|
"""
|
||||||
|
Manage states of Content Library
|
||||||
|
"""
|
||||||
|
self.desired_state = self.params.get('state')
|
||||||
|
library_states = {
|
||||||
|
'absent': {
|
||||||
|
'present': self.state_destroy_library,
|
||||||
|
'absent': self.state_exit_unchanged,
|
||||||
|
},
|
||||||
|
'present': {
|
||||||
|
'present': self.state_update_library,
|
||||||
|
'absent': self.state_create_library,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
library_states[self.desired_state][self.check_content_library_status()]()
|
||||||
|
|
||||||
|
def get_all_libraries(self):
|
||||||
|
content_libs = self.content_service.content.LocalLibrary.list()
|
||||||
|
if content_libs:
|
||||||
|
for content_lib in content_libs:
|
||||||
|
lib_details = self.content_service.content.LocalLibrary.get(content_lib)
|
||||||
|
self.local_libraries[lib_details.name] = dict(
|
||||||
|
lib_name=lib_details.name,
|
||||||
|
lib_description=lib_details.description,
|
||||||
|
lib_id=lib_details.id,
|
||||||
|
lib_type=lib_details.type
|
||||||
|
)
|
||||||
|
|
||||||
|
def check_content_library_status(self):
|
||||||
|
"""
|
||||||
|
Check if Content Library exists or not
|
||||||
|
Returns: 'present' if library found, else 'absent'
|
||||||
|
|
||||||
|
"""
|
||||||
|
ret = 'present' if self.library_name in self.local_libraries else 'absent'
|
||||||
|
return ret
|
||||||
|
|
||||||
|
def state_create_library(self):
|
||||||
|
# Find the datastore by the given datastore name
|
||||||
|
datastore_id = self.pyv.find_datastore_by_name(datastore_name=self.datastore_name)
|
||||||
|
if not datastore_id:
|
||||||
|
self.module.fail_json(msg="Failed to find the datastore %s" % self.datastore_name)
|
||||||
|
self.datastore_id = datastore_id._moId
|
||||||
|
# Build the storage backing for the library to be created
|
||||||
|
storage_backings = []
|
||||||
|
storage_backing = StorageBacking(type=StorageBacking.Type.DATASTORE, datastore_id=self.datastore_id)
|
||||||
|
storage_backings.append(storage_backing)
|
||||||
|
|
||||||
|
# Build the specification for the library to be created
|
||||||
|
create_spec = LibraryModel()
|
||||||
|
create_spec.name = self.library_name
|
||||||
|
create_spec.description = self.library_description
|
||||||
|
self.library_types = {'local': create_spec.LibraryType.LOCAL,
|
||||||
|
'subscribed': create_spec.LibraryType.SUBSCRIBED}
|
||||||
|
create_spec.type = self.library_types[self.library_type]
|
||||||
|
create_spec.storage_backings = storage_backings
|
||||||
|
|
||||||
|
# Create a local content library backed the VC datastore
|
||||||
|
library_id = self.content_service.content.LocalLibrary.create(create_spec=create_spec,
|
||||||
|
client_token=str(uuid.uuid4()))
|
||||||
|
if library_id:
|
||||||
|
self.module.exit_json(
|
||||||
|
changed=True,
|
||||||
|
content_library_info=dict(
|
||||||
|
msg="Content Library '%s' created." % create_spec.name,
|
||||||
|
library_id=library_id,
|
||||||
|
library_description=self.library_description,
|
||||||
|
library_type=create_spec.type,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
self.module.exit_json(changed=False,
|
||||||
|
content_library_info=dict(msg="Content Library not created. Datastore and library_type required", library_id=''))
|
||||||
|
|
||||||
|
def state_update_library(self):
|
||||||
|
"""
|
||||||
|
Update Content Library
|
||||||
|
|
||||||
|
"""
|
||||||
|
changed = False
|
||||||
|
library_id = self.local_libraries[self.library_name]['lib_id']
|
||||||
|
content_library_info = dict(msg="Content Library %s is unchanged." % self.library_name, library_id=library_id)
|
||||||
|
library_update_spec = LibraryModel()
|
||||||
|
library_desc = self.local_libraries[self.library_name]['lib_description']
|
||||||
|
desired_lib_desc = self.params.get('library_description')
|
||||||
|
if library_desc != desired_lib_desc:
|
||||||
|
library_update_spec.description = desired_lib_desc
|
||||||
|
self.content_service.content.LocalLibrary.update(library_id, library_update_spec)
|
||||||
|
content_library_info['msg'] = 'Content Library %s updated.' % self.library_name
|
||||||
|
changed = True
|
||||||
|
|
||||||
|
self.module.exit_json(changed=changed, content_library_info=content_library_info)
|
||||||
|
|
||||||
|
def state_destroy_library(self):
|
||||||
|
"""
|
||||||
|
Delete Content Library
|
||||||
|
|
||||||
|
"""
|
||||||
|
library_id = self.local_libraries[self.library_name]['lib_id']
|
||||||
|
self.content_service.content.LocalLibrary.delete(library_id=library_id)
|
||||||
|
self.module.exit_json(
|
||||||
|
changed=True,
|
||||||
|
content_library_info=dict(
|
||||||
|
msg="Content Library '%s' deleted." % self.library_name,
|
||||||
|
library_id=library_id
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
def state_exit_unchanged(self):
|
||||||
|
"""
|
||||||
|
Return unchanged state
|
||||||
|
|
||||||
|
"""
|
||||||
|
self.module.exit_json(changed=False)
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
argument_spec = VmwareRestClient.vmware_client_argument_spec()
|
||||||
|
argument_spec.update(
|
||||||
|
library_name=dict(type='str', required=False),
|
||||||
|
library_description=dict(type='str', required=False),
|
||||||
|
library_type=dict(type='str', required=False, choices=['local', 'subscribed'], default='local'),
|
||||||
|
datastore_name=dict(type='str', required=False, aliases=['datastore']),
|
||||||
|
state=dict(type='str', choices=['present', 'absent'], default='present', required=False),
|
||||||
|
)
|
||||||
|
module = AnsibleModule(argument_spec=argument_spec,
|
||||||
|
supports_check_mode=True)
|
||||||
|
|
||||||
|
vmware_contentlib_create = VmwareContentLibCreate(module)
|
||||||
|
vmware_contentlib_create.process_state()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
@ -0,0 +1,3 @@
|
|||||||
|
cloud/vcenter
|
||||||
|
shippable/vcenter/group1
|
||||||
|
needs/target/prepare_vmware_tests
|
@ -0,0 +1,91 @@
|
|||||||
|
# Test code for the vmware_content_library CRUD Operations.
|
||||||
|
# Copyright: (c) 2019, Pavan Bidkar <pbidkar@vmware.com>
|
||||||
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
- import_role:
|
||||||
|
name: prepare_vmware_tests
|
||||||
|
vars:
|
||||||
|
setup_attach_host: true
|
||||||
|
setup_datastore: true
|
||||||
|
|
||||||
|
- when: vcsim is not defined
|
||||||
|
block:
|
||||||
|
- &content_lib_delete
|
||||||
|
name: Delete content library if exists
|
||||||
|
vmware_content_library_manager:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
validate_certs: no
|
||||||
|
library_name: Sample_Library
|
||||||
|
state: absent
|
||||||
|
|
||||||
|
- &content_lib_create
|
||||||
|
name: Create content library
|
||||||
|
vmware_content_library_manager:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
validate_certs: False
|
||||||
|
library_name: Sample_Library
|
||||||
|
library_description: Sample Description
|
||||||
|
datastore_name: '{{ ds1 }}'
|
||||||
|
state: present
|
||||||
|
register: content_lib_create_result
|
||||||
|
|
||||||
|
- name: Check content library is created
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- content_lib_create_result.changed
|
||||||
|
|
||||||
|
- <<: *content_lib_create
|
||||||
|
name: Create content library again
|
||||||
|
|
||||||
|
- name: Check if no changes are made
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- not content_lib_create_result.changed
|
||||||
|
|
||||||
|
# Testcase Update Content Library
|
||||||
|
- &update_content_lib
|
||||||
|
name: Update a content library
|
||||||
|
vmware_content_library_manager:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
validate_certs: no
|
||||||
|
library_name: Sample_Library
|
||||||
|
library_description: Update Sample Description
|
||||||
|
state: present
|
||||||
|
register: content_lib_update_result
|
||||||
|
|
||||||
|
- name: Check content library is updated
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- content_lib_update_result.changed
|
||||||
|
|
||||||
|
- <<: *update_content_lib
|
||||||
|
name: Update a content library again
|
||||||
|
|
||||||
|
- name: Check content library is not updated
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- not content_lib_update_result.changed
|
||||||
|
|
||||||
|
# Testcase Delete the content library
|
||||||
|
- <<: *content_lib_delete
|
||||||
|
name: Delete content library
|
||||||
|
register: content_lib_delete_result
|
||||||
|
|
||||||
|
- name: Check content library is deleted
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- content_lib_delete_result.changed
|
||||||
|
|
||||||
|
- <<: *content_lib_delete
|
||||||
|
name: Delete content library again
|
||||||
|
register: content_lib_delete_result
|
||||||
|
|
||||||
|
- name: Check if no changes are made
|
||||||
|
assert:
|
||||||
|
that:
|
||||||
|
- not content_lib_delete_result.changed
|
Loading…
Reference in New Issue