mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-24 21:14:00 -07:00 
			
		
		
		
	Added new module for managing LIGs in HPE OneView
This commit is contained in:
		
					parent
					
						
							
								54859a2132
							
						
					
				
			
			
				commit
				
					
						637571abfb
					
				
			
		
					 2 changed files with 425 additions and 0 deletions
				
			
		|  | @ -0,0 +1,167 @@ | |||
| #!/usr/bin/python | ||||
| # Copyright (c) 2016-2017 Hewlett Packard Enterprise Development LP | ||||
| # 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 = ''' | ||||
| --- | ||||
| module: oneview_logical_interconnect_group | ||||
| short_description: Manage OneView Logical Interconnect Group resources | ||||
| description: | ||||
|     - Provides an interface to manage Logical Interconnect Group resources. Can create, update, or delete. | ||||
| version_added: "2.5" | ||||
| requirements: | ||||
|     - hpOneView >= 4.0.0 | ||||
| author: | ||||
|     - Felipe Bulsoni (@fgbulsoni) | ||||
|     - Thiago Miotto (@tmiotto) | ||||
|     - Adriane Cardozo (@adriane-cardozo) | ||||
| options: | ||||
|     state: | ||||
|         description: | ||||
|             - Indicates the desired state for the Logical Interconnect Group resource. | ||||
|               C(present) will ensure data properties are compliant with OneView. | ||||
|               C(absent) will remove the resource from OneView, if it exists. | ||||
|         default: present | ||||
|         choices: ['present', 'absent'] | ||||
|     data: | ||||
|         description: | ||||
|             - List with the Logical Interconnect Group properties. | ||||
|         required: true | ||||
| extends_documentation_fragment: | ||||
|     - oneview | ||||
|     - oneview.validateetag | ||||
| ''' | ||||
| 
 | ||||
| EXAMPLES = ''' | ||||
| - name: Ensure that the Logical Interconnect Group is present | ||||
|   oneview_logical_interconnect_group: | ||||
|     config: /etc/oneview/oneview_config.json | ||||
|     state: present | ||||
|     data: | ||||
|       name: 'Test Logical Interconnect Group' | ||||
|       uplinkSets: [] | ||||
|       enclosureType: 'C7000' | ||||
|       interconnectMapTemplate: | ||||
|         interconnectMapEntryTemplates: | ||||
|           - logicalDownlinkUri: ~ | ||||
|             logicalLocation: | ||||
|                 locationEntries: | ||||
|                     - relativeValue: "1" | ||||
|                       type: "Bay" | ||||
|                     - relativeValue: 1 | ||||
|                       type: "Enclosure" | ||||
|             permittedInterconnectTypeName: 'HP VC Flex-10/10D Module' | ||||
|             # Alternatively you can inform permittedInterconnectTypeUri | ||||
|   delegate_to: localhost | ||||
| 
 | ||||
| - name: Ensure that the Logical Interconnect Group has the specified scopes | ||||
|   oneview_logical_interconnect_group: | ||||
|     config: /etc/oneview/oneview_config.json | ||||
|     state: present | ||||
|     data: | ||||
|       name: Test Logical Interconnect Group | ||||
|       scopeUris: | ||||
|         - /rest/scopes/00SC123456 | ||||
|         - /rest/scopes/01SC123456 | ||||
|   delegate_to: localhost | ||||
| 
 | ||||
| - name: Ensure that the Logical Interconnect Group is present with name 'Test' | ||||
|   oneview_logical_interconnect_group: | ||||
|     config: /etc/oneview/oneview_config.json | ||||
|     state: present | ||||
|     data: | ||||
|       name: New Logical Interconnect Group | ||||
|       newName: Test | ||||
|   delegate_to: localhost | ||||
| 
 | ||||
| - name: Ensure that the Logical Interconnect Group is absent | ||||
|   oneview_logical_interconnect_group: | ||||
|     config: /etc/oneview/oneview_config.json | ||||
|     state: absent | ||||
|     data: | ||||
|       name: New Logical Interconnect Group | ||||
|   delegate_to: localhost | ||||
| ''' | ||||
| 
 | ||||
| RETURN = ''' | ||||
| logical_interconnect_group: | ||||
|     description: Has the facts about the OneView Logical Interconnect Group. | ||||
|     returned: On state 'present'. Can be null. | ||||
|     type: dict | ||||
| ''' | ||||
| 
 | ||||
| from ansible.module_utils.oneview import OneViewModuleBase, OneViewModuleResourceNotFound | ||||
| 
 | ||||
| 
 | ||||
| class LogicalInterconnectGroupModule(OneViewModuleBase): | ||||
|     MSG_CREATED = 'Logical Interconnect Group created successfully.' | ||||
|     MSG_UPDATED = 'Logical Interconnect Group updated successfully.' | ||||
|     MSG_DELETED = 'Logical Interconnect Group deleted successfully.' | ||||
|     MSG_ALREADY_PRESENT = 'Logical Interconnect Group is already present.' | ||||
|     MSG_ALREADY_ABSENT = 'Logical Interconnect Group is already absent.' | ||||
|     MSG_INTERCONNECT_TYPE_NOT_FOUND = 'Interconnect Type was not found.' | ||||
| 
 | ||||
|     RESOURCE_FACT_NAME = 'logical_interconnect_group' | ||||
| 
 | ||||
|     def __init__(self): | ||||
|         argument_spec = dict( | ||||
|             state=dict(default='present', choices=['present', 'absent']), | ||||
|             data=dict(required=True, type='dict') | ||||
|         ) | ||||
| 
 | ||||
|         super(LogicalInterconnectGroupModule, self).__init__(additional_arg_spec=argument_spec, | ||||
|                                                              validate_etag_support=True) | ||||
|         self.resource_client = self.oneview_client.logical_interconnect_groups | ||||
| 
 | ||||
|     def execute_module(self): | ||||
|         resource = self.get_by_name(self.data['name']) | ||||
| 
 | ||||
|         if self.state == 'present': | ||||
|             return self.__present(resource) | ||||
|         elif self.state == 'absent': | ||||
|             return self.resource_absent(resource) | ||||
| 
 | ||||
|     def __present(self, resource): | ||||
|         scope_uris = self.data.pop('scopeUris', None) | ||||
| 
 | ||||
|         self.__replace_name_by_uris(self.data) | ||||
|         result = self.resource_present(resource, self.RESOURCE_FACT_NAME) | ||||
| 
 | ||||
|         if scope_uris is not None: | ||||
|             result = self.resource_scopes_set(result, 'logical_interconnect_group', scope_uris) | ||||
| 
 | ||||
|         return result | ||||
| 
 | ||||
|     def __replace_name_by_uris(self, data): | ||||
|         map_template = data.get('interconnectMapTemplate') | ||||
| 
 | ||||
|         if map_template: | ||||
|             map_entry_templates = map_template.get('interconnectMapEntryTemplates') | ||||
|             if map_entry_templates: | ||||
|                 for value in map_entry_templates: | ||||
|                     permitted_interconnect_type_name = value.pop('permittedInterconnectTypeName', None) | ||||
|                     if permitted_interconnect_type_name: | ||||
|                         value['permittedInterconnectTypeUri'] = self.__get_interconnect_type_by_name( | ||||
|                             permitted_interconnect_type_name).get('uri') | ||||
| 
 | ||||
|     def __get_interconnect_type_by_name(self, name): | ||||
|         i_type = self.oneview_client.interconnect_types.get_by('name', name) | ||||
|         if i_type: | ||||
|             return i_type[0] | ||||
|         else: | ||||
|             raise OneViewModuleResourceNotFound(self.MSG_INTERCONNECT_TYPE_NOT_FOUND) | ||||
| 
 | ||||
| 
 | ||||
| def main(): | ||||
|     LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == '__main__': | ||||
|     main() | ||||
|  | @ -0,0 +1,258 @@ | |||
| # Copyright (c) 2016-2017 Hewlett Packard Enterprise Development LP | ||||
| # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||
| 
 | ||||
| from copy import deepcopy | ||||
| 
 | ||||
| from ansible.compat.tests import unittest, mock | ||||
| from oneview_module_loader import OneViewModuleBase | ||||
| from ansible.modules.remote_management.oneview.oneview_logical_interconnect_group import LogicalInterconnectGroupModule | ||||
| from hpe_test_utils import OneViewBaseTestCase | ||||
| 
 | ||||
| 
 | ||||
| FAKE_MSG_ERROR = 'Fake message error' | ||||
| 
 | ||||
| DEFAULT_LIG_NAME = 'Test Logical Interconnect Group' | ||||
| RENAMED_LIG = 'Renamed Logical Interconnect Group' | ||||
| 
 | ||||
| DEFAULT_LIG_TEMPLATE = dict( | ||||
|     name=DEFAULT_LIG_NAME, | ||||
|     uplinkSets=[], | ||||
|     enclosureType='C7000', | ||||
|     interconnectMapTemplate=dict( | ||||
|         interconnectMapEntryTemplates=[] | ||||
|     ) | ||||
| ) | ||||
| 
 | ||||
| PARAMS_LIG_TEMPLATE_WITH_MAP = dict( | ||||
|     config='config.json', | ||||
|     state='present', | ||||
|     data=dict( | ||||
|         name=DEFAULT_LIG_NAME, | ||||
|         uplinkSets=[], | ||||
|         enclosureType='C7000', | ||||
|         interconnectMapTemplate=dict( | ||||
|             interconnectMapEntryTemplates=[ | ||||
|                 { | ||||
|                     "logicalDownlinkUri": None, | ||||
|                     "logicalLocation": { | ||||
|                         "locationEntries": [ | ||||
|                             { | ||||
|                                 "relativeValue": "1", | ||||
|                                 "type": "Bay" | ||||
|                             }, | ||||
|                             { | ||||
|                                 "relativeValue": 1, | ||||
|                                 "type": "Enclosure" | ||||
|                             } | ||||
|                         ] | ||||
|                     }, | ||||
|                     "permittedInterconnectTypeName": "HP VC Flex-10/10D Module" | ||||
|                 }] | ||||
|         ) | ||||
|     )) | ||||
| 
 | ||||
| PARAMS_FOR_PRESENT = dict( | ||||
|     config='config.json', | ||||
|     state='present', | ||||
|     data=dict(name=DEFAULT_LIG_NAME) | ||||
| ) | ||||
| 
 | ||||
| PARAMS_TO_RENAME = dict( | ||||
|     config='config.json', | ||||
|     state='present', | ||||
|     data=dict(name=DEFAULT_LIG_NAME, | ||||
|               newName=RENAMED_LIG) | ||||
| ) | ||||
| 
 | ||||
| PARAMS_WITH_CHANGES = dict( | ||||
|     config='config.json', | ||||
|     state='present', | ||||
|     data=dict(name=DEFAULT_LIG_NAME, | ||||
|               description='It is an example') | ||||
| ) | ||||
| 
 | ||||
| PARAMS_FOR_ABSENT = dict( | ||||
|     config='config.json', | ||||
|     state='absent', | ||||
|     data=dict(name=DEFAULT_LIG_NAME) | ||||
| ) | ||||
| 
 | ||||
| 
 | ||||
| class LogicalInterconnectGroupGeneralSpec(unittest.TestCase, | ||||
|                                           OneViewBaseTestCase): | ||||
|     def setUp(self): | ||||
|         self.configure_mocks(self, LogicalInterconnectGroupModule) | ||||
|         self.resource = self.mock_ov_client.logical_interconnect_groups | ||||
| 
 | ||||
|     def test_should_create_new_lig(self): | ||||
|         self.resource.get_by.return_value = [] | ||||
|         self.resource.create.return_value = DEFAULT_LIG_TEMPLATE | ||||
| 
 | ||||
|         self.mock_ansible_module.params = PARAMS_FOR_PRESENT | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.mock_ansible_module.exit_json.assert_called_once_with( | ||||
|             changed=True, | ||||
|             msg=LogicalInterconnectGroupModule.MSG_CREATED, | ||||
|             ansible_facts=dict(logical_interconnect_group=DEFAULT_LIG_TEMPLATE) | ||||
|         ) | ||||
| 
 | ||||
|     def test_should_create_new_with_named_permitted_interconnect_type(self): | ||||
|         self.resource.get_by.return_value = [] | ||||
|         self.resource.create.return_value = PARAMS_FOR_PRESENT | ||||
| 
 | ||||
|         self.mock_ansible_module.params = deepcopy(PARAMS_LIG_TEMPLATE_WITH_MAP) | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.mock_ansible_module.exit_json.assert_called_once_with( | ||||
|             changed=True, | ||||
|             msg=LogicalInterconnectGroupModule.MSG_CREATED, | ||||
|             ansible_facts=dict(logical_interconnect_group=PARAMS_FOR_PRESENT.copy()) | ||||
|         ) | ||||
| 
 | ||||
|     def test_should_fail_when_permitted_interconnect_type_name_not_exists(self): | ||||
|         self.resource.get_by.return_value = [] | ||||
|         self.resource.create.return_value = PARAMS_FOR_PRESENT | ||||
|         self.mock_ov_client.interconnect_types.get_by.return_value = [] | ||||
| 
 | ||||
|         self.mock_ansible_module.params = deepcopy(PARAMS_LIG_TEMPLATE_WITH_MAP) | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.mock_ansible_module.fail_json.assert_called_once_with( | ||||
|             exception=mock.ANY, | ||||
|             msg=LogicalInterconnectGroupModule.MSG_INTERCONNECT_TYPE_NOT_FOUND) | ||||
| 
 | ||||
|     def test_should_not_update_when_data_is_equals(self): | ||||
|         self.resource.get_by.return_value = [DEFAULT_LIG_TEMPLATE] | ||||
| 
 | ||||
|         self.mock_ansible_module.params = PARAMS_FOR_PRESENT | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.mock_ansible_module.exit_json.assert_called_once_with( | ||||
|             changed=False, | ||||
|             msg=LogicalInterconnectGroupModule.MSG_ALREADY_PRESENT, | ||||
|             ansible_facts=dict(logical_interconnect_group=DEFAULT_LIG_TEMPLATE) | ||||
|         ) | ||||
| 
 | ||||
|     def test_update_when_data_has_modified_attributes(self): | ||||
|         data_merged = DEFAULT_LIG_TEMPLATE.copy() | ||||
|         data_merged['description'] = 'New description' | ||||
| 
 | ||||
|         self.resource.get_by.return_value = [DEFAULT_LIG_TEMPLATE] | ||||
|         self.resource.update.return_value = data_merged | ||||
| 
 | ||||
|         self.mock_ansible_module.params = PARAMS_WITH_CHANGES | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.mock_ansible_module.exit_json.assert_called_once_with( | ||||
|             changed=True, | ||||
|             msg=LogicalInterconnectGroupModule.MSG_UPDATED, | ||||
|             ansible_facts=dict(logical_interconnect_group=data_merged) | ||||
|         ) | ||||
| 
 | ||||
|     def test_rename_when_resource_exists(self): | ||||
|         data_merged = DEFAULT_LIG_TEMPLATE.copy() | ||||
|         data_merged['name'] = RENAMED_LIG | ||||
|         params_to_rename = PARAMS_TO_RENAME.copy() | ||||
| 
 | ||||
|         self.resource.get_by.return_value = [DEFAULT_LIG_TEMPLATE] | ||||
|         self.resource.update.return_value = data_merged | ||||
| 
 | ||||
|         self.mock_ansible_module.params = params_to_rename | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.resource.update.assert_called_once_with(data_merged) | ||||
| 
 | ||||
|     def test_create_with_newName_when_resource_not_exists(self): | ||||
|         data_merged = DEFAULT_LIG_TEMPLATE.copy() | ||||
|         data_merged['name'] = RENAMED_LIG | ||||
|         params_to_rename = PARAMS_TO_RENAME.copy() | ||||
| 
 | ||||
|         self.resource.get_by.return_value = [] | ||||
|         self.resource.create.return_value = DEFAULT_LIG_TEMPLATE | ||||
| 
 | ||||
|         self.mock_ansible_module.params = params_to_rename | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.resource.create.assert_called_once_with(PARAMS_TO_RENAME['data']) | ||||
| 
 | ||||
|     def test_should_remove_lig(self): | ||||
|         self.resource.get_by.return_value = [DEFAULT_LIG_TEMPLATE] | ||||
| 
 | ||||
|         self.mock_ansible_module.params = PARAMS_FOR_ABSENT | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.mock_ansible_module.exit_json.assert_called_once_with( | ||||
|             changed=True, | ||||
|             msg=LogicalInterconnectGroupModule.MSG_DELETED | ||||
|         ) | ||||
| 
 | ||||
|     def test_should_do_nothing_when_lig_not_exist(self): | ||||
|         self.resource.get_by.return_value = [] | ||||
| 
 | ||||
|         self.mock_ansible_module.params = PARAMS_FOR_ABSENT | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.mock_ansible_module.exit_json.assert_called_once_with( | ||||
|             changed=False, | ||||
|             msg=LogicalInterconnectGroupModule.MSG_ALREADY_ABSENT | ||||
|         ) | ||||
| 
 | ||||
|     def test_update_scopes_when_different(self): | ||||
|         params_to_scope = PARAMS_FOR_PRESENT.copy() | ||||
|         params_to_scope['data']['scopeUris'] = ['test'] | ||||
|         self.mock_ansible_module.params = params_to_scope | ||||
| 
 | ||||
|         resource_data = DEFAULT_LIG_TEMPLATE.copy() | ||||
|         resource_data['scopeUris'] = ['fake'] | ||||
|         resource_data['uri'] = 'rest/lig/fake' | ||||
|         self.resource.get_by.return_value = [resource_data] | ||||
| 
 | ||||
|         patch_return = resource_data.copy() | ||||
|         patch_return['scopeUris'] = ['test'] | ||||
|         self.resource.patch.return_value = patch_return | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.resource.patch.assert_called_once_with('rest/lig/fake', | ||||
|                                                     operation='replace', | ||||
|                                                     path='/scopeUris', | ||||
|                                                     value=['test']) | ||||
| 
 | ||||
|         self.mock_ansible_module.exit_json.assert_called_once_with( | ||||
|             changed=True, | ||||
|             ansible_facts=dict(logical_interconnect_group=patch_return), | ||||
|             msg=LogicalInterconnectGroupModule.MSG_UPDATED | ||||
|         ) | ||||
| 
 | ||||
|     def test_should_do_nothing_when_scopes_are_the_same(self): | ||||
|         params_to_scope = PARAMS_FOR_PRESENT.copy() | ||||
|         params_to_scope['data']['scopeUris'] = ['test'] | ||||
|         self.mock_ansible_module.params = params_to_scope | ||||
| 
 | ||||
|         resource_data = DEFAULT_LIG_TEMPLATE.copy() | ||||
|         resource_data['scopeUris'] = ['test'] | ||||
|         self.resource.get_by.return_value = [resource_data] | ||||
| 
 | ||||
|         LogicalInterconnectGroupModule().run() | ||||
| 
 | ||||
|         self.resource.patch.not_been_called() | ||||
| 
 | ||||
|         self.mock_ansible_module.exit_json.assert_called_once_with( | ||||
|             changed=False, | ||||
|             ansible_facts=dict(logical_interconnect_group=resource_data), | ||||
|             msg=LogicalInterconnectGroupModule.MSG_ALREADY_PRESENT | ||||
|         ) | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue