mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-24 21:14:00 -07:00 
			
		
		
		
	Fix ios_facts return values (#35239)
* Revert model and serialnum to older version * Add stacked versions of model and serialnum as separate facts * Add unit test to check stacked output * Alter model regex to address #34768
This commit is contained in:
		
					parent
					
						
							
								f5022de5d6
							
						
					
				
			
			
				commit
				
					
						4a79112e5c
					
				
			
		
					 3 changed files with 157 additions and 27 deletions
				
			
		|  | @ -49,33 +49,19 @@ options: | ||||||
| """ | """ | ||||||
| 
 | 
 | ||||||
| EXAMPLES = """ | EXAMPLES = """ | ||||||
| # Note: examples below use the following provider dict to handle |  | ||||||
| #       transport and authentication to the node. |  | ||||||
| --- |  | ||||||
| vars: |  | ||||||
|   cli: |  | ||||||
|     host: "{{ inventory_hostname }}" |  | ||||||
|     username: cisco |  | ||||||
|     password: cisco |  | ||||||
|     transport: cli |  | ||||||
| 
 |  | ||||||
| --- |  | ||||||
| # Collect all facts from the device | # Collect all facts from the device | ||||||
| - ios_facts: | - ios_facts: | ||||||
|     gather_subset: all |     gather_subset: all | ||||||
|     provider: "{{ cli }}" |  | ||||||
| 
 | 
 | ||||||
| # Collect only the config and default facts | # Collect only the config and default facts | ||||||
| - ios_facts: | - ios_facts: | ||||||
|     gather_subset: |     gather_subset: | ||||||
|       - config |       - config | ||||||
|     provider: "{{ cli }}" |  | ||||||
| 
 | 
 | ||||||
| # Do not collect hardware facts | # Do not collect hardware facts | ||||||
| - ios_facts: | - ios_facts: | ||||||
|     gather_subset: |     gather_subset: | ||||||
|       - "!hardware" |       - "!hardware" | ||||||
|     provider: "{{ cli }}" |  | ||||||
| """ | """ | ||||||
| 
 | 
 | ||||||
| RETURN = """ | RETURN = """ | ||||||
|  | @ -105,6 +91,14 @@ ansible_net_image: | ||||||
|   description: The image file the device is running |   description: The image file the device is running | ||||||
|   returned: always |   returned: always | ||||||
|   type: string |   type: string | ||||||
|  | ansible_net_stacked_models: | ||||||
|  |   description: The model names of each device in the stack | ||||||
|  |   returned: when multiple devices are configured in a stack | ||||||
|  |   type: list | ||||||
|  | ansible_net_stacked_serialnums: | ||||||
|  |   description: The serial numbers of each device in the stack | ||||||
|  |   returned: when multiple devices are configured in a stack | ||||||
|  |   type: list | ||||||
| 
 | 
 | ||||||
| # hardware | # hardware | ||||||
| ansible_net_filesystems: | ansible_net_filesystems: | ||||||
|  | @ -163,10 +157,10 @@ class FactsBase(object): | ||||||
|         self.responses = None |         self.responses = None | ||||||
| 
 | 
 | ||||||
|     def populate(self): |     def populate(self): | ||||||
|         self.responses = run_commands(self.module, self.COMMANDS, check_rc=False) |         self.responses = run_commands(self.module, commands=self.COMMANDS, check_rc=False) | ||||||
| 
 | 
 | ||||||
|     def run(self, cmd): |     def run(self, cmd): | ||||||
|         return run_commands(self.module, cmd, check_rc=False) |         return run_commands(self.module, commands=cmd, check_rc=False) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class Default(FactsBase): | class Default(FactsBase): | ||||||
|  | @ -182,6 +176,7 @@ class Default(FactsBase): | ||||||
|             self.facts['model'] = self.parse_model(data) |             self.facts['model'] = self.parse_model(data) | ||||||
|             self.facts['image'] = self.parse_image(data) |             self.facts['image'] = self.parse_image(data) | ||||||
|             self.facts['hostname'] = self.parse_hostname(data) |             self.facts['hostname'] = self.parse_hostname(data) | ||||||
|  |             self.parse_stacks(data) | ||||||
| 
 | 
 | ||||||
|     def parse_version(self, data): |     def parse_version(self, data): | ||||||
|         match = re.search(r'Version (\S+?)(?:,\s|\s)', data) |         match = re.search(r'Version (\S+?)(?:,\s|\s)', data) | ||||||
|  | @ -194,13 +189,9 @@ class Default(FactsBase): | ||||||
|             return match.group(1) |             return match.group(1) | ||||||
| 
 | 
 | ||||||
|     def parse_model(self, data): |     def parse_model(self, data): | ||||||
|         match = re.findall(r'^Model number\s+: (\S+)', data, re.M) |         match = re.search(r'^[Cc]isco (\S+).+bytes of .*memory', data, re.M) | ||||||
|         if match: |         if match: | ||||||
|             return match |             return match.group(1) | ||||||
|         else: |  | ||||||
|             match = re.search(r'^[Cc]isco (\S+).+bytes of memory', data, re.M) |  | ||||||
|             if match: |  | ||||||
|                 return [match.group(1)] |  | ||||||
| 
 | 
 | ||||||
|     def parse_image(self, data): |     def parse_image(self, data): | ||||||
|         match = re.search(r'image file is "(.+)"', data) |         match = re.search(r'image file is "(.+)"', data) | ||||||
|  | @ -208,13 +199,18 @@ class Default(FactsBase): | ||||||
|             return match.group(1) |             return match.group(1) | ||||||
| 
 | 
 | ||||||
|     def parse_serialnum(self, data): |     def parse_serialnum(self, data): | ||||||
|         match = re.findall(r'^System serial number\s+: (\S+)', data, re.M) |  | ||||||
|         if match: |  | ||||||
|             return match |  | ||||||
|         else: |  | ||||||
|         match = re.search(r'board ID (\S+)', data) |         match = re.search(r'board ID (\S+)', data) | ||||||
|         if match: |         if match: | ||||||
|                 return [match.group(1)] |             return match.group(1) | ||||||
|  | 
 | ||||||
|  |     def parse_stacks(self, data): | ||||||
|  |         match = re.findall(r'^Model number\s+: (\S+)', data, re.M) | ||||||
|  |         if match: | ||||||
|  |             self.facts['stacked_models'] = match | ||||||
|  | 
 | ||||||
|  |         match = re.findall(r'^System serial number\s+: (\S+)', data, re.M) | ||||||
|  |         if match: | ||||||
|  |             self.facts['stacked_serialnums'] = match | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class Hardware(FactsBase): | class Hardware(FactsBase): | ||||||
|  |  | ||||||
|  | @ -0,0 +1,68 @@ | ||||||
|  | Cisco Internetwork Operating System Software | ||||||
|  | IOS (tm) C3750 Software (C3750-I5-M), Version 12.1(14)EA1, RELEASE SOFTWARE (fc1) | ||||||
|  | Copyright (c) 1986-2003 by cisco Systems, Inc. | ||||||
|  | Compiled Tue 22-Jul-03 13:17 by antonino | ||||||
|  | Image text-base: 0x00003000, data-base: 0x008F0CF8 | ||||||
|  | 
 | ||||||
|  | ROM: Bootstrap program is C3750 boot loader | ||||||
|  | BOOTLDR: C3750 Boot Loader (C3750-HBOOT-M) Version 12.1(11r)AX, RELEASE SOFTWARE (fc1) | ||||||
|  | 
 | ||||||
|  | 3750RJ uptime is 1 hour, 29 minutes | ||||||
|  | System returned to ROM by power-on | ||||||
|  | System image file is "flash:c3750-i5-mz.121.14-EA1/c3750-i5-mz.121.14-EA1.bin" | ||||||
|  | 
 | ||||||
|  | cisco WS-C3750-24TS (PowerPC405) processor (revision A0) with 120822K/10240K bytes of memory. | ||||||
|  | Processor board ID CAT0726R0ZU | ||||||
|  | Last reset from power-on | ||||||
|  | Bridging software. | ||||||
|  | 2 Virtual Ethernet/IEEE 802.3 interface(s) | ||||||
|  | 48 FastEthernet/IEEE 802.3 interface(s) | ||||||
|  | 16 Gigabit Ethernet/IEEE 802.3 interface(s) | ||||||
|  | The password-recovery mechanism is enabled. | ||||||
|  | 
 | ||||||
|  | 512K bytes of flash-simulated non-volatile configuration memory. | ||||||
|  | Base ethernet MAC Address : 00:0D:29:B4:18:00 | ||||||
|  | Motherboard assembly number : 73-7055-06 | ||||||
|  | Power supply part number : 341-0034-01 | ||||||
|  | Motherboard serial number : CAT0726043V | ||||||
|  | Power supply serial number : PHI0708009K | ||||||
|  | Model revision number : A0 | ||||||
|  | Motherboard revision number : A0 | ||||||
|  | Model number : WS-C3750-24TS-E | ||||||
|  | System serial number : CAT0726R0ZU | ||||||
|  | 
 | ||||||
|  | Switch Ports Model SW Version SW Image | ||||||
|  | ------ ----- ----- ---------- ---------- | ||||||
|  | * 1 26 WS-C3750-24TS 12.1(14)EA1 C3750-I5-M | ||||||
|  | 2 26 WS-C3750-24TS 12.1(14)EA1 C3750-I5-M | ||||||
|  | 3 12 WS-C3750G-12S 12.1(14)EA1 C3750-I5-M | ||||||
|  | 
 | ||||||
|  | Switch 02 | ||||||
|  | --------- | ||||||
|  | Switch Uptime : 1 hour, 29 minutes | ||||||
|  | Base ethernet MAC Address : 00:0D:29:B4:3F:00 | ||||||
|  | Motherboard assembly number : 73-7055-06 | ||||||
|  | Power supply part number : 341-0034-01 | ||||||
|  | Motherboard serial number : CAT07260438 | ||||||
|  | Power supply serial number : PHI0708008X | ||||||
|  | Model revision number : A0 | ||||||
|  | Motherboard revision number : A0 | ||||||
|  | Model number : WS-C3750-24TS-E | ||||||
|  | System serial number : CAT0726R10A | ||||||
|  | 
 | ||||||
|  | Switch 03 | ||||||
|  | --------- | ||||||
|  | Switch Uptime : 1 hour, 29 minutes | ||||||
|  | Base ethernet MAC Address : 00:0D:BD:6A:3E:00 | ||||||
|  | Motherboard assembly number : 73-8307-06 | ||||||
|  | Power supply part number : 341-0048-01 | ||||||
|  | Motherboard serial number : CAT073205S2 | ||||||
|  | Power supply serial number : DTH0731055Z | ||||||
|  | Model revision number : A0 | ||||||
|  | Motherboard revision number : A0 | ||||||
|  | Model number : WS-C3750G-12S-E | ||||||
|  | System serial number : CAT0732R0M4 | ||||||
|  | Top assembly part number : 800-23419-01 | ||||||
|  | Top assembly revision number : A0 | ||||||
|  | 
 | ||||||
|  | Configuration register is 0xF | ||||||
							
								
								
									
										66
									
								
								test/units/modules/network/ios/test_ios_facts.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								test/units/modules/network/ios/test_ios_facts.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,66 @@ | ||||||
|  | # 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 | ||||||
|  | 
 | ||||||
|  | from ansible.compat.tests.mock import patch | ||||||
|  | from ansible.modules.network.ios import ios_facts | ||||||
|  | from units.modules.utils import set_module_args | ||||||
|  | from .ios_module import TestIosModule, load_fixture | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class TestIosFactsModule(TestIosModule): | ||||||
|  | 
 | ||||||
|  |     module = ios_facts | ||||||
|  | 
 | ||||||
|  |     def setUp(self): | ||||||
|  |         super(TestIosFactsModule, self).setUp() | ||||||
|  |         self.mock_run_commands = patch('ansible.modules.network.ios.ios_facts.run_commands') | ||||||
|  |         self.run_commands = self.mock_run_commands.start() | ||||||
|  | 
 | ||||||
|  |     def tearDown(self): | ||||||
|  |         super(TestIosFactsModule, self).tearDown() | ||||||
|  |         self.mock_run_commands.stop() | ||||||
|  | 
 | ||||||
|  |     def load_fixtures(self, commands=None): | ||||||
|  |         def load_from_file(*args, **kwargs): | ||||||
|  |             module = args | ||||||
|  |             commands = kwargs['commands'] | ||||||
|  |             output = list() | ||||||
|  | 
 | ||||||
|  |             for command in commands: | ||||||
|  |                 filename = str(command).split(' | ')[0].replace(' ', '_') | ||||||
|  |                 output.append(load_fixture('ios_facts_%s' % filename)) | ||||||
|  |             return output | ||||||
|  | 
 | ||||||
|  |         self.run_commands.side_effect = load_from_file | ||||||
|  | 
 | ||||||
|  |     def test_ios_facts_stacked(self): | ||||||
|  |         set_module_args(dict(gather_subset='default')) | ||||||
|  |         result = self.execute_module() | ||||||
|  |         self.assertEqual( | ||||||
|  |             result['ansible_facts']['ansible_net_model'], 'WS-C3750-24TS' | ||||||
|  |         ) | ||||||
|  |         self.assertEqual( | ||||||
|  |             result['ansible_facts']['ansible_net_serialnum'], 'CAT0726R0ZU' | ||||||
|  |         ) | ||||||
|  |         self.assertEqual( | ||||||
|  |             result['ansible_facts']['ansible_net_stacked_models'], ['WS-C3750-24TS-E', 'WS-C3750-24TS-E', 'WS-C3750G-12S-E'] | ||||||
|  |         ) | ||||||
|  |         self.assertEqual( | ||||||
|  |             result['ansible_facts']['ansible_net_stacked_serialnums'], ['CAT0726R0ZU', 'CAT0726R10A', 'CAT0732R0M4'] | ||||||
|  |         ) | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue