From ae32c8037571714f8d0db3407f3c4c78901a454b Mon Sep 17 00:00:00 2001 From: Ashley James Date: Fri, 1 Dec 2023 17:59:00 +0530 Subject: [PATCH 1/3] Remove test numbers in test_collector.py --- tests/unit/test_collector.py | 66 ++++++++++++++++++------------------ 1 file changed, 33 insertions(+), 33 deletions(-) diff --git a/tests/unit/test_collector.py b/tests/unit/test_collector.py index c7f4943..eb421f7 100644 --- a/tests/unit/test_collector.py +++ b/tests/unit/test_collector.py @@ -27,7 +27,7 @@ class TestCustomCollector(unittest.TestCase): """Custom test class.""" - def test_00_mega_raid_collector_not_installed(self): + def test_mega_raid_collector_not_installed(self): """Test mega raid collector when storcli is not installed.""" mega_raid_collector = MegaRAIDCollector(Mock()) mega_raid_collector.sasircu = Mock() @@ -36,7 +36,7 @@ def test_00_mega_raid_collector_not_installed(self): self.assertEqual(len(list(payloads)), 1) - def test_01_mega_raid_collector_installed_and_okay(self): + def test_mega_raid_collector_installed_and_okay(self): """Test mega raid collector can fetch correct number of metrics.""" mega_raid_collector = MegaRAIDCollector(Mock()) mega_raid_collector.storcli = Mock() @@ -100,7 +100,7 @@ def test_01_mega_raid_collector_installed_and_okay(self): for payload in payloads: self.assertIn(payload.name, available_metrics) - def test_10_lsi_sas_2_collector_not_installed(self): + def test_lsi_sas_2_collector_not_installed(self): """Test LSI SAS 2 collector when sas2ircu is not installed.""" lsi_sas_2_collector = LSISASControllerCollector(2, Mock()) lsi_sas_2_collector.sasircu = Mock() @@ -111,7 +111,7 @@ def test_10_lsi_sas_2_collector_not_installed(self): self.assertEqual(len(list(payloads)), 1) - def test_11_lsi_sas_2_collector_installed_and_okay(self): + def test_lsi_sas_2_collector_installed_and_okay(self): """Test LSI SAS 2 collector can fetch correct number of metrics.""" lsi_sas_2_collector = LSISASControllerCollector(2, Mock()) lsi_sas_2_collector.sasircu = Mock() @@ -194,7 +194,7 @@ def test_11_lsi_sas_2_collector_installed_and_okay(self): for payload in payloads: self.assertIn(payload.name, available_metrics) - def test_20_lsi_sas_3_collector_not_installed(self): + def test_lsi_sas_3_collector_not_installed(self): """Test LSI SAS 3 collector when sas3ircu is not installed.""" lsi_sas_3_collector = LSISASControllerCollector(3, Mock()) lsi_sas_3_collector.sasircu = Mock() @@ -205,7 +205,7 @@ def test_20_lsi_sas_3_collector_not_installed(self): self.assertEqual(len(list(payloads)), 1) - def test_21_lsi_sas_3_collector_installed_and_okay(self): + def test_lsi_sas_3_collector_installed_and_okay(self): """Test LSI SAS 3 collector can fetch correct number of metrics.""" lsi_sas_3_collector = LSISASControllerCollector(3, Mock()) lsi_sas_3_collector.sasircu = Mock() @@ -302,7 +302,7 @@ def test_30_ipmi_dcmi_collector_not_installed(self): self.assertEqual(len(list(payloads)), 1) - def test_31_ipmi_dcmi_collector_installed_and_okay(self): + def test_ipmi_dcmi_collector_installed_and_okay(self): """Test ipmi dcmi collector can fetch correct number of metrics.""" ipmi_dcmi_collector = IpmiDcmiCollector(Mock()) ipmi_dcmi_collector.ipmi_dcmi = Mock() @@ -322,7 +322,7 @@ def test_31_ipmi_dcmi_collector_installed_and_okay(self): for payload in payloads: self.assertIn(payload.name, available_metrics) - def test_32_ipmi_dcmi_collector_get_ps_redundancy_not_ok(self): + def test_ipmi_dcmi_collector_get_ps_redundancy_not_ok(self): """Test ipmi dcmi collector with ps_redundancy return is not ok.""" ipmi_dcmi_collector = IpmiDcmiCollector(Mock()) ipmi_dcmi_collector.ipmi_dcmi = Mock() @@ -342,7 +342,7 @@ def test_32_ipmi_dcmi_collector_get_ps_redundancy_not_ok(self): for payload in payloads: self.assertIn(payload.name, available_metrics) - def test_33_ipmi_dcmi_collector_power_consumption_percentage_valid(self): + def test_ipmi_dcmi_collector_power_consumption_percentage_valid(self): """Test ipmi dcmi collector can fetch correct number of metrics.""" ipmi_dcmi_collector = IpmiDcmiCollector(Mock()) ipmi_dcmi_collector.ipmi_dcmi = Mock() @@ -372,7 +372,7 @@ def test_33_ipmi_dcmi_collector_power_consumption_percentage_valid(self): self.assertEqual(payload.samples[0].value, expect) self.assertTrue(payload_exists) - def test_40_ipmi_sel_not_installed(self): + def test_ipmi_sel_not_installed(self): """Test ipmi sel collector when ipmi sel is not installed.""" ipmi_sel_collector = IpmiSelCollector(Mock()) ipmi_sel_collector.ipmi_sel = Mock() @@ -382,7 +382,7 @@ def test_40_ipmi_sel_not_installed(self): self.assertEqual(len(list(payloads)), 1) - def test_41_ipmi_sel_installed_and_okay(self): + def test_ipmi_sel_installed_and_okay(self): """Test ipmi sel collector can fetch correct number of metrics.""" ipmi_sel_collector = IpmiSelCollector(Mock()) ipmi_sel_collector.ipmi_sel = Mock() @@ -406,7 +406,7 @@ def test_41_ipmi_sel_installed_and_okay(self): self.assertDictEqual(payloads_labels_value_map, expected_payloads_label_value_map) - def test_42_ipmi_sel_cmd_fail(self): + def test_ipmi_sel_cmd_fail(self): """Test ipmi sel collector when ipmi sel is not installed.""" ipmi_sel_collector = IpmiSelCollector(Mock()) ipmi_sel_collector.ipmi_sel = Mock() @@ -417,7 +417,7 @@ def test_42_ipmi_sel_cmd_fail(self): assert payloads[0].name == "ipmi_sel_command_success" assert payloads[0].samples[0].value == 0.0 - def test_50_ipmimonitoring_not_installed(self): + def test_ipmimonitoring_not_installed(self): """Test ipmi sensor collector when ipmimonitoring is not installed.""" ipmi_sensors_collector = IpmiSensorsCollector(Mock()) ipmi_sensors_collector.ipmimonitoring = Mock() @@ -427,7 +427,7 @@ def test_50_ipmimonitoring_not_installed(self): self.assertEqual(len(list(payloads)), 1) - def test_51_ipmimonitoring_installed_and_okay(self): + def test_ipmimonitoring_installed_and_okay(self): """Test ipmi sensors collector can fetch correct number of metrics.""" ipmi_sensors_collector = IpmiSensorsCollector(Mock()) ipmi_sensors_collector.ipmimonitoring = Mock() @@ -442,7 +442,7 @@ def test_51_ipmimonitoring_installed_and_okay(self): for payload in payloads: self.assertIn(payload.name, available_metrics) - def test_60_ssacli_not_installed(self): + def test_ssacli_not_installed(self): ssacli_collector = SsaCLICollector(Mock()) ssacli_collector.ssacli = Mock() ssacli_collector.ssacli.installed = False @@ -451,7 +451,7 @@ def test_60_ssacli_not_installed(self): self.assertEqual(len(list(payloads)), 1) - def test_61_ssacli_installed_and_okay(self): + def test_ssacli_installed_and_okay(self): ssacli_collector = SsaCLICollector(Mock()) ssacli_collector.ssacli = Mock() mock_payload = { @@ -469,7 +469,7 @@ def test_61_ssacli_installed_and_okay(self): for payload in payloads: self.assertIn(payload.name, available_metrics) - def test_101_perccli_collector_command_success(self): + def test_perccli_collector_command_success(self): with patch.object(PowerEdgeRAIDCollector, "perccli") as mock_cli: # 1 success, 1 fail mock_cli.ctrl_exists.return_value = True @@ -491,7 +491,7 @@ def test_101_perccli_collector_command_success(self): assert payloads[2].samples[0].labels["controller_id"] == "1" assert payloads[2].samples[0].name == "perccli_command_ctrl_success" - def test_102_perccli_virtual_device_command_success(self): + def test_perccli_virtual_device_command_success(self): with patch.object(PowerEdgeRAIDCollector, "perccli") as mock_cli: mock_cli.success.return_value = True mock_cli.ctrl_successes.return_value = {0: False, 1: True} @@ -526,7 +526,7 @@ def test_102_perccli_virtual_device_command_success(self): ]: assert name in get_payloads - def test_103_perccli_cmd_fail(self): + def test_perccli_cmd_fail(self): with patch.object(PowerEdgeRAIDCollector, "perccli") as mock_cli: mock_cli.success.return_value = False power_edge_collector = PowerEdgeRAIDCollector(Mock()) @@ -534,7 +534,7 @@ def test_103_perccli_cmd_fail(self): assert len(payloads) == 1 assert payloads[0].samples[0].value == 0.0 - def test_104_perccli_no_controller_exists(self): + def test_perccli_no_controller_exists(self): with patch.object(PowerEdgeRAIDCollector, "perccli") as mock_cli: mock_cli.success.return_value = True mock_cli.ctrl_exists.return_value = False @@ -543,7 +543,7 @@ def test_104_perccli_no_controller_exists(self): assert len(payloads) == 2 assert payloads[1].samples[0].value == 0.0 - def test_105_perccli_physical_device_command_success(self): + def test_perccli_physical_device_command_success(self): with patch.object(PowerEdgeRAIDCollector, "perccli") as mock_cli: mock_cli.success.return_value = True mock_cli.ctrl_successes.return_value = {0: False, 1: True} @@ -613,7 +613,7 @@ def test_105_perccli_physical_device_command_success(self): @patch( "prometheus_hardware_exporter.collectors.redfish.RedfishHelper.get_cached_discover_method" ) - def test_200_redfish_no_discovered_services( + def test_redfish_no_discovered_services( self, mock_get_cached_discover_method, mock_redfish_client ): """Test redfish collector no redfish services have been discovered.""" @@ -639,7 +639,7 @@ def _discover(*args, **kwargs): @patch( "prometheus_hardware_exporter.collectors.redfish.RedfishHelper.get_cached_discover_method" ) - def test_201_redfish_no_login( + def test_redfish_no_login( self, mock_get_cached_discover_method, mock_redfish_client, mock_logger ): """Test redfish collector when redfish login doesn't work.""" @@ -679,7 +679,7 @@ def _discover(*args, **kwargs): @patch( "prometheus_hardware_exporter.collectors.redfish.RedfishHelper.get_cached_discover_method" ) - def test_202_redfish_installed_and_okay( + def test_redfish_installed_and_okay( self, mock_get_cached_discover_method, mock_redfish_client, @@ -865,7 +865,7 @@ def _discover(*args, **kwargs): "Failed to get %s via redfish", "network_adapter_count" ) - def test_203_redfish_create_sensor_metric_payload(self): + def test_redfish_create_sensor_metric_payload(self): mock_sensor_data = { "1": [ {"Sensor": "State", "Reading": "Enabled", "Health": "OK"}, @@ -918,7 +918,7 @@ def test_203_redfish_create_sensor_metric_payload(self): ], ) - def test_204_redfish_create_processor_metric_payload(self): + def test_redfish_create_processor_metric_payload(self): mock_processor_count = {"s1": 2, "s2": 1} mock_processor_data = { "s1": [ @@ -1002,7 +1002,7 @@ def test_204_redfish_create_processor_metric_payload(self): ], ) - def test_205_redfish_create_storage_controller_metric_payload(self): + def test_redfish_create_storage_controller_metric_payload(self): mock_storage_controller_count = {"s1": 2} mock_storage_controller_data = { "s1": [ @@ -1060,7 +1060,7 @@ def test_205_redfish_create_storage_controller_metric_payload(self): ], ) - def test_207_redfish_create_network_adapter_metric_payload(self): + def test_redfish_create_network_adapter_metric_payload(self): mock_network_adapter_count = {"c1": 2, "c2": 1} redfish_collector = RedfishCollector(Mock()) network_adapter_payload = redfish_collector._create_network_adapter_metric_payload( @@ -1084,7 +1084,7 @@ def test_207_redfish_create_network_adapter_metric_payload(self): ], ) - def test_206_redfish_create_chassis_metric_payload(self): + def test_redfish_create_chassis_metric_payload(self): mock_chassis_data = { "c1": { "chassis_type": "RackMount", @@ -1135,7 +1135,7 @@ def test_206_redfish_create_chassis_metric_payload(self): ], ) - def test_208_redfish_create_storage_drive_metric_payload(self): + def test_redfish_create_storage_drive_metric_payload(self): mock_storage_drive_count = {"s1": 3} mock_storage_drive_data = { "s1": [ @@ -1211,7 +1211,7 @@ def test_208_redfish_create_storage_drive_metric_payload(self): ], ) - def test_209_redfish_create_memory_dimm_metric_payload(self): + def test_redfish_create_memory_dimm_metric_payload(self): mock_memory_dimm_count = {"s1": 1, "s2": 1} mock_memory_dimm_data = { "s1": [ @@ -1273,7 +1273,7 @@ def test_209_redfish_create_memory_dimm_metric_payload(self): ], ) - def test_210_redfish_create_smart_storage_health_metric_payload(self): + def test_redfish_create_smart_storage_health_metric_payload(self): mock_smart_storage_health_data = {"c1": {"health": "OK"}} redfish_collector = RedfishCollector(Mock()) smart_storage_health_payload = ( @@ -1293,7 +1293,7 @@ def test_210_redfish_create_smart_storage_health_metric_payload(self): ], ) - def test_1000_collector_fetch_failed(self): + def test_collector_fetch_failed(self): for collector_cls, expected_name, expected_labels in [ ( MegaRAIDCollector, From 76ae3b75ffff4ba50d20816a9d22d32d86e5bd1d Mon Sep 17 00:00:00 2001 From: Ashley James Date: Fri, 1 Dec 2023 18:43:02 +0530 Subject: [PATCH 2/3] Fix failed metric names. Previously, the failed metric had "collector" twice in its name. Eg: "ipmidcmicollector_collector_failed" This change aims to remove the duplication. Previous example becomes "ipmidcmi_collector_failed" The label is also changed similarly to reflect the new name. Unit test was also modified to reflect this change. --- prometheus_hardware_exporter/core.py | 12 ++++++++---- tests/unit/test_collector.py | 25 +++++++++++++++++++------ 2 files changed, 27 insertions(+), 10 deletions(-) diff --git a/prometheus_hardware_exporter/core.py b/prometheus_hardware_exporter/core.py index ba9434c..b0b2d9a 100644 --- a/prometheus_hardware_exporter/core.py +++ b/prometheus_hardware_exporter/core.py @@ -96,14 +96,18 @@ def failed_metrics(self) -> Iterable[Metric]: Yields: metrics: the internal metrics """ - name = self.__class__.__name__ + # if `__class_.__name__` is "IpmiDcmiCollector" + # then `name` becomes "ipmidcmi" + suffix = "collector" + class_name = self.__class__.__name__.lower() + name = class_name[: -len(suffix)] if class_name.endswith(suffix) else class_name metric = GaugeMetricFamily( - name=f"{name.lower()}_collector_failed", - documentation=f"{name} Collector failed to fetch metrics", + name=f"{name}_collector_failed", + documentation=f"{name} collector failed to fetch metrics", labels=["collector"], ) metric.add_metric( - labels=[self.__class__.__name__], + labels=[name], value=1, ) yield metric diff --git a/tests/unit/test_collector.py b/tests/unit/test_collector.py index eb421f7..7f4a73b 100644 --- a/tests/unit/test_collector.py +++ b/tests/unit/test_collector.py @@ -1294,21 +1294,34 @@ def test_redfish_create_smart_storage_health_metric_payload(self): ) def test_collector_fetch_failed(self): + class NoCollectorSuffix(MegaRAIDCollector): + """Dummy class whose name doesn't end with "Collector". + + Inherits from an existing collector for required methods to test suffix removal. + """ + + pass + for collector_cls, expected_name, expected_labels in [ ( MegaRAIDCollector, - "megaraidcollector_collector_failed", - {"collector": "MegaRAIDCollector"}, + "megaraid_collector_failed", + {"collector": "megaraid"}, ), ( RedfishCollector, - "redfishcollector_collector_failed", - {"collector": "RedfishCollector"}, + "redfish_collector_failed", + {"collector": "redfish"}, ), ( IpmiSensorsCollector, - "ipmisensorscollector_collector_failed", - {"collector": "IpmiSensorsCollector"}, + "ipmisensors_collector_failed", + {"collector": "ipmisensors"}, + ), + ( + NoCollectorSuffix, + "nocollectorsuffix_collector_failed", + {"collector": "nocollectorsuffix"}, ), ]: collector = collector_cls(Mock()) From 54b043c7185c437ea05b60813809dcec0a742d61 Mon Sep 17 00:00:00 2001 From: Ashley James Date: Tue, 5 Dec 2023 10:49:48 +0530 Subject: [PATCH 3/3] Use `str.replace` to manipulate name. --- prometheus_hardware_exporter/core.py | 8 +++----- tests/unit/test_collector.py | 8 ++++---- 2 files changed, 7 insertions(+), 9 deletions(-) diff --git a/prometheus_hardware_exporter/core.py b/prometheus_hardware_exporter/core.py index b0b2d9a..9e8189e 100644 --- a/prometheus_hardware_exporter/core.py +++ b/prometheus_hardware_exporter/core.py @@ -96,11 +96,9 @@ def failed_metrics(self) -> Iterable[Metric]: Yields: metrics: the internal metrics """ - # if `__class_.__name__` is "IpmiDcmiCollector" - # then `name` becomes "ipmidcmi" - suffix = "collector" - class_name = self.__class__.__name__.lower() - name = class_name[: -len(suffix)] if class_name.endswith(suffix) else class_name + # NOTE(dashmage): remove "Collector" from class name to avoid duplication in metric name + # e.g. `ipmidcmicollector_collector_failed` becomes `ipmidcmi_collector_failed` + name = self.__class__.__name__.lower().replace("collector", "") metric = GaugeMetricFamily( name=f"{name}_collector_failed", documentation=f"{name} collector failed to fetch metrics", diff --git a/tests/unit/test_collector.py b/tests/unit/test_collector.py index 7f4a73b..5435fc7 100644 --- a/tests/unit/test_collector.py +++ b/tests/unit/test_collector.py @@ -1294,7 +1294,7 @@ def test_redfish_create_smart_storage_health_metric_payload(self): ) def test_collector_fetch_failed(self): - class NoCollectorSuffix(MegaRAIDCollector): + class NoSuffix(MegaRAIDCollector): """Dummy class whose name doesn't end with "Collector". Inherits from an existing collector for required methods to test suffix removal. @@ -1319,9 +1319,9 @@ class NoCollectorSuffix(MegaRAIDCollector): {"collector": "ipmisensors"}, ), ( - NoCollectorSuffix, - "nocollectorsuffix_collector_failed", - {"collector": "nocollectorsuffix"}, + NoSuffix, + "nosuffix_collector_failed", + {"collector": "nosuffix"}, ), ]: collector = collector_cls(Mock())