diff --git a/dev-requirements.txt b/dev-requirements.txt index 61b2b21..9bcde5f 100644 --- a/dev-requirements.txt +++ b/dev-requirements.txt @@ -5,3 +5,5 @@ pytest-cov flake8 flake8-isort tox +mock +pyfakefs diff --git a/netbox_agent/ipmi.py b/netbox_agent/ipmi.py index 10ccb8c..a0c9df6 100644 --- a/netbox_agent/ipmi.py +++ b/netbox_agent/ipmi.py @@ -37,7 +37,12 @@ class IPMI(): """ def __init__(self): - self.ret, self.output = subprocess.getstatusoutput('ipmitool lan print') + try: + self.ret, self.output = subprocess.getstatusoutput('ipmitool lan print') + except Exception as e: + self.ret = 42 + self.output = str(e) + if self.ret != 0: logging.error('Cannot get ipmi info: {}'.format(self.output)) diff --git a/tests/conftest.py b/tests/conftest.py index 4a97a8c..9966e37 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -15,7 +15,7 @@ def get_fixture_paths(path): def parametrize_with_fixtures(path, base_path='tests/fixtures', - argname='fixture', only_filenames=None): + argname='dmi_fixture', only_filenames=None): path = os.path.join(base_path, path) fixture_paths = get_fixture_paths(path) argvalues = [] diff --git a/tests/constants.py b/tests/constants.py new file mode 100644 index 0000000..dd58e51 --- /dev/null +++ b/tests/constants.py @@ -0,0 +1 @@ +DEFAULT_DATACENTER = 'DC1' diff --git a/tests/fixtures/dmidecode/HP_SL4540_Gen8 b/tests/fixtures/dmidecode/HP_SL4540_Gen8 index f25f9f3..e50e25c 100644 --- a/tests/fixtures/dmidecode/HP_SL4540_Gen8 +++ b/tests/fixtures/dmidecode/HP_SL4540_Gen8 @@ -829,7 +829,7 @@ System Power Supply Location: Not Specified Name: Power Supply 2 Manufacturer: HP - Serial Number: 4242 + Serial Number: 4243 Asset Tag: Not Specified Model Part Number: 656364-B21 Revision: Not Specified @@ -846,7 +846,7 @@ System Power Supply Location: Not Specified Name: Power Supply 3 Manufacturer: HP - Serial Number: 4242 + Serial Number: 4244 Asset Tag: Not Specified Model Part Number: 656364-B21 Revision: Not Specified @@ -863,7 +863,7 @@ System Power Supply Location: Not Specified Name: Power Supply 4 Manufacturer: HP - Serial Number: 4242 + Serial Number: 4245 Asset Tag: Not Specified Model Part Number: 656364-B21 Revision: Not Specified diff --git a/tests/network.py b/tests/network.py index 7b341ed..b8706a0 100644 --- a/tests/network.py +++ b/tests/network.py @@ -6,8 +6,8 @@ from tests.conftest import parametrize_with_fixtures 'lldp/', only_filenames=[ 'dedibox1.txt', ]) -def test_lldp_parse_with_port_desc(fixture): - lldp = LLDP(fixture) +def test_lldp_parse_with_port_desc(dmi_fixture): + lldp = LLDP(dmi_fixture) assert lldp.get_switch_port('enp1s0f0') == 'RJ-9' @@ -15,6 +15,6 @@ def test_lldp_parse_with_port_desc(fixture): 'lldp/', only_filenames=[ 'qfx.txt', ]) -def test_lldp_parse_without_ifname(fixture): - lldp = LLDP(fixture) +def test_lldp_parse_without_ifname(dmi_fixture): + lldp = LLDP(dmi_fixture) assert lldp.get_switch_port('eth0') == 'xe-0/0/1' diff --git a/tests/server.py b/tests/server.py index 6e781ad..167680d 100644 --- a/tests/server.py +++ b/tests/server.py @@ -1,11 +1,20 @@ + +import mock +import netifaces + +from netbox_agent.config import config +from netbox_agent.config import netbox_instance as nb from netbox_agent.dmidecode import parse from netbox_agent.server import ServerBase +from netbox_agent.vendors.hp import HPHost from tests.conftest import parametrize_with_fixtures +from tests.constants import DEFAULT_DATACENTER +from tests.utils import setup_netbox @parametrize_with_fixtures('dmidecode/') -def test_init(fixture): - dmi = parse(fixture) +def test_init(dmi_fixture): + dmi = parse(dmi_fixture) server = ServerBase(dmi) assert server @@ -17,8 +26,8 @@ def test_init(fixture): 'HP_DL380p_Gen8', 'HP_SL4540_Gen8' ]) -def test_hp_service_tag(fixture): - dmi = parse(fixture) +def test_hp_service_tag(dmi_fixture): + dmi = parse(dmi_fixture) server = ServerBase(dmi) assert server.get_service_tag() == '4242' @@ -27,8 +36,8 @@ def test_hp_service_tag(fixture): 'dmidecode/', only_filenames=[ 'unknown.txt' ]) -def test_generic_host_service_tag(fixture): - dmi = parse(fixture) +def test_generic_host_service_tag(dmi_fixture): + dmi = parse(dmi_fixture) server = ServerBase(dmi) assert server.get_service_tag() == '42' @@ -37,7 +46,56 @@ def test_generic_host_service_tag(fixture): 'dmidecode/', only_filenames=[ 'unknown.txt' ]) -def test_generic_host_product_name(fixture): - dmi = parse(fixture) +def test_generic_host_product_name(dmi_fixture): + dmi = parse(dmi_fixture) server = ServerBase(dmi) assert server.get_product_name() == 'SR' + + +@mock.patch('netifaces.ifaddresses') +@mock.patch('netifaces.interfaces') +@parametrize_with_fixtures( + 'dmidecode/', only_filenames=[ + 'HP_SL4540_Gen8', + ], argname='dmi_fixture') +def test_create_server( + mock_interfaces, + mock_ifaddresses, + fs, + dmi_fixture, +): + fake_addresses = {} + fake_addresses[netifaces.AF_INET] = [{'addr': '42.42.42.42', 'netmask': '255.255.255.0'}] + fake_addresses[netifaces.AF_LINK] = [{'addr': 'a8:1e:84:f2:9e:69'}] + + mock_interfaces.return_value = ['enp1s0f0'] + mock_ifaddresses.return_value = fake_addresses + + dmi = parse(dmi_fixture) + server = HPHost(dmi) + + setup_netbox( + DEFAULT_DATACENTER, + 'Server', + 'HP', + server.get_product_name(), + ) + + # Create fake /sys/class/net directory with fake interface and MAC addr + fs.create_file('/tmp/enp1s0f0/address', contents='a8:1e:84:f2:9e:69') + fs.create_symlink('/sys/class/net/enp1s0f0', '/tmp/enp1s0f0') + + server.netbox_create(config) + + # Check serial tag is correct + assert server.get_service_tag() == '4242' + network_card = server.network.get_netbox_network_card({'name': 'enp1s0f0', 'mac': None}) + + # Check network card is correct + assert network_card.name == 'enp1s0f0' + + # Check IP on network card + ips = nb.ipam.ip_addresses.filter( + interface_id=network_card.id + ) + assert ips[0].address == '42.42.42.42/24' diff --git a/tests/utils.py b/tests/utils.py new file mode 100644 index 0000000..ef6a140 --- /dev/null +++ b/tests/utils.py @@ -0,0 +1,51 @@ + +import pynetbox + +from netbox_agent.config import netbox_instance as nb + + +def setup_netbox(dc, device_role, manufacturer, model): + try: + nb_dc = nb.dcim.sites.create( + name=dc, + slug=dc.lower(), + ) + except pynetbox.RequestError: + nb_dc = nb.dcim.sites.get( + slug=dc.lower() + ) + + try: + nb_manufacturer = nb.dcim.manufacturers.create( + name=manufacturer, + slug=manufacturer.lower(), + ) + except pynetbox.RequestError: + nb_manufacturer = nb.dcim.manufacturers.get( + slug=manufacturer.lower() + ) + + try: + nb_device_role = nb.dcim.device_roles.create( + name=device_role, + slug=device_role.lower(), + color='f44336', + ) + except pynetbox.RequestError: + nb_device_role = nb.dcim.device_roles.get( + slug=device_role.lower(), + ) + + try: + nb_device_type = nb.dcim.device_types.create( + model=model, + slug=model.lower().replace(' ', '_'), + device_role=nb_device_role.id, + manufacturer=nb_manufacturer.id, + ) + except pynetbox.RequestError: + nb_device_type = nb.dcim.device_types.get( + slug=model.lower().replace(' ', '_'), + ) + + return nb_dc, nb_manufacturer, nb_device_role, nb_device_type