mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
Fix nxos_hsrp and add unit test (#24770)
* nxos_hsrp fix Signed-off-by: Trishna Guha <trishnaguha17@gmail.com> * unit test nxos_hsrp Signed-off-by: Trishna Guha <trishnaguha17@gmail.com> * ansibot told me to do this * revert apply_key_map and simplify method
This commit is contained in:
parent
4782a4e62f
commit
1e5a0982b9
3 changed files with 138 additions and 113 deletions
|
@ -17,10 +17,11 @@
|
||||||
#
|
#
|
||||||
|
|
||||||
|
|
||||||
ANSIBLE_METADATA = {'metadata_version': '1.0',
|
ANSIBLE_METADATA = {
|
||||||
|
'metadata_version': '1.0',
|
||||||
'status': ['preview'],
|
'status': ['preview'],
|
||||||
'supported_by': 'community'}
|
'supported_by': 'community'
|
||||||
|
}
|
||||||
|
|
||||||
DOCUMENTATION = '''
|
DOCUMENTATION = '''
|
||||||
---
|
---
|
||||||
|
@ -116,42 +117,19 @@ EXAMPLES = '''
|
||||||
'''
|
'''
|
||||||
|
|
||||||
RETURN = '''
|
RETURN = '''
|
||||||
proposed:
|
commands:
|
||||||
description: k/v pairs of parameters passed into module
|
|
||||||
returned: always
|
|
||||||
type: dict
|
|
||||||
sample: {"group": "30", "version": "2", "vip": "10.30.1.1"}
|
|
||||||
existing:
|
|
||||||
description: k/v pairs of existing hsrp info on the interface
|
|
||||||
returned: always
|
|
||||||
type: dict
|
|
||||||
sample: {}
|
|
||||||
end_state:
|
|
||||||
description: k/v pairs of hsrp after module execution
|
|
||||||
returned: always
|
|
||||||
type: dict
|
|
||||||
sample: {"auth_string": "cisco", "auth_type": "text",
|
|
||||||
"group": "30", "interface": "vlan10", "preempt": "disabled",
|
|
||||||
"priority": "100", "version": "2", "vip": "10.30.1.1"}
|
|
||||||
updates:
|
|
||||||
description: commands sent to the device
|
description: commands sent to the device
|
||||||
returned: always
|
returned: always
|
||||||
type: list
|
type: list
|
||||||
sample: ["interface vlan10", "hsrp version 2", "hsrp 30", "ip 10.30.1.1"]
|
sample: ["interface vlan10", "hsrp version 2", "hsrp 30", "ip 10.30.1.1"]
|
||||||
changed:
|
|
||||||
description: check to see if a change was made on the device
|
|
||||||
returned: always
|
|
||||||
type: boolean
|
|
||||||
sample: true
|
|
||||||
'''
|
'''
|
||||||
|
|
||||||
from ansible.module_utils.nxos import get_config, load_config, run_commands
|
from ansible.module_utils.nxos import load_config, run_commands
|
||||||
from ansible.module_utils.nxos import nxos_argument_spec, check_args
|
from ansible.module_utils.nxos import nxos_argument_spec, check_args
|
||||||
from ansible.module_utils.basic import AnsibleModule
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
|
||||||
|
|
||||||
|
def execute_show_command(command, module):
|
||||||
def execute_show_command(command, module, command_type='cli_show'):
|
|
||||||
if module.params['transport'] == 'cli':
|
if module.params['transport'] == 'cli':
|
||||||
command += ' | json'
|
command += ' | json'
|
||||||
cmds = [command]
|
cmds = [command]
|
||||||
|
@ -165,7 +143,7 @@ def execute_show_command(command, module, command_type='cli_show'):
|
||||||
|
|
||||||
def apply_key_map(key_map, table):
|
def apply_key_map(key_map, table):
|
||||||
new_dict = {}
|
new_dict = {}
|
||||||
for key, value in table.items():
|
for key in table:
|
||||||
new_key = key_map.get(key)
|
new_key = key_map.get(key)
|
||||||
if new_key:
|
if new_key:
|
||||||
value = table.get(key)
|
value = table.get(key)
|
||||||
|
@ -197,9 +175,12 @@ def get_interface_mode(interface, intf_type, module):
|
||||||
command = 'show interface {0}'.format(interface)
|
command = 'show interface {0}'.format(interface)
|
||||||
interface = {}
|
interface = {}
|
||||||
mode = 'unknown'
|
mode = 'unknown'
|
||||||
|
try:
|
||||||
|
body = execute_show_command(command, module)[0]
|
||||||
|
except IndexError:
|
||||||
|
return None
|
||||||
|
|
||||||
if intf_type in ['ethernet', 'portchannel']:
|
if intf_type in ['ethernet', 'portchannel']:
|
||||||
body = execute_show_command(command, module)[0]
|
|
||||||
interface_table = body['TABLE_interface']['ROW_interface']
|
interface_table = body['TABLE_interface']['ROW_interface']
|
||||||
mode = str(interface_table.get('eth_mode', 'layer3'))
|
mode = str(interface_table.get('eth_mode', 'layer3'))
|
||||||
if mode == 'access' or mode == 'trunk':
|
if mode == 'access' or mode == 'trunk':
|
||||||
|
@ -211,12 +192,12 @@ def get_interface_mode(interface, intf_type, module):
|
||||||
|
|
||||||
def get_hsrp_groups_on_interfaces(device, module):
|
def get_hsrp_groups_on_interfaces(device, module):
|
||||||
command = 'show hsrp all'
|
command = 'show hsrp all'
|
||||||
body = execute_show_command(command, module)
|
|
||||||
hsrp = {}
|
hsrp = {}
|
||||||
|
|
||||||
try:
|
try:
|
||||||
get_data = body[0]['TABLE_grp_detail']['ROW_grp_detail']
|
body = execute_show_command(command, module)[0]
|
||||||
except (KeyError, AttributeError):
|
get_data = body['TABLE_grp_detail']['ROW_grp_detail']
|
||||||
|
except (IndexError, KeyError, AttributeError):
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
for entry in get_data:
|
for entry in get_data:
|
||||||
|
@ -232,7 +213,6 @@ def get_hsrp_groups_on_interfaces(device, module):
|
||||||
|
|
||||||
def get_hsrp_group(group, interface, module):
|
def get_hsrp_group(group, interface, module):
|
||||||
command = 'show hsrp group {0}'.format(group)
|
command = 'show hsrp group {0}'.format(group)
|
||||||
body = execute_show_command(command, module)
|
|
||||||
hsrp = {}
|
hsrp = {}
|
||||||
|
|
||||||
hsrp_key = {
|
hsrp_key = {
|
||||||
|
@ -247,7 +227,8 @@ def get_hsrp_group(group, interface, module):
|
||||||
}
|
}
|
||||||
|
|
||||||
try:
|
try:
|
||||||
hsrp_table = body[0]['TABLE_grp_detail']['ROW_grp_detail']
|
body = execute_show_command(command, module)[0]
|
||||||
|
hsrp_table = body['TABLE_grp_detail']['ROW_grp_detail']
|
||||||
except (AttributeError, IndexError, TypeError):
|
except (AttributeError, IndexError, TypeError):
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
|
@ -271,9 +252,7 @@ def get_hsrp_group(group, interface, module):
|
||||||
|
|
||||||
|
|
||||||
def get_commands_remove_hsrp(group, interface):
|
def get_commands_remove_hsrp(group, interface):
|
||||||
commands = []
|
commands = ['interface {0}'.format(interface), 'no hsrp {0}'.format(group)]
|
||||||
commands.append('interface {0}'.format(interface))
|
|
||||||
commands.append('no hsrp {0}'.format(group))
|
|
||||||
return commands
|
return commands
|
||||||
|
|
||||||
|
|
||||||
|
@ -295,7 +274,7 @@ def get_commands_config_hsrp(delta, interface, args):
|
||||||
elif preempt == 'disabled':
|
elif preempt == 'disabled':
|
||||||
delta['preempt'] = 'no preempt'
|
delta['preempt'] = 'no preempt'
|
||||||
|
|
||||||
for key, value in delta.items():
|
for key in delta:
|
||||||
command = config_args.get(key, 'DNE').format(**delta)
|
command = config_args.get(key, 'DNE').format(**delta)
|
||||||
if command and command != 'DNE':
|
if command and command != 'DNE':
|
||||||
if key == 'group':
|
if key == 'group':
|
||||||
|
@ -391,13 +370,11 @@ def main():
|
||||||
interface=dict(required=True),
|
interface=dict(required=True),
|
||||||
version=dict(choices=['1', '2'], default='2', required=False),
|
version=dict(choices=['1', '2'], default='2', required=False),
|
||||||
priority=dict(type='str', required=False),
|
priority=dict(type='str', required=False),
|
||||||
preempt=dict(type='str', choices=['disabled', 'enabled'],
|
preempt=dict(type='str', choices=['disabled', 'enabled'], required=False),
|
||||||
required=False),
|
|
||||||
vip=dict(type='str', required=False),
|
vip=dict(type='str', required=False),
|
||||||
auth_type=dict(choices=['text', 'md5'], required=False),
|
auth_type=dict(choices=['text', 'md5'], required=False),
|
||||||
auth_string=dict(type='str', required=False),
|
auth_string=dict(type='str', required=False),
|
||||||
state=dict(choices=['absent', 'present'], required=False,
|
state=dict(choices=['absent', 'present'], required=False, default='present'),
|
||||||
default='present'),
|
|
||||||
include_defaults=dict(default=True),
|
include_defaults=dict(default=True),
|
||||||
config=dict(),
|
config=dict(),
|
||||||
save=dict(type='bool', default=False)
|
save=dict(type='bool', default=False)
|
||||||
|
@ -405,12 +382,11 @@ def main():
|
||||||
|
|
||||||
argument_spec.update(nxos_argument_spec)
|
argument_spec.update(nxos_argument_spec)
|
||||||
|
|
||||||
module = AnsibleModule(argument_spec=argument_spec,
|
module = AnsibleModule(argument_spec=argument_spec, supports_check_mode=True)
|
||||||
supports_check_mode=True)
|
|
||||||
|
|
||||||
warnings = list()
|
warnings = list()
|
||||||
check_args(module, warnings)
|
check_args(module, warnings)
|
||||||
|
results = dict(changed=False, warnings=warnings)
|
||||||
|
|
||||||
interface = module.params['interface'].lower()
|
interface = module.params['interface'].lower()
|
||||||
group = module.params['group']
|
group = module.params['group']
|
||||||
|
@ -470,8 +446,6 @@ def main():
|
||||||
module.fail_json(msg="Existing auth_type is md5. It's recommended "
|
module.fail_json(msg="Existing auth_type is md5. It's recommended "
|
||||||
"to use HSRP v2 when using md5")
|
"to use HSRP v2 when using md5")
|
||||||
|
|
||||||
changed = False
|
|
||||||
end_state = existing
|
|
||||||
commands = []
|
commands = []
|
||||||
if state == 'present':
|
if state == 'present':
|
||||||
delta = dict(
|
delta = dict(
|
||||||
|
@ -487,28 +461,20 @@ def main():
|
||||||
|
|
||||||
if commands:
|
if commands:
|
||||||
if module.check_mode:
|
if module.check_mode:
|
||||||
module.exit_json(changed=True, commands=commands)
|
module.exit_json(**results)
|
||||||
else:
|
else:
|
||||||
load_config(module, commands)
|
load_config(module, commands)
|
||||||
if transport == 'cli':
|
if transport == 'cli':
|
||||||
body = run_commands(module, commands)
|
body = run_commands(module, commands)
|
||||||
validate_config(body, vip, module)
|
validate_config(body, vip, module)
|
||||||
changed = True
|
results['changed'] = True
|
||||||
end_state = get_hsrp_group(group, interface, module)
|
end_state = get_hsrp_group(group, interface, module)
|
||||||
if 'configure' in commands:
|
if 'configure' in commands:
|
||||||
commands.pop(0)
|
commands.pop(0)
|
||||||
|
|
||||||
results = {}
|
results['commands'] = commands
|
||||||
results['proposed'] = proposed
|
|
||||||
results['existing'] = existing
|
|
||||||
results['end_state'] = end_state
|
|
||||||
results['updates'] = commands
|
|
||||||
results['changed'] = changed
|
|
||||||
results['warnings'] = warnings
|
|
||||||
|
|
||||||
module.exit_json(**results)
|
module.exit_json(**results)
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
main()
|
main()
|
||||||
|
|
||||||
|
|
|
@ -473,7 +473,6 @@ lib/ansible/modules/network/nxos/nxos_facts.py
|
||||||
lib/ansible/modules/network/nxos/nxos_feature.py
|
lib/ansible/modules/network/nxos/nxos_feature.py
|
||||||
lib/ansible/modules/network/nxos/nxos_gir.py
|
lib/ansible/modules/network/nxos/nxos_gir.py
|
||||||
lib/ansible/modules/network/nxos/nxos_gir_profile_management.py
|
lib/ansible/modules/network/nxos/nxos_gir_profile_management.py
|
||||||
lib/ansible/modules/network/nxos/nxos_hsrp.py
|
|
||||||
lib/ansible/modules/network/nxos/nxos_igmp.py
|
lib/ansible/modules/network/nxos/nxos_igmp.py
|
||||||
lib/ansible/modules/network/nxos/nxos_igmp_interface.py
|
lib/ansible/modules/network/nxos/nxos_igmp_interface.py
|
||||||
lib/ansible/modules/network/nxos/nxos_igmp_snooping.py
|
lib/ansible/modules/network/nxos/nxos_igmp_snooping.py
|
||||||
|
|
60
test/units/modules/network/nxos/test_nxos_hsrp.py
Normal file
60
test/units/modules/network/nxos/test_nxos_hsrp.py
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
# (c) 2016 Red Hat Inc.
|
||||||
|
#
|
||||||
|
# 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/>.
|
||||||
|
|
||||||
|
# Make coding more python3-ish
|
||||||
|
from __future__ import (absolute_import, division, print_function)
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
import json
|
||||||
|
|
||||||
|
from ansible.compat.tests.mock import patch
|
||||||
|
from ansible.modules.network.nxos import nxos_hsrp
|
||||||
|
from .nxos_module import TestNxosModule, load_fixture, set_module_args
|
||||||
|
|
||||||
|
|
||||||
|
class TestNxosHsrpModule(TestNxosModule):
|
||||||
|
|
||||||
|
module = nxos_hsrp
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.mock_run_commands = patch('ansible.modules.network.nxos.nxos_hsrp.run_commands')
|
||||||
|
self.run_commands = self.mock_run_commands.start()
|
||||||
|
|
||||||
|
self.mock_load_config = patch('ansible.modules.network.nxos.nxos_hsrp.load_config')
|
||||||
|
self.load_config = self.mock_load_config.start()
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
self.mock_run_commands.stop()
|
||||||
|
self.mock_load_config.stop()
|
||||||
|
|
||||||
|
def load_fixtures(self, commands=None):
|
||||||
|
self.load_config.return_value = None
|
||||||
|
|
||||||
|
def test_nxos_hsrp(self):
|
||||||
|
set_module_args(dict(group='10',
|
||||||
|
vip='192.0.2.2',
|
||||||
|
priority='150',
|
||||||
|
interface='Ethernet1/2',
|
||||||
|
preempt='enabled',
|
||||||
|
host='192.0.2.1'))
|
||||||
|
result = self.execute_module(changed=True)
|
||||||
|
self.assertEqual(sorted(result['commands']), sorted(['interface ethernet1/2',
|
||||||
|
'hsrp version 2',
|
||||||
|
'hsrp 10',
|
||||||
|
'priority 150',
|
||||||
|
'ip 192.0.2.2',
|
||||||
|
'preempt']))
|
Loading…
Reference in a new issue