Support NetBSD 7.1+ style ifconfig -a output (#25442)
* Support NetBSD 7.1+ style ifconfig -a output network facts on NetBSD after 7.1 cvs would fail because of format changes in 'ifconfig -a' output. update code to support new and old format. add unit tests for both based on examples from Bruce V Chiarelli. * wrap use of interfaces.keys() in list() for py3 compat * sort interface ids for stability
This commit is contained in:
parent
d7b3ace0b0
commit
dde3dac9f8
3 changed files with 222 additions and 17 deletions
|
@ -51,7 +51,7 @@ class GenericBsdIfconfigNetwork(Network):
|
||||||
|
|
||||||
self.merge_default_interface(default_ipv4, interfaces, 'ipv4')
|
self.merge_default_interface(default_ipv4, interfaces, 'ipv4')
|
||||||
self.merge_default_interface(default_ipv6, interfaces, 'ipv6')
|
self.merge_default_interface(default_ipv6, interfaces, 'ipv6')
|
||||||
network_facts['interfaces'] = interfaces.keys()
|
network_facts['interfaces'] = sorted(list(interfaces.keys()))
|
||||||
|
|
||||||
for iface in interfaces:
|
for iface in interfaces:
|
||||||
network_facts[iface] = interfaces[iface]
|
network_facts[iface] = interfaces[iface]
|
||||||
|
@ -198,7 +198,23 @@ class GenericBsdIfconfigNetwork(Network):
|
||||||
# inet alias 127.1.1.1 netmask 0xff000000
|
# inet alias 127.1.1.1 netmask 0xff000000
|
||||||
if words[1] == 'alias':
|
if words[1] == 'alias':
|
||||||
del words[1]
|
del words[1]
|
||||||
|
|
||||||
address = {'address': words[1]}
|
address = {'address': words[1]}
|
||||||
|
# cidr style ip address (eg, 127.0.0.1/24) in inet line
|
||||||
|
# used in netbsd ifconfig -e output after 7.1
|
||||||
|
if '/' in address['address']:
|
||||||
|
ip_address, cidr_mask = address['address'].split('/')
|
||||||
|
|
||||||
|
address['address'] = ip_address
|
||||||
|
|
||||||
|
netmask_length = int(cidr_mask)
|
||||||
|
netmask_bin = (1 << 32) - (1 << 32 >> int(netmask_length))
|
||||||
|
address['netmask'] = socket.inet_ntoa(struct.pack('!L', netmask_bin))
|
||||||
|
|
||||||
|
if len(words) > 5:
|
||||||
|
address['broadcast'] = words[3]
|
||||||
|
|
||||||
|
else:
|
||||||
# deal with hex netmask
|
# deal with hex netmask
|
||||||
if re.match('([0-9a-f]){8}', words[3]) and len(words[3]) == 8:
|
if re.match('([0-9a-f]){8}', words[3]) and len(words[3]) == 8:
|
||||||
words[3] = '0x' + words[3]
|
words[3] = '0x' + words[3]
|
||||||
|
@ -211,11 +227,13 @@ class GenericBsdIfconfigNetwork(Network):
|
||||||
address_bin = struct.unpack('!L', socket.inet_aton(address['address']))[0]
|
address_bin = struct.unpack('!L', socket.inet_aton(address['address']))[0]
|
||||||
netmask_bin = struct.unpack('!L', socket.inet_aton(address['netmask']))[0]
|
netmask_bin = struct.unpack('!L', socket.inet_aton(address['netmask']))[0]
|
||||||
address['network'] = socket.inet_ntoa(struct.pack('!L', address_bin & netmask_bin))
|
address['network'] = socket.inet_ntoa(struct.pack('!L', address_bin & netmask_bin))
|
||||||
|
if 'broadcast' not in address:
|
||||||
# broadcast may be given or we need to calculate
|
# broadcast may be given or we need to calculate
|
||||||
if len(words) > 5:
|
if len(words) > 5:
|
||||||
address['broadcast'] = words[5]
|
address['broadcast'] = words[5]
|
||||||
else:
|
else:
|
||||||
address['broadcast'] = socket.inet_ntoa(struct.pack('!L', address_bin | (~netmask_bin & 0xffffffff)))
|
address['broadcast'] = socket.inet_ntoa(struct.pack('!L', address_bin | (~netmask_bin & 0xffffffff)))
|
||||||
|
|
||||||
# add to our list of addresses
|
# add to our list of addresses
|
||||||
if not words[1].startswith('127.'):
|
if not words[1].startswith('127.'):
|
||||||
ips['all_ipv4_addresses'].append(address['address'])
|
ips['all_ipv4_addresses'].append(address['address'])
|
||||||
|
@ -223,10 +241,22 @@ class GenericBsdIfconfigNetwork(Network):
|
||||||
|
|
||||||
def parse_inet6_line(self, words, current_if, ips):
|
def parse_inet6_line(self, words, current_if, ips):
|
||||||
address = {'address': words[1]}
|
address = {'address': words[1]}
|
||||||
|
|
||||||
|
# using cidr style addresses, ala NetBSD ifconfig post 7.1
|
||||||
|
if '/' in address['address']:
|
||||||
|
ip_address, cidr_mask = address['address'].split('/')
|
||||||
|
|
||||||
|
address['address'] = ip_address
|
||||||
|
address['prefix'] = cidr_mask
|
||||||
|
|
||||||
|
if len(words) > 5:
|
||||||
|
address['scope'] = words[5]
|
||||||
|
else:
|
||||||
if (len(words) >= 4) and (words[2] == 'prefixlen'):
|
if (len(words) >= 4) and (words[2] == 'prefixlen'):
|
||||||
address['prefix'] = words[3]
|
address['prefix'] = words[3]
|
||||||
if (len(words) >= 6) and (words[4] == 'scopeid'):
|
if (len(words) >= 6) and (words[4] == 'scopeid'):
|
||||||
address['scope'] = words[5]
|
address['scope'] = words[5]
|
||||||
|
|
||||||
localhost6 = ['::1', '::1/128', 'fe80::1%lo0']
|
localhost6 = ['::1', '::1/128', 'fe80::1%lo0']
|
||||||
if address['address'] not in localhost6:
|
if address['address'] not in localhost6:
|
||||||
ips['all_ipv6_addresses'].append(address['address'])
|
ips['all_ipv6_addresses'].append(address['address'])
|
||||||
|
|
0
test/units/module_utils/facts/network/__init__.py
Normal file
0
test/units/module_utils/facts/network/__init__.py
Normal file
175
test/units/module_utils/facts/network/test_generic_bsd.py
Normal file
175
test/units/module_utils/facts/network/test_generic_bsd.py
Normal file
|
@ -0,0 +1,175 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
#
|
||||||
|
# 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)
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
from ansible.compat.tests.mock import Mock
|
||||||
|
from ansible.compat.tests import unittest
|
||||||
|
|
||||||
|
from ansible.module_utils.facts.network import generic_bsd
|
||||||
|
|
||||||
|
|
||||||
|
def get_bin_path(command):
|
||||||
|
if command == 'ifconfig':
|
||||||
|
return 'fake/ifconfig'
|
||||||
|
elif command == 'route':
|
||||||
|
return 'fake/route'
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
netbsd_ifconfig_a_out_7_1 = r'''
|
||||||
|
lo0: flags=8049<UP,LOOPBACK,RUNNING,MULTICAST> mtu 33624
|
||||||
|
inet 127.0.0.1 netmask 0xff000000
|
||||||
|
inet6 ::1 prefixlen 128
|
||||||
|
inet6 fe80::1%lo0 prefixlen 64 scopeid 0x1
|
||||||
|
re0: flags=8843<UP,BROADCAST,RUNNING,SIMPLEX,MULTICAST> mtu 1500
|
||||||
|
capabilities=3f80<TSO4,IP4CSUM_Rx,IP4CSUM_Tx,TCP4CSUM_Rx,TCP4CSUM_Tx>
|
||||||
|
capabilities=3f80<UDP4CSUM_Rx,UDP4CSUM_Tx>
|
||||||
|
enabled=0
|
||||||
|
ec_capabilities=3<VLAN_MTU,VLAN_HWTAGGING>
|
||||||
|
ec_enabled=0
|
||||||
|
address: 52:54:00:63:55:af
|
||||||
|
media: Ethernet autoselect (100baseTX full-duplex)
|
||||||
|
status: active
|
||||||
|
inet 192.168.122.205 netmask 0xffffff00 broadcast 192.168.122.255
|
||||||
|
inet6 fe80::5054:ff:fe63:55af%re0 prefixlen 64 scopeid 0x2
|
||||||
|
'''
|
||||||
|
|
||||||
|
netbsd_ifconfig_a_out_post_7_1 = r'''
|
||||||
|
lo0: flags=0x8049<UP,LOOPBACK,RUNNING,MULTICAST> mtu 33624
|
||||||
|
inet 127.0.0.1/8 flags 0x0
|
||||||
|
inet6 ::1/128 flags 0x20<NODAD>
|
||||||
|
inet6 fe80::1%lo0/64 flags 0x0 scopeid 0x1
|
||||||
|
re0: flags=0x8843<UP,BROADCAST,RUNNING,SIMPLEX,MULTICAST> mtu 1500
|
||||||
|
capabilities=3f80<TSO4,IP4CSUM_Rx,IP4CSUM_Tx,TCP4CSUM_Rx,TCP4CSUM_Tx>
|
||||||
|
capabilities=3f80<UDP4CSUM_Rx,UDP4CSUM_Tx>
|
||||||
|
enabled=0
|
||||||
|
ec_capabilities=3<VLAN_MTU,VLAN_HWTAGGING>
|
||||||
|
ec_enabled=0
|
||||||
|
address: 52:54:00:63:55:af
|
||||||
|
media: Ethernet autoselect (100baseTX full-duplex)
|
||||||
|
status: active
|
||||||
|
inet 192.168.122.205/24 broadcast 192.168.122.255 flags 0x0
|
||||||
|
inet6 fe80::5054:ff:fe63:55af%re0/64 flags 0x0 scopeid 0x2
|
||||||
|
'''
|
||||||
|
|
||||||
|
NETBSD_EXPECTED = {'all_ipv4_addresses': ['192.168.122.205'],
|
||||||
|
'all_ipv6_addresses': ['fe80::5054:ff:fe63:55af%re0'],
|
||||||
|
'default_ipv4': {},
|
||||||
|
'default_ipv6': {},
|
||||||
|
'interfaces': ['lo0', 're0'],
|
||||||
|
'lo0': {'device': 'lo0',
|
||||||
|
'flags': ['UP', 'LOOPBACK', 'RUNNING', 'MULTICAST'],
|
||||||
|
'ipv4': [{'address': '127.0.0.1',
|
||||||
|
'broadcast': '127.255.255.255',
|
||||||
|
'netmask': '255.0.0.0',
|
||||||
|
'network': '127.0.0.0'}],
|
||||||
|
'ipv6': [{'address': '::1', 'prefix': '128'},
|
||||||
|
{'address': 'fe80::1%lo0', 'prefix': '64', 'scope': '0x1'}],
|
||||||
|
'macaddress': 'unknown',
|
||||||
|
'mtu': '33624',
|
||||||
|
'type': 'loopback'},
|
||||||
|
're0': {'device': 're0',
|
||||||
|
'flags': ['UP', 'BROADCAST', 'RUNNING', 'SIMPLEX', 'MULTICAST'],
|
||||||
|
'ipv4': [{'address': '192.168.122.205',
|
||||||
|
'broadcast': '192.168.122.255',
|
||||||
|
'netmask': '255.255.255.0',
|
||||||
|
'network': '192.168.122.0'}],
|
||||||
|
'ipv6': [{'address': 'fe80::5054:ff:fe63:55af%re0',
|
||||||
|
'prefix': '64',
|
||||||
|
'scope': '0x2'}],
|
||||||
|
'macaddress': 'unknown',
|
||||||
|
'media': 'Ethernet',
|
||||||
|
'media_options': [],
|
||||||
|
'media_select': 'autoselect',
|
||||||
|
'media_type': '100baseTX',
|
||||||
|
'mtu': '1500',
|
||||||
|
'status': 'active',
|
||||||
|
'type': 'ether'}}
|
||||||
|
|
||||||
|
|
||||||
|
def run_command_old_ifconfig(command):
|
||||||
|
if command == 'fake/route':
|
||||||
|
return 0, 'Foo', ''
|
||||||
|
if command == ['fake/ifconfig', '-a']:
|
||||||
|
return 0, netbsd_ifconfig_a_out_7_1, ''
|
||||||
|
return 1, '', ''
|
||||||
|
|
||||||
|
|
||||||
|
def run_command_post_7_1_ifconfig(command):
|
||||||
|
if command == 'fake/route':
|
||||||
|
return 0, 'Foo', ''
|
||||||
|
if command == ['fake/ifconfig', '-a']:
|
||||||
|
return 0, netbsd_ifconfig_a_out_post_7_1, ''
|
||||||
|
return 1, '', ''
|
||||||
|
|
||||||
|
|
||||||
|
class TestGenericBsdNetworkNetBSD(unittest.TestCase):
|
||||||
|
gather_subset = ['all']
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.maxDiff = None
|
||||||
|
self.longMessage = True
|
||||||
|
|
||||||
|
# TODO: extract module run_command/get_bin_path usage to methods I can mock without mocking all of run_command
|
||||||
|
def test(self):
|
||||||
|
module = self._mock_module()
|
||||||
|
module.get_bin_path.side_effect = get_bin_path
|
||||||
|
module.run_command.side_effect = run_command_old_ifconfig
|
||||||
|
|
||||||
|
bsd_net = generic_bsd.GenericBsdIfconfigNetwork(module)
|
||||||
|
|
||||||
|
res = bsd_net.populate()
|
||||||
|
self.assertDictEqual(res, NETBSD_EXPECTED)
|
||||||
|
|
||||||
|
def test_ifconfig_post_7_1(self):
|
||||||
|
module = self._mock_module()
|
||||||
|
module.get_bin_path.side_effect = get_bin_path
|
||||||
|
module.run_command.side_effect = run_command_post_7_1_ifconfig
|
||||||
|
|
||||||
|
bsd_net = generic_bsd.GenericBsdIfconfigNetwork(module)
|
||||||
|
|
||||||
|
res = bsd_net.populate()
|
||||||
|
self.assertDictEqual(res, NETBSD_EXPECTED)
|
||||||
|
|
||||||
|
def test_netbsd_ifconfig_old_and_new(self):
|
||||||
|
module_new = self._mock_module()
|
||||||
|
module_new.get_bin_path.side_effect = get_bin_path
|
||||||
|
module_new.run_command.side_effect = run_command_post_7_1_ifconfig
|
||||||
|
|
||||||
|
bsd_net_new = generic_bsd.GenericBsdIfconfigNetwork(module_new)
|
||||||
|
res_new = bsd_net_new.populate()
|
||||||
|
|
||||||
|
module_old = self._mock_module()
|
||||||
|
module_old.get_bin_path.side_effect = get_bin_path
|
||||||
|
module_old.run_command.side_effect = run_command_old_ifconfig
|
||||||
|
|
||||||
|
bsd_net_old = generic_bsd.GenericBsdIfconfigNetwork(module_old)
|
||||||
|
res_old = bsd_net_old.populate()
|
||||||
|
|
||||||
|
self.assertDictEqual(res_old, res_new)
|
||||||
|
self.assertDictEqual(res_old, NETBSD_EXPECTED)
|
||||||
|
self.assertDictEqual(res_new, NETBSD_EXPECTED)
|
||||||
|
|
||||||
|
def _mock_module(self):
|
||||||
|
mock_module = Mock()
|
||||||
|
mock_module.params = {'gather_subset': self.gather_subset,
|
||||||
|
'gather_timeout': 5,
|
||||||
|
'filter': '*'}
|
||||||
|
mock_module.get_bin_path = Mock(return_value=None)
|
||||||
|
return mock_module
|
Loading…
Reference in a new issue