Skip to content

Commit

Permalink
fix: support smart array
Browse files Browse the repository at this point in the history
Fetch hardware information from hwinfo and add support items

Fix: canonical#90
  • Loading branch information
jneo8 committed Feb 22, 2024
1 parent a7cee56 commit 0568730
Show file tree
Hide file tree
Showing 4 changed files with 227 additions and 14 deletions.
53 changes: 52 additions & 1 deletion src/hardware.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@
logger = logging.getLogger(__name__)


SUPPORTED_STORAGES = {
LSHW_SUPPORTED_STORAGES = {
HWTool.SAS2IRCU: [
# Broadcom
"SAS2004",
Expand All @@ -33,6 +33,17 @@
],
}

HWINFO_SUPPORTED_STORAGES = {
HWTool.SSACLI: [
[
"Hardware Class: storage",
'Vendor: pci 0x9005 "Adaptec"',
'Device: pci 0x028f "Smart Storage PQI 12G SAS/PCIe 3"',
'SubDevice: pci 0x1100 "Smart Array P816i-a SR Gen10"',
]
]
}


def lshw(class_filter: t.Optional[str] = None) -> t.Any:
"""Return lshw output as dict."""
Expand Down Expand Up @@ -66,3 +77,43 @@ def get_bmc_address() -> t.Optional[str]:
except subprocess.CalledProcessError:
logger.debug("IPMI is not available")
return None


def _split_at(s: str, c: str, n: int) -> t.Tuple[str, str]:
"""Split a string 's' at the 'n'th occurrence of delimiter 'c'.
Parameters:
s (str): The string to split.
c (str): Delimiter for splitting.
n (int): Occurrence of 'c' to split at.
Returns:
Tuple[str, str]: The string before and after the 'n'th occurrence of 'c'.
"""
words = s.split(c)
return c.join(words[:n]), c.join(words[n:])


def hwinfo(*args: str) -> t.Dict[str, str]:
"""Run hwinfo command and return output as dicturary.
Args:
args: Probe for a particular hardware class.
Returns:
hw_info: hardware information dicturary
"""
apt.add_package("hwinfo", update_cache=False)
hw_classes = list(args)
for idx, hw_item in enumerate(args):
hw_classes[idx] = "--" + hw_item
hw_classes.insert(0, "hwinfo")

output = subprocess.check_output(hw_classes, text=True)
if "start debug info" in output.splitlines()[0]:
output = _split_at(output, "=========== end debug info ============", 1)[1]

hardwares: t.Dict[str, str] = {}
for item in output.split("\n\n"):
key = item.splitlines()[0].strip()
hardwares[key] = item
return hardwares
43 changes: 33 additions & 10 deletions src/hw_tools.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@
from abc import ABCMeta, abstractmethod
from functools import lru_cache
from pathlib import Path
from typing import Dict, List, Tuple
from typing import Dict, List, Set, Tuple

from charms.operator_libs_linux.v0 import apt
from ops.model import ModelError, Resources
Expand Down Expand Up @@ -42,7 +42,13 @@
StorageVendor,
SystemVendor,
)
from hardware import SUPPORTED_STORAGES, get_bmc_address, lshw
from hardware import (
HWINFO_SUPPORTED_STORAGES,
LSHW_SUPPORTED_STORAGES,
get_bmc_address,
hwinfo,
lshw,
)
from keys import HP_KEYS

logger = logging.getLogger(__name__)
Expand Down Expand Up @@ -342,18 +348,31 @@ def check(self) -> bool:
return True


def _raid_hw_verifier_hwinfo() -> Set[HWTool]:
"""Verify if the HWTool support RAID card exists on machine with hwinfo."""
hwinfo_output = hwinfo("storage")

tools = set()
for _, hwinfo_content in hwinfo_output.items():
# ssacli
for support_storage in HWINFO_SUPPORTED_STORAGES[HWTool.SSACLI]:
if all(item in hwinfo_content for item in support_storage):
tools.add(HWTool.SSACLI)
return tools


# Using cache here to avoid repeat call.
# The lru_cache should be clean everytime the hook been triggered.
@lru_cache
def raid_hw_verifier() -> List[HWTool]:
"""Verify if the HWTool support RAID card exists on machine."""
hw_info = lshw()
system_vendor = hw_info.get("vendor")
storage_info = lshw(class_filter="storage")
lshw_output = lshw()
system_vendor = lshw_output.get("vendor")
lshw_storage = lshw(class_filter="storage")

tools = set()

for info in storage_info:
for info in lshw_storage:
_id = info.get("id")
product = info.get("product")
vendor = info.get("vendor")
Expand All @@ -363,7 +382,7 @@ def raid_hw_verifier() -> List[HWTool]:
if (
any(
_product
for _product in SUPPORTED_STORAGES[HWTool.SAS3IRCU]
for _product in LSHW_SUPPORTED_STORAGES[HWTool.SAS3IRCU]
if _product in product
)
and vendor == StorageVendor.BROADCOM
Expand All @@ -373,7 +392,7 @@ def raid_hw_verifier() -> List[HWTool]:
if (
any(
_product
for _product in SUPPORTED_STORAGES[HWTool.SAS2IRCU]
for _product in LSHW_SUPPORTED_STORAGES[HWTool.SAS2IRCU]
if _product in product
)
and vendor == StorageVendor.BROADCOM
Expand All @@ -383,7 +402,9 @@ def raid_hw_verifier() -> List[HWTool]:
if _id == "raid":
# ssacli
if system_vendor == SystemVendor.HP and any(
_product for _product in SUPPORTED_STORAGES[HWTool.SSACLI] if _product in product
_product
for _product in LSHW_SUPPORTED_STORAGES[HWTool.SSACLI]
if _product in product
):
tools.add(HWTool.SSACLI)
# perccli
Expand All @@ -392,7 +413,9 @@ def raid_hw_verifier() -> List[HWTool]:
# storcli
elif driver == "megaraid_sas" and vendor == StorageVendor.BROADCOM:
tools.add(HWTool.STORCLI)
return list(tools)

hwinfo_tools = _raid_hw_verifier_hwinfo()
return list(tools | hwinfo_tools)


# Using cache here to avoid repeat call.
Expand Down
100 changes: 98 additions & 2 deletions tests/unit/test_hardware.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,108 @@
import unittest
from unittest import mock

from hardware import get_bmc_address, lshw
import pytest

from hardware import _split_at, get_bmc_address, hwinfo, lshw


@pytest.mark.parametrize(
"s,delimiter,num,expect",
[("abc-d1-d1-d1-def", "d1", 2, ("abc-d1-", "-d1-def"))],
)
def test_split_at(s, delimiter, num, expect):
output = _split_at(s, delimiter, num)
case = unittest.TestCase()
case.assertCountEqual(output, expect)


class TestHwinfo:
@pytest.mark.parametrize(
"hw_classes,expect_cmd, hwinfo_output,expect",
[
(
[],
["hwinfo"],
(
""
"============ start debug info ============"
"random-string"
"random-string"
"random-string"
"random-string"
"=========== end debug info ============"
"10: key-a\n"
" [Created at pci.386]\n"
" Unique ID: unique-id-a\n"
" Parent ID: parent-id-a\n"
"\n"
"11: key-b\n"
" [Created at pci.386]\n"
" Unique ID: unique-id-b\n"
" Parent ID: parent-id-b\n"
),
{
"10: key-a": (
"10: key-a\n"
" [Created at pci.386]\n"
" Unique ID: unique-id-a\n"
" Parent ID: parent-id-a"
),
"11: key-b": (
"11: key-b\n"
" [Created at pci.386]\n"
" Unique ID: unique-id-b\n"
" Parent ID: parent-id-b\n"
),
},
),
(
["storage"],
["hwinfo", "--storage"],
(
""
"10: key-a\n"
" [Created at pci.386]\n"
" Unique ID: unique-id-a\n"
" Parent ID: parent-id-a\n"
"\n"
"11: key-b\n"
" [Created at pci.386]\n"
" Unique ID: unique-id-b\n"
" Parent ID: parent-id-b\n"
),
{
"10: key-a": (
"10: key-a\n"
" [Created at pci.386]\n"
" Unique ID: unique-id-a\n"
" Parent ID: parent-id-a"
),
"11: key-b": (
"11: key-b\n"
" [Created at pci.386]\n"
" Unique ID: unique-id-b\n"
" Parent ID: parent-id-b\n"
),
},
),
],
)
@mock.patch("hardware.apt")
@mock.patch("hardware.subprocess.check_output")
def test_hwinfo_output(
self, mock_subprocess, mock_apt, hw_classes, expect_cmd, hwinfo_output, expect
):
mock_subprocess.return_value = hwinfo_output
output = hwinfo(*hw_classes)
mock_subprocess.assert_called_with(expect_cmd, text=True)
assert output == expect


class TestLshw(unittest.TestCase):
@mock.patch("hardware.apt")
@mock.patch("hardware.subprocess.check_output")
def test_lshw_list_output(self, mock_subprocess):
def test_lshw_list_output(self, mock_subprocess, mock_apt):
mock_subprocess.return_value = """[{"expected_output": 1}]"""
for class_filter in [None, "storage"]:
output = lshw(class_filter)
Expand Down
45 changes: 44 additions & 1 deletion tests/unit/test_hw_tools.py
Original file line number Diff line number Diff line change
Expand Up @@ -764,8 +764,51 @@ def test_get_hw_tool_white_list(mock_raid_verifier, mock_bmc_hw_verifier):
],
)
@mock.patch("hw_tools.lshw")
def test_raid_hw_verifier(mock_lshw, lshw_output, lshw_storage_output, expect):
@mock.patch("hw_tools.hwinfo")
def test_raid_hw_verifier_lshw(mock_hwinfo, mock_lshw, lshw_output, lshw_storage_output, expect):
mock_lshw.side_effect = [lshw_output, lshw_storage_output]
mock_hwinfo.return_value = {}
raid_hw_verifier.cache_clear()
output = raid_hw_verifier()
case = unittest.TestCase()
case.assertCountEqual(output, expect)


@pytest.mark.parametrize(
"hwinfo_output, expect",
[
({}, []),
(
{
"random-key-a": """
[Created at pci.386]
Hardware Class: storage
Vendor: pci 0x9005 "Adaptec"
Device: pci 0x028f "Smart Storage PQI 12G SAS/PCIe 3"
SubDevice: pci 0x1100 "Smart Array P816i-a SR Gen10"
"""
},
[HWTool.SSACLI],
),
(
{
"random-key-a": """
[Created at pci.386]
Hardware Class: not-valid-class
Vendor: pci 0x9005 "Adaptec"
Device: pci 0x028f "Smart Storage PQI 12G SAS/PCIe 3"
SubDevice: pci 0x1100 "Smart Array P816i-a SR Gen10"
"""
},
[],
),
],
)
@mock.patch("hw_tools.lshw")
@mock.patch("hw_tools.hwinfo")
def test_raid_hw_verifier_hwinfo(mock_hwinfo, mock_lshw, hwinfo_output, expect):
mock_lshw.side_effect = [{"vendor": "somevendor"}, []]
mock_hwinfo.return_value = hwinfo_output
raid_hw_verifier.cache_clear()
output = raid_hw_verifier()
case = unittest.TestCase()
Expand Down

0 comments on commit 0568730

Please sign in to comment.