mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
VMware: Tag management to VMware object (#46945)
Signed-off-by: Abhijeet Kasurde <akasurde@redhat.com>
This commit is contained in:
parent
fab815fc3b
commit
47e1bf1862
2 changed files with 249 additions and 0 deletions
|
@ -185,3 +185,23 @@ class VmwareRestClient(object):
|
||||||
for t in temp_tags_model:
|
for t in temp_tags_model:
|
||||||
tags.append(t.name)
|
tags.append(t.name)
|
||||||
return tags
|
return tags
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def search_svc_object_by_name(service, svc_obj_name=None):
|
||||||
|
"""
|
||||||
|
Return service object by name
|
||||||
|
Args:
|
||||||
|
service: Service object
|
||||||
|
svc_obj_name: Name of service object to find
|
||||||
|
|
||||||
|
Returns: Service object if found else None
|
||||||
|
|
||||||
|
"""
|
||||||
|
if not svc_obj_name:
|
||||||
|
return None
|
||||||
|
|
||||||
|
for svc_object in service.list():
|
||||||
|
svc_obj = service.get(svc_object)
|
||||||
|
if svc_obj.name == svc_obj_name:
|
||||||
|
return svc_obj
|
||||||
|
return None
|
||||||
|
|
229
lib/ansible/modules/cloud/vmware/vmware_tag_manager.py
Normal file
229
lib/ansible/modules/cloud/vmware/vmware_tag_manager.py
Normal file
|
@ -0,0 +1,229 @@
|
||||||
|
#!/usr/bin/python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright: (c) 2018, Ansible Project
|
||||||
|
# Copyright: (c) 2018, Abhijeet Kasurde <akasurde@redhat.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_tag_manager
|
||||||
|
short_description: Manage association of VMware tags with VMware objects
|
||||||
|
description:
|
||||||
|
- This module can be used to assign / remove VMware tags from the given VMware objects.
|
||||||
|
- Tag feature is introduced in vSphere 6 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.8
|
||||||
|
author:
|
||||||
|
- Abhijeet Kasurde (@Akasurde)
|
||||||
|
notes:
|
||||||
|
- Tested on vSphere 6.5
|
||||||
|
requirements:
|
||||||
|
- python >= 2.6
|
||||||
|
- PyVmomi
|
||||||
|
- vSphere Automation SDK
|
||||||
|
- vCloud Suite SDK
|
||||||
|
options:
|
||||||
|
tag_names:
|
||||||
|
description:
|
||||||
|
- List of tag(s) to be managed.
|
||||||
|
- You can also specify category name by specifying colon separated value. For example, "category_name:tag_name".
|
||||||
|
- You can skip category name if you have unique tag names.
|
||||||
|
required: True
|
||||||
|
state:
|
||||||
|
description:
|
||||||
|
- If C(state) is set to C(add) or C(present) will add the tags to the existing tag list of the given object.
|
||||||
|
- If C(state) is set to C(remove) or C(absent) will remove the tags from the existing tag list of the given object.
|
||||||
|
- If C(state) is set to C(set) will replace the tags of the given objects with the user defined list of tags.
|
||||||
|
default: add
|
||||||
|
choices: [ present, absent, add, remove, set ]
|
||||||
|
object_type:
|
||||||
|
description:
|
||||||
|
- Type of object to work with.
|
||||||
|
required: True
|
||||||
|
choices: [ VirtualMachine ]
|
||||||
|
object_name:
|
||||||
|
description:
|
||||||
|
- Name of the object to work with.
|
||||||
|
required: True
|
||||||
|
extends_documentation_fragment: vmware_rest_client.documentation
|
||||||
|
'''
|
||||||
|
|
||||||
|
EXAMPLES = r'''
|
||||||
|
- name: Add tags to a virtual machine
|
||||||
|
vmware_tag_manager:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
validate_certs: no
|
||||||
|
tag_names:
|
||||||
|
- Sample_Tag_0002
|
||||||
|
- Category_0001:Sample_Tag_0003
|
||||||
|
object_name: Fedora_VM
|
||||||
|
object_type: VirtualMachine
|
||||||
|
state: add
|
||||||
|
delegate_to: localhost
|
||||||
|
|
||||||
|
- name: Remove a tag to a virtual machine
|
||||||
|
vmware_tag_manager:
|
||||||
|
hostname: '{{ vcenter_hostname }}'
|
||||||
|
username: '{{ vcenter_username }}'
|
||||||
|
password: '{{ vcenter_password }}'
|
||||||
|
validate_certs: no
|
||||||
|
tag_names:
|
||||||
|
- Sample_Tag_0002
|
||||||
|
object_name: Fedora_VM
|
||||||
|
object_type: VirtualMachine
|
||||||
|
state: remove
|
||||||
|
delegate_to: localhost
|
||||||
|
'''
|
||||||
|
|
||||||
|
RETURN = r'''
|
||||||
|
tag_status:
|
||||||
|
description: metadata about tags related to object configuration
|
||||||
|
returned: on success
|
||||||
|
type: list
|
||||||
|
sample: {
|
||||||
|
"current_tags": [
|
||||||
|
"backup",
|
||||||
|
"security"
|
||||||
|
],
|
||||||
|
"desired_tags": [
|
||||||
|
"security"
|
||||||
|
],
|
||||||
|
"previous_tags": [
|
||||||
|
"backup",
|
||||||
|
"security"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
'''
|
||||||
|
|
||||||
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
from ansible.module_utils.vmware_rest_client import VmwareRestClient
|
||||||
|
from ansible.module_utils.vmware import PyVmomi
|
||||||
|
try:
|
||||||
|
from com.vmware.vapi.std_client import DynamicID
|
||||||
|
from com.vmware.cis.tagging_client import Tag, TagAssociation, Category
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class VmwareTagManager(VmwareRestClient):
|
||||||
|
def __init__(self, module):
|
||||||
|
"""
|
||||||
|
Constructor
|
||||||
|
"""
|
||||||
|
super(VmwareTagManager, self).__init__(module)
|
||||||
|
self.pyv = PyVmomi(module=module)
|
||||||
|
|
||||||
|
self.object_type = self.params.get('object_type')
|
||||||
|
self.object_name = self.params.get('object_name')
|
||||||
|
|
||||||
|
if self.object_type == 'VirtualMachine':
|
||||||
|
self.managed_object = self.pyv.get_vm_or_template(self.object_name)
|
||||||
|
self.dynamic_managed_object = DynamicID(type=self.object_type, id=self.managed_object._moId)
|
||||||
|
|
||||||
|
if self.managed_object is None:
|
||||||
|
self.module.fail_json(msg="Failed to find the managed object for %s with type %s" % (self.object_name, self.object_type))
|
||||||
|
|
||||||
|
self.tag_service = Tag(self.connect)
|
||||||
|
self.category_service = Category(self.connect)
|
||||||
|
self.tag_association_svc = TagAssociation(self.connect)
|
||||||
|
|
||||||
|
self.tag_names = self.params.get('tag_names')
|
||||||
|
|
||||||
|
def is_tag_category(self, cat_obj, tag_obj):
|
||||||
|
for tag in self.tag_service.list_tags_for_category(cat_obj.id):
|
||||||
|
if tag_obj.name == self.tag_service.get(tag).name:
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
def ensure_state(self):
|
||||||
|
"""
|
||||||
|
Manage the internal state of tags
|
||||||
|
|
||||||
|
"""
|
||||||
|
results = dict(
|
||||||
|
changed=False,
|
||||||
|
tag_status=dict(),
|
||||||
|
)
|
||||||
|
changed = False
|
||||||
|
action = self.params.get('state')
|
||||||
|
available_tag_obj = self.get_tags_for_object(tag_service=self.tag_service,
|
||||||
|
tag_assoc_svc=self.tag_association_svc,
|
||||||
|
dobj=self.dynamic_managed_object)
|
||||||
|
# Already existing tags from the given object
|
||||||
|
avail_tag_obj_name_list = [tag.name for tag in available_tag_obj]
|
||||||
|
results['tag_status']['previous_tags'] = avail_tag_obj_name_list
|
||||||
|
results['tag_status']['desired_tags'] = self.tag_names
|
||||||
|
|
||||||
|
# Check if category and tag combination exists as per user request
|
||||||
|
removed_tags_for_set = False
|
||||||
|
for tag in self.tag_names:
|
||||||
|
category_obj, category_name, tag_name = None, None, None
|
||||||
|
if ":" in tag:
|
||||||
|
# User specified category
|
||||||
|
category_name, tag_name = tag.split(":", 1)
|
||||||
|
category_obj = self.search_svc_object_by_name(self.category_service, category_name)
|
||||||
|
if not category_obj:
|
||||||
|
self.module.fail_json(msg="Unable to find the category %s" % category_name)
|
||||||
|
else:
|
||||||
|
# User specified only tag
|
||||||
|
tag_name = tag
|
||||||
|
|
||||||
|
tag_obj = self.search_svc_object_by_name(self.tag_service, tag_name)
|
||||||
|
if not tag_obj:
|
||||||
|
self.module.fail_json(msg="Unable to find the tag %s" % tag_name)
|
||||||
|
|
||||||
|
if category_name and category_obj and not self.is_tag_category(category_obj, tag_obj):
|
||||||
|
self.module.fail_json(msg="Category %s does not contain tag %s" % (category_name, tag_name))
|
||||||
|
|
||||||
|
if action in ('add', 'present'):
|
||||||
|
if tag_obj not in available_tag_obj:
|
||||||
|
# Tag is not already applied
|
||||||
|
self.tag_association_svc.attach(tag_id=tag_obj.id, object_id=self.dynamic_managed_object)
|
||||||
|
changed = True
|
||||||
|
elif action == 'set':
|
||||||
|
# Remove all tags first
|
||||||
|
if not removed_tags_for_set:
|
||||||
|
for av_tag in available_tag_obj:
|
||||||
|
self.tag_association_svc.detach(tag_id=av_tag.id, object_id=self.dynamic_managed_object)
|
||||||
|
removed_tags_for_set = True
|
||||||
|
self.tag_association_svc.attach(tag_id=tag_obj.id, object_id=self.dynamic_managed_object)
|
||||||
|
changed = True
|
||||||
|
elif action in ('remove', 'absent'):
|
||||||
|
if tag_obj in available_tag_obj:
|
||||||
|
self.tag_association_svc.detach(tag_id=tag_obj.id, object_id=self.dynamic_managed_object)
|
||||||
|
changed = True
|
||||||
|
|
||||||
|
results['tag_status']['current_tags'] = [tag.name for tag in self.get_tags_for_object(self.tag_service,
|
||||||
|
self.tag_association_svc,
|
||||||
|
self.dynamic_managed_object)]
|
||||||
|
results['changed'] = changed
|
||||||
|
self.module.exit_json(**results)
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
argument_spec = VmwareRestClient.vmware_client_argument_spec()
|
||||||
|
argument_spec.update(
|
||||||
|
tag_names=dict(type='list', required=True),
|
||||||
|
state=dict(type='str', choices=['absent', 'add', 'present', 'remove', 'set'], default='add'),
|
||||||
|
object_name=dict(type='str', required=True),
|
||||||
|
object_type=dict(type='str', required=True, choices=['VirtualMachine']),
|
||||||
|
)
|
||||||
|
module = AnsibleModule(argument_spec=argument_spec)
|
||||||
|
|
||||||
|
vmware_tag_manager = VmwareTagManager(module)
|
||||||
|
vmware_tag_manager.ensure_state()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
Loading…
Reference in a new issue