mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 05:23:58 -07:00 
			
		
		
		
	Adds bigip management route module (#39520)
This module can be used for managing the management routes on a bigip
This commit is contained in:
		
					parent
					
						
							
								ec9c59f52b
							
						
					
				
			
			
				commit
				
					
						5f9b7046ed
					
				
			
		
					 3 changed files with 556 additions and 0 deletions
				
			
		
							
								
								
									
										420
									
								
								lib/ansible/modules/network/f5/bigip_management_route.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										420
									
								
								lib/ansible/modules/network/f5/bigip_management_route.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,420 @@ | |||
| #!/usr/bin/python | ||||
| # -*- coding: utf-8 -*- | ||||
| # | ||||
| # Copyright: (c) 2018, F5 Networks Inc. | ||||
| # 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: bigip_management_route | ||||
| short_description: Manage system management routes on a BIG-IP | ||||
| description: | ||||
|   - Configures route settings for the management interface of a BIG-IP. | ||||
| version_added: 2.6 | ||||
| options: | ||||
|   name: | ||||
|     description: | ||||
|       - Specifies the name of the management route. | ||||
|     required: True | ||||
|   description: | ||||
|     description: | ||||
|       - Description of the management route. | ||||
|   gateway: | ||||
|     description: | ||||
|       - Specifies that the system forwards packets to the destination through the | ||||
|         gateway with the specified IP address. | ||||
|   network: | ||||
|     description: | ||||
|       - The subnet and netmask to be used for the route. | ||||
|       - To specify that the route is the default route for the system, provide the | ||||
|         value C(default). | ||||
|       - Only one C(default) entry is allowed. | ||||
|       - This parameter cannot be changed after it is set. Therefore, if you do need to change | ||||
|         it, it is required that you delete and create a new route. | ||||
|   partition: | ||||
|     description: | ||||
|       - Device partition to manage resources on. | ||||
|     default: Common | ||||
|   state: | ||||
|     description: | ||||
|       - When C(present), ensures that the resource exists. | ||||
|       - When C(absent), ensures the resource is removed. | ||||
|     default: present | ||||
|     choices: | ||||
|       - present | ||||
|       - absent | ||||
| extends_documentation_fragment: f5 | ||||
| author: | ||||
|   - Tim Rupp (@caphrim007) | ||||
| ''' | ||||
| 
 | ||||
| EXAMPLES = r''' | ||||
| - name: Create a management route | ||||
|   bigip_management_route: | ||||
|     name: tacacs | ||||
|     description: Route to TACACS | ||||
|     gateway: 10.10.10.10 | ||||
|     network: 11.11.11.0/24 | ||||
|     password: secret | ||||
|     server: lb.mydomain.com | ||||
|     state: present | ||||
|     user: admin | ||||
|   delegate_to: localhost | ||||
| ''' | ||||
| 
 | ||||
| RETURN = r''' | ||||
| description: | ||||
|   description: The new description of the management route. | ||||
|   returned: changed | ||||
|   type: string | ||||
|   sample: Route to TACACS | ||||
| gateway: | ||||
|   description: The new gateway of the management route. | ||||
|   returned: changed | ||||
|   type: string | ||||
|   sample: 10.10.10.10 | ||||
| network: | ||||
|   description: The new network to use for the management route. | ||||
|   returned: changed | ||||
|   type: string | ||||
|   sample: default | ||||
| ''' | ||||
| 
 | ||||
| from ansible.module_utils.basic import AnsibleModule | ||||
| from ansible.module_utils.basic import env_fallback | ||||
| 
 | ||||
| try: | ||||
|     from library.module_utils.network.f5.bigip import HAS_F5SDK | ||||
|     from library.module_utils.network.f5.bigip import F5Client | ||||
|     from library.module_utils.network.f5.common import F5ModuleError | ||||
|     from library.module_utils.network.f5.common import AnsibleF5Parameters | ||||
|     from library.module_utils.network.f5.common import cleanup_tokens | ||||
|     from library.module_utils.network.f5.common import fq_name | ||||
|     from library.module_utils.network.f5.common import f5_argument_spec | ||||
|     try: | ||||
|         from library.module_utils.network.f5.common import iControlUnexpectedHTTPError | ||||
|     except ImportError: | ||||
|         HAS_F5SDK = False | ||||
| except ImportError: | ||||
|     from ansible.module_utils.network.f5.bigip import HAS_F5SDK | ||||
|     from ansible.module_utils.network.f5.bigip import F5Client | ||||
|     from ansible.module_utils.network.f5.common import F5ModuleError | ||||
|     from ansible.module_utils.network.f5.common import AnsibleF5Parameters | ||||
|     from ansible.module_utils.network.f5.common import cleanup_tokens | ||||
|     from ansible.module_utils.network.f5.common import fq_name | ||||
|     from ansible.module_utils.network.f5.common import f5_argument_spec | ||||
|     try: | ||||
|         from ansible.module_utils.network.f5.common import iControlUnexpectedHTTPError | ||||
|     except ImportError: | ||||
|         HAS_F5SDK = False | ||||
| 
 | ||||
| try: | ||||
|     import netaddr | ||||
|     HAS_NETADDR = True | ||||
| except ImportError: | ||||
|     HAS_NETADDR = False | ||||
| 
 | ||||
| 
 | ||||
| class Parameters(AnsibleF5Parameters): | ||||
|     api_map = { | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     api_attributes = [ | ||||
|         'description', | ||||
|         'gateway', | ||||
|         'network', | ||||
|     ] | ||||
| 
 | ||||
|     returnables = [ | ||||
|         'description', | ||||
|         'gateway', | ||||
|         'network', | ||||
|     ] | ||||
| 
 | ||||
|     updatables = [ | ||||
|         'description', | ||||
|         'gateway', | ||||
|         'network', | ||||
|     ] | ||||
| 
 | ||||
| 
 | ||||
| class ApiParameters(Parameters): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| class ModuleParameters(Parameters): | ||||
|     @property | ||||
|     def network(self): | ||||
|         if self._values['network'] is None: | ||||
|             return None | ||||
|         if self._values['network'] == 'default': | ||||
|             return 'default' | ||||
|         try: | ||||
|             addr = netaddr.IPNetwork(self._values['network']) | ||||
|             return str(addr) | ||||
|         except netaddr.core.AddrFormatError: | ||||
|             raise F5ModuleError( | ||||
|                 "The 'network' must either be a network address (with CIDR) or the word 'default'." | ||||
|             ) | ||||
| 
 | ||||
|     @property | ||||
|     def gateway(self): | ||||
|         if self._values['gateway'] is None: | ||||
|             return None | ||||
|         try: | ||||
|             addr = netaddr.IPNetwork(self._values['gateway']) | ||||
|             return str(addr.ip) | ||||
|         except netaddr.core.AddrFormatError: | ||||
|             raise F5ModuleError( | ||||
|                 "The 'gateway' must either be an IP address." | ||||
|             ) | ||||
| 
 | ||||
| 
 | ||||
| class Changes(Parameters): | ||||
|     def to_return(self): | ||||
|         result = {} | ||||
|         try: | ||||
|             for returnable in self.returnables: | ||||
|                 result[returnable] = getattr(self, returnable) | ||||
|             result = self._filter_params(result) | ||||
|         except Exception: | ||||
|             pass | ||||
|         return result | ||||
| 
 | ||||
| 
 | ||||
| class UsableChanges(Changes): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| class ReportableChanges(Changes): | ||||
|     pass | ||||
| 
 | ||||
| 
 | ||||
| class Difference(object): | ||||
|     def __init__(self, want, have=None): | ||||
|         self.want = want | ||||
|         self.have = have | ||||
| 
 | ||||
|     def compare(self, param): | ||||
|         try: | ||||
|             result = getattr(self, param) | ||||
|             return result | ||||
|         except AttributeError: | ||||
|             return self.__default(param) | ||||
| 
 | ||||
|     def __default(self, param): | ||||
|         attr1 = getattr(self.want, param) | ||||
|         try: | ||||
|             attr2 = getattr(self.have, param) | ||||
|             if attr1 != attr2: | ||||
|                 return attr1 | ||||
|         except AttributeError: | ||||
|             return attr1 | ||||
| 
 | ||||
|     @property | ||||
|     def network(self): | ||||
|         if self.want.network != self.have.network: | ||||
|             raise F5ModuleError( | ||||
|                 "'network' cannot be changed after it is set." | ||||
|             ) | ||||
| 
 | ||||
| 
 | ||||
| class ModuleManager(object): | ||||
|     def __init__(self, *args, **kwargs): | ||||
|         self.module = kwargs.get('module', None) | ||||
|         self.client = kwargs.get('client', None) | ||||
|         self.want = ModuleParameters(params=self.module.params) | ||||
|         self.have = ApiParameters() | ||||
|         self.changes = UsableChanges() | ||||
| 
 | ||||
|     def _set_changed_options(self): | ||||
|         changed = {} | ||||
|         for key in Parameters.returnables: | ||||
|             if getattr(self.want, key) is not None: | ||||
|                 changed[key] = getattr(self.want, key) | ||||
|         if changed: | ||||
|             self.changes = UsableChanges(params=changed) | ||||
| 
 | ||||
|     def _update_changed_options(self): | ||||
|         diff = Difference(self.want, self.have) | ||||
|         updatables = Parameters.updatables | ||||
|         changed = dict() | ||||
|         for k in updatables: | ||||
|             change = diff.compare(k) | ||||
|             if change is None: | ||||
|                 continue | ||||
|             else: | ||||
|                 if isinstance(change, dict): | ||||
|                     changed.update(change) | ||||
|                 else: | ||||
|                     changed[k] = change | ||||
|         if changed: | ||||
|             self.changes = UsableChanges(params=changed) | ||||
|             return True | ||||
|         return False | ||||
| 
 | ||||
|     def should_update(self): | ||||
|         result = self._update_changed_options() | ||||
|         if result: | ||||
|             return True | ||||
|         return False | ||||
| 
 | ||||
|     def exec_module(self): | ||||
|         changed = False | ||||
|         result = dict() | ||||
|         state = self.want.state | ||||
| 
 | ||||
|         try: | ||||
|             if state == "present": | ||||
|                 changed = self.present() | ||||
|             elif state == "absent": | ||||
|                 changed = self.absent() | ||||
|         except iControlUnexpectedHTTPError as e: | ||||
|             raise F5ModuleError(str(e)) | ||||
| 
 | ||||
|         reportable = ReportableChanges(params=self.changes.to_return()) | ||||
|         changes = reportable.to_return() | ||||
|         result.update(**changes) | ||||
|         result.update(dict(changed=changed)) | ||||
|         self._announce_deprecations(result) | ||||
|         return result | ||||
| 
 | ||||
|     def _announce_deprecations(self, result): | ||||
|         warnings = result.pop('__warnings', []) | ||||
|         for warning in warnings: | ||||
|             self.client.module.deprecate( | ||||
|                 msg=warning['msg'], | ||||
|                 version=warning['version'] | ||||
|             ) | ||||
| 
 | ||||
|     def present(self): | ||||
|         if self.exists(): | ||||
|             return self.update() | ||||
|         else: | ||||
|             return self.create() | ||||
| 
 | ||||
|     def exists(self): | ||||
|         result = self.client.api.tm.sys.management_routes.management_route.exists( | ||||
|             name=self.want.name, | ||||
|             partition=self.want.partition | ||||
|         ) | ||||
|         return result | ||||
| 
 | ||||
|     def update(self): | ||||
|         self.have = self.read_current_from_device() | ||||
|         if not self.should_update(): | ||||
|             return False | ||||
|         if self.module.check_mode: | ||||
|             return True | ||||
|         self.update_on_device() | ||||
|         return True | ||||
| 
 | ||||
|     def remove(self): | ||||
|         if self.module.check_mode: | ||||
|             return True | ||||
|         self.remove_from_device() | ||||
|         if self.exists(): | ||||
|             raise F5ModuleError("Failed to delete the resource.") | ||||
|         return True | ||||
| 
 | ||||
|     def create(self): | ||||
|         self._set_changed_options() | ||||
|         if self.module.check_mode: | ||||
|             return True | ||||
|         self.create_on_device() | ||||
|         return True | ||||
| 
 | ||||
|     def create_on_device(self): | ||||
|         params = self.changes.api_params() | ||||
|         self.client.api.tm.sys.management_routes.management_route.create( | ||||
|             name=self.want.name, | ||||
|             partition=self.want.partition, | ||||
|             **params | ||||
|         ) | ||||
| 
 | ||||
|     def update_on_device(self): | ||||
|         params = self.changes.api_params() | ||||
|         resource = self.client.api.tm.sys.management_routes.management_route.load( | ||||
|             name=self.want.name, | ||||
|             partition=self.want.partition | ||||
|         ) | ||||
|         resource.modify(**params) | ||||
| 
 | ||||
|     def absent(self): | ||||
|         if self.exists(): | ||||
|             return self.remove() | ||||
|         return False | ||||
| 
 | ||||
|     def remove_from_device(self): | ||||
|         resource = self.client.api.tm.sys.management_routes.management_route.load( | ||||
|             name=self.want.name, | ||||
|             partition=self.want.partition | ||||
|         ) | ||||
|         if resource: | ||||
|             resource.delete() | ||||
| 
 | ||||
|     def read_current_from_device(self): | ||||
|         resource = self.client.api.tm.sys.management_routes.management_route.load( | ||||
|             name=self.want.name, | ||||
|             partition=self.want.partition | ||||
|         ) | ||||
|         result = resource.attrs | ||||
|         return ApiParameters(params=result) | ||||
| 
 | ||||
| 
 | ||||
| class ArgumentSpec(object): | ||||
|     def __init__(self): | ||||
|         self.supports_check_mode = True | ||||
|         argument_spec = dict( | ||||
|             name=dict(required=True), | ||||
|             gateway=dict(), | ||||
|             network=dict(), | ||||
|             description=dict(), | ||||
|             partition=dict( | ||||
|                 default='Common', | ||||
|                 fallback=(env_fallback, ['F5_PARTITION']) | ||||
|             ), | ||||
|             state=dict( | ||||
|                 default='present', | ||||
|                 choices=['present', 'absent'] | ||||
|             ) | ||||
|         ) | ||||
|         self.argument_spec = {} | ||||
|         self.argument_spec.update(f5_argument_spec) | ||||
|         self.argument_spec.update(argument_spec) | ||||
| 
 | ||||
| 
 | ||||
| def main(): | ||||
|     spec = ArgumentSpec() | ||||
| 
 | ||||
|     module = AnsibleModule( | ||||
|         argument_spec=spec.argument_spec, | ||||
|         supports_check_mode=spec.supports_check_mode | ||||
|     ) | ||||
|     if not HAS_F5SDK: | ||||
|         module.fail_json(msg="The python f5-sdk module is required") | ||||
|     if not HAS_NETADDR: | ||||
|         module.fail_json(msg="The python netaddr module is required") | ||||
| 
 | ||||
|     try: | ||||
|         client = F5Client(**module.params) | ||||
|         mm = ModuleManager(module=module, client=client) | ||||
|         results = mm.exec_module() | ||||
|         cleanup_tokens(client) | ||||
|         module.exit_json(**results) | ||||
|     except F5ModuleError as ex: | ||||
|         cleanup_tokens(client) | ||||
|         module.fail_json(msg=str(ex)) | ||||
| 
 | ||||
| 
 | ||||
| if __name__ == '__main__': | ||||
|     main() | ||||
|  | @ -0,0 +1,12 @@ | |||
| { | ||||
|     "kind": "tm:sys:management-route:management-routestate", | ||||
|     "name": "default", | ||||
|     "partition": "Common", | ||||
|     "fullPath": "/Common/default", | ||||
|     "generation": 1, | ||||
|     "selfLink": "https://localhost/mgmt/tm/sys/management-route/~Common~default?ver=13.1.0.4", | ||||
|     "description": "configured-by-dhcp", | ||||
|     "gateway": "10.0.2.2", | ||||
|     "mtu": 0, | ||||
|     "network": "default" | ||||
| } | ||||
							
								
								
									
										124
									
								
								test/units/modules/network/f5/test_bigip_management_route.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								test/units/modules/network/f5/test_bigip_management_route.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,124 @@ | |||
| # -*- coding: utf-8 -*- | ||||
| # | ||||
| # Copyright: (c) 2017, F5 Networks Inc. | ||||
| # 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 | ||||
| 
 | ||||
| import os | ||||
| import json | ||||
| import pytest | ||||
| import sys | ||||
| 
 | ||||
| from nose.plugins.skip import SkipTest | ||||
| if sys.version_info < (2, 7): | ||||
|     raise SkipTest("F5 Ansible modules require Python >= 2.7") | ||||
| 
 | ||||
| from ansible.compat.tests import unittest | ||||
| from ansible.compat.tests.mock import Mock | ||||
| from ansible.compat.tests.mock import patch | ||||
| from ansible.module_utils.basic import AnsibleModule | ||||
| 
 | ||||
| try: | ||||
|     from library.modules.bigip_management_route import ApiParameters | ||||
|     from library.modules.bigip_management_route import ModuleParameters | ||||
|     from library.modules.bigip_management_route import ModuleManager | ||||
|     from library.modules.bigip_management_route import ArgumentSpec | ||||
|     from library.module_utils.network.f5.common import F5ModuleError | ||||
|     from library.module_utils.network.f5.common import iControlUnexpectedHTTPError | ||||
|     from test.unit.modules.utils import set_module_args | ||||
| except ImportError: | ||||
|     try: | ||||
|         from ansible.modules.network.f5.bigip_management_route import ApiParameters | ||||
|         from ansible.modules.network.f5.bigip_management_route import ModuleParameters | ||||
|         from ansible.modules.network.f5.bigip_management_route import ModuleManager | ||||
|         from ansible.modules.network.f5.bigip_management_route import ArgumentSpec | ||||
|         from ansible.module_utils.network.f5.common import F5ModuleError | ||||
|         from ansible.module_utils.network.f5.common import iControlUnexpectedHTTPError | ||||
|         from units.modules.utils import set_module_args | ||||
|     except ImportError: | ||||
|         raise SkipTest("F5 Ansible modules require the f5-sdk Python library") | ||||
| 
 | ||||
| fixture_path = os.path.join(os.path.dirname(__file__), 'fixtures') | ||||
| fixture_data = {} | ||||
| 
 | ||||
| 
 | ||||
| def load_fixture(name): | ||||
|     path = os.path.join(fixture_path, name) | ||||
| 
 | ||||
|     if path in fixture_data: | ||||
|         return fixture_data[path] | ||||
| 
 | ||||
|     with open(path) as f: | ||||
|         data = f.read() | ||||
| 
 | ||||
|     try: | ||||
|         data = json.loads(data) | ||||
|     except Exception: | ||||
|         pass | ||||
| 
 | ||||
|     fixture_data[path] = data | ||||
|     return data | ||||
| 
 | ||||
| 
 | ||||
| class TestParameters(unittest.TestCase): | ||||
|     def test_module_parameters(self): | ||||
|         args = dict( | ||||
|             name='foo', | ||||
|             gateway='1.1.1.1', | ||||
|             network='default', | ||||
|             description='my description' | ||||
|         ) | ||||
| 
 | ||||
|         p = ModuleParameters(params=args) | ||||
|         assert p.name == 'foo' | ||||
|         assert p.gateway == '1.1.1.1' | ||||
|         assert p.network == 'default' | ||||
|         assert p.description == 'my description' | ||||
| 
 | ||||
|     def test_api_parameters(self): | ||||
|         args = load_fixture('load_sys_management_route_1.json') | ||||
| 
 | ||||
|         p = ApiParameters(params=args) | ||||
|         assert p.name == 'default' | ||||
|         assert p.gateway == '10.0.2.2' | ||||
|         assert p.network == 'default' | ||||
|         assert p.description == 'configured-by-dhcp' | ||||
| 
 | ||||
| 
 | ||||
| @patch('ansible.module_utils.f5_utils.AnsibleF5Client._get_mgmt_root', | ||||
|        return_value=True) | ||||
| class TestManager(unittest.TestCase): | ||||
| 
 | ||||
|     def setUp(self): | ||||
|         self.spec = ArgumentSpec() | ||||
| 
 | ||||
|     def test_create_monitor(self, *args): | ||||
|         set_module_args(dict( | ||||
|             name='foo', | ||||
|             gateway='1.1.1.1', | ||||
|             network='default', | ||||
|             description='my description', | ||||
|             partition='Common', | ||||
|             server='localhost', | ||||
|             password='password', | ||||
|             user='admin' | ||||
|         )) | ||||
| 
 | ||||
|         module = AnsibleModule( | ||||
|             argument_spec=self.spec.argument_spec, | ||||
|             supports_check_mode=self.spec.supports_check_mode | ||||
|         ) | ||||
| 
 | ||||
|         # Override methods in the specific type of manager | ||||
|         mm = ModuleManager(module=module) | ||||
|         mm.exists = Mock(side_effect=[False, True]) | ||||
|         mm.create_on_device = Mock(return_value=True) | ||||
| 
 | ||||
|         results = mm.exec_module() | ||||
| 
 | ||||
|         assert results['changed'] is True | ||||
|         assert results['gateway'] == '1.1.1.1' | ||||
|         assert results['network'] == 'default' | ||||
|         assert results['description'] == 'my description' | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue