e789619b34
This patch brings some of the physical and virtual drive attributes as `custom_fields` to the disks inventory. The goal is to have this information present to ease disks maintenance when a drive becomes unavailable and its attributes can't be read anymore from the RAID controller. It also helps to standardize the extended disk attributes across the different manufacturers. As the disk physical identifers were not available under the correct format (hexadecimal format using the `xml` output as opposed as `X:Y:Z` format using the default `list` format), the command line parser has been refactored to read the `list` format, rather than `xml` one in the `omreport` raid controller parser. As the custom fields have to be created prior being able to register the disks extended attributes, this feature is only activated using the `--process-virtual-drives` command line parameter, or by setting `process_virtual_drives` to `true` in the configuration file. The custom fields to create as `DCIM > inventory item` `Text` are described below. NAME LABEL DESCRIPTION mount_point Mount point Device mount point(s) pd_identifier Physical disk identifier Physical disk identifier in the RAID controller vd_array Virtual drive array Virtual drive array the disk is member of vd_consistency Virtual drive consistency Virtual disk array consistency vd_device Virtual drive device Virtual drive system device vd_raid_type Virtual drive RAID Virtual drive array RAID type vd_size Virtual drive size Virtual drive array size In the current implementation, the disks attributes ore not updated: if a disk with the correct serial number is found, it's sufficient to consider it as up to date. To force the reprocessing of the disks extended attributes, the `--force-disk-refresh` command line option can be used: it removes all existing disks to before populating them with the correct parsing. Unless this option is specified, the extended attributes won't be modified unless a disk is replaced. It is possible to dump the physical/virtual disks map on the filesystem under the JSON notation to ease or automate disks management. The file path has to be provided using the `--dump-disks-map` command line parameter.
190 lines
5.9 KiB
Python
190 lines
5.9 KiB
Python
from netbox_agent.raid.base import Raid, RaidController
|
|
from netbox_agent.misc import get_vendor
|
|
from netbox_agent.config import config
|
|
import subprocess
|
|
import logging
|
|
import re
|
|
|
|
REGEXP_CONTROLLER_HP = re.compile(r'Smart Array ([a-zA-Z0-9- ]+) in Slot ([0-9]+)')
|
|
|
|
|
|
def ssacli(command):
|
|
output = subprocess.getoutput('ssacli {}'.format(command) )
|
|
lines = output.split('\n')
|
|
lines = list(filter(None, lines))
|
|
return lines
|
|
|
|
|
|
def _parse_ctrl_output(lines):
|
|
controllers = {}
|
|
current_ctrl = None
|
|
|
|
for line in lines:
|
|
if not line or line.startswith('Note:'):
|
|
continue
|
|
ctrl = REGEXP_CONTROLLER_HP.search(line)
|
|
if ctrl is not None:
|
|
current_ctrl = ctrl.group(1)
|
|
controllers[current_ctrl] = {'Slot': ctrl.group(2)}
|
|
if 'Embedded' not in line:
|
|
controllers[current_ctrl]['External'] = True
|
|
continue
|
|
attr, val = line.split(': ', 1)
|
|
attr = attr.strip()
|
|
val = val.strip()
|
|
controllers[current_ctrl][attr] = val
|
|
return controllers
|
|
|
|
|
|
def _parse_pd_output(lines):
|
|
drives = {}
|
|
current_array = None
|
|
current_drv = None
|
|
|
|
for line in lines:
|
|
line = line.strip()
|
|
if not line or line.startswith('Note:'):
|
|
continue
|
|
# Parses the Array the drives are in
|
|
if line.startswith('Array'):
|
|
current_array = line.split(None, 1)[1]
|
|
# Detects new physical drive
|
|
if line.startswith('physicaldrive'):
|
|
current_drv = line.split(None, 1)[1]
|
|
drives[current_drv] = {}
|
|
if current_array is not None:
|
|
drives[current_drv]['Array'] = current_array
|
|
continue
|
|
if ': ' not in line:
|
|
continue
|
|
attr, val = line.split(': ', 1)
|
|
drives.setdefault(current_drv, {})[attr] = val
|
|
return drives
|
|
|
|
|
|
def _parse_ld_output(lines):
|
|
drives = {}
|
|
current_array = None
|
|
current_drv = None
|
|
|
|
for line in lines:
|
|
line = line.strip()
|
|
if not line or line.startswith('Note:'):
|
|
continue
|
|
# Parses the Array the drives are in
|
|
if line.startswith('Array'):
|
|
current_array = line.split(None, 1)[1]
|
|
drives[current_array] = {}
|
|
# Detects new physical drive
|
|
if line.startswith('Logical Drive'):
|
|
current_drv = line.split(': ', 1)[1]
|
|
drives.setdefault(current_array, {})['LogicalDrive'] = current_drv
|
|
continue
|
|
if ': ' not in line:
|
|
continue
|
|
attr, val = line.split(': ', 1)
|
|
drives.setdefault(current_array, {})[attr] = val
|
|
return drives
|
|
|
|
|
|
class HPRaidController(RaidController):
|
|
def __init__(self, controller_name, data):
|
|
self.controller_name = controller_name
|
|
self.data = data
|
|
self.pdrives = self._get_physical_disks()
|
|
self.ldrives = self._get_logical_drives()
|
|
self._get_virtual_drives_map()
|
|
|
|
def get_product_name(self):
|
|
return self.controller_name
|
|
|
|
def get_manufacturer(self):
|
|
return 'HP'
|
|
|
|
def get_serial_number(self):
|
|
return self.data['Serial Number']
|
|
|
|
def get_firmware_version(self):
|
|
return self.data['Firmware Version']
|
|
|
|
def is_external(self):
|
|
return self.data.get('External', False)
|
|
|
|
def _get_physical_disks(self):
|
|
lines = ssacli('ctrl slot={} pd all show detail'.format(self.data['Slot']))
|
|
pdrives = _parse_pd_output(lines)
|
|
ret = {}
|
|
|
|
for name, attrs in pdrives.items():
|
|
array = attrs.get('Array', '')
|
|
model = attrs.get('Model', '').strip()
|
|
vendor = None
|
|
if model.startswith('HP'):
|
|
vendor = 'HP'
|
|
elif len(model.split()) > 1:
|
|
vendor = get_vendor(model.split()[1])
|
|
else:
|
|
vendor = get_vendor(model)
|
|
|
|
ret[name] = {
|
|
'Array': array,
|
|
'Model': model,
|
|
'Vendor': vendor,
|
|
'SN': attrs.get('Serial Number', '').strip(),
|
|
'Size': attrs.get('Size', '').strip(),
|
|
'Type': 'SSD' if attrs.get('Interface Type') == 'Solid State SATA'
|
|
else 'HDD',
|
|
'_src': self.__class__.__name__,
|
|
}
|
|
return ret
|
|
|
|
def _get_logical_drives(self):
|
|
lines = ssacli('ctrl slot={} ld all show detail'.format(self.data['Slot']))
|
|
ldrives = _parse_ld_output(lines)
|
|
ret = {}
|
|
|
|
for array, attrs in ldrives.items():
|
|
ret[array] = {
|
|
'vd_array': array,
|
|
'vd_size': attrs['Size'],
|
|
'vd_consistency': attrs['Status'],
|
|
'vd_raid_type': 'RAID {}'.format(attrs['Fault Tolerance']),
|
|
'vd_device': attrs['LogicalDrive'],
|
|
'mount_point': attrs['Mount Points']
|
|
}
|
|
return ret
|
|
|
|
def _get_virtual_drives_map(self):
|
|
for name, attrs in self.pdrives.items():
|
|
array = attrs["Array"]
|
|
ld = self.ldrives.get(array)
|
|
if ld is None:
|
|
logging.error(
|
|
"Failed to find array information for physical drive {}."
|
|
" Ignoring.".format(name)
|
|
)
|
|
continue
|
|
attrs['custom_fields'] = ld
|
|
attrs['custom_fields']['pd_identifier'] = name
|
|
|
|
def get_physical_disks(self):
|
|
return list(self.pdrives.values())
|
|
|
|
|
|
class HPRaid(Raid):
|
|
def __init__(self):
|
|
self.output = subprocess.getoutput('ssacli ctrl all show detail')
|
|
self.controllers = []
|
|
self.convert_to_dict()
|
|
|
|
def convert_to_dict(self):
|
|
lines = self.output.split('\n')
|
|
lines = list(filter(None, lines))
|
|
controllers = _parse_ctrl_output(lines)
|
|
for controller, attrs in controllers.items():
|
|
self.controllers.append(
|
|
HPRaidController(controller, attrs)
|
|
)
|
|
|
|
def get_controllers(self):
|
|
return self.controllers
|