support netbox >=2.9 #177

Merged
Solvik merged 22 commits from 176-netbox-2.9 into master 2022-03-07 16:03:05 +01:00
12 changed files with 505 additions and 223 deletions
Showing only changes of commit e789619b34 - Show all commits

View file

@ -92,6 +92,8 @@ netbox:
token: supersecrettoken token: supersecrettoken
# uncomment to disable ssl verification # uncomment to disable ssl verification
# ssl_verify: false # ssl_verify: false
# uncomment to use the system's CA certificates
# ssl_ca_certs_file: /etc/ssl/certs/ca-certificates.crt
# Network configuration # Network configuration
network: network:
@ -160,6 +162,36 @@ The `get_blade_slot` method return the name of the `Device Bay`.
Certain vendors don't report the blade slot in `dmidecode`, so we can use the `slot_location` regex feature of the configuration file. Certain vendors don't report the blade slot in `dmidecode`, so we can use the `slot_location` regex feature of the configuration file.
Some blade servers can be equipped with additional hardware using expansion blades, next to the processing blade, such as GPU expansion, or drives bay expansion. By default, the hardware from the expnasion is associated with the blade server itself, but it's possible to register the expansion as its own device using the `--expansion-as-device` command line parameter, or by setting `expansion_as_device` to `true` in the configuration file.
## Drives attributes processing
It is possible to process drives extended attributes such as the drive's physical or logical identifier, logical drive RAID type, size, consistency and so on.
Those attributes as set as `custom_fields` in Netbox, and need to be registered properly before being able to specify them during the inventory phase.
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.
## Anycast IP ## Anycast IP
The default behavior of the agent is to assign an interface to an IP. The default behavior of the agent is to assign an interface to an IP.

View file

@ -78,6 +78,13 @@ def get_config():
p.add_argument('--network.lldp', help='Enable auto-cabling feature through LLDP infos') p.add_argument('--network.lldp', help='Enable auto-cabling feature through LLDP infos')
p.add_argument('--inventory', action='store_true', p.add_argument('--inventory', action='store_true',
help='Enable HW inventory (CPU, Memory, RAID Cards, Disks) feature') help='Enable HW inventory (CPU, Memory, RAID Cards, Disks) feature')
p.add_argument('--process-virtual-drives', action='store_true',
help='Process virtual drives information from RAID '
'controllers to fill disk custom_fields')
p.add_argument('--force-disk-refresh', action='store_true',
help='Forces disks detection reprocessing')
p.add_argument('--dump-disks-map',
help='File path to dump physical/virtual disks map')
options = p.parse_args() options = p.parse_args()
return options return options

View file

@ -1,8 +1,3 @@
import logging
import re
import pynetbox
from netbox_agent.config import config from netbox_agent.config import config
from netbox_agent.config import netbox_instance as nb from netbox_agent.config import netbox_instance as nb
from netbox_agent.lshw import LSHW from netbox_agent.lshw import LSHW
@ -10,6 +5,12 @@ from netbox_agent.misc import get_vendor, is_tool
from netbox_agent.raid.hp import HPRaid from netbox_agent.raid.hp import HPRaid
from netbox_agent.raid.omreport import OmreportRaid from netbox_agent.raid.omreport import OmreportRaid
from netbox_agent.raid.storcli import StorcliRaid from netbox_agent.raid.storcli import StorcliRaid
import traceback
import pynetbox
import logging
import json
import re
INVENTORY_TAG = { INVENTORY_TAG = {
'cpu': {'name': 'hw:cpu', 'slug': 'hw-cpu'}, 'cpu': {'name': 'hw:cpu', 'slug': 'hw-cpu'},
@ -226,7 +227,7 @@ class Inventory():
def get_raid_cards(self, filter_cards=False): def get_raid_cards(self, filter_cards=False):
raid_class = None raid_class = None
if self.server.manufacturer == 'Dell': if self.server.manufacturer in ('Dell', 'Huawei'):
if is_tool('omreport'): if is_tool('omreport'):
raid_class = OmreportRaid raid_class = OmreportRaid
if is_tool('storcli'): if is_tool('storcli'):
@ -302,53 +303,60 @@ class Inventory():
if raid_card.get_serial_number() not in [x.serial for x in nb_raid_cards]: if raid_card.get_serial_number() not in [x.serial for x in nb_raid_cards]:
self.create_netbox_raid_card(raid_card) self.create_netbox_raid_card(raid_card)
def is_virtual_disk(self, disk): def is_virtual_disk(self, disk, raid_devices):
disk_type = disk.get('type')
logicalname = disk.get('logicalname') logicalname = disk.get('logicalname')
description = disk.get('description') description = disk.get('description')
size = disk.get('size') size = disk.get('size')
product = disk.get('product') product = disk.get('product')
if logicalname in raid_devices or disk_type is None:
return True
non_raid_disks = [ non_raid_disks = [
'MR9361-8i', 'MR9361-8i',
] ]
if size is None and logicalname is None or \ if logicalname in raid_devices or \
'virtual' in product.lower() or 'logical' in product.lower() or \ disk_type is None or \
product in non_raid_disks or \ product in non_raid_disks or \
'virtual' in product.lower() or \
'logical' in product.lower() or \
'volume' in description.lower() or \
description == 'SCSI Enclosure' or \ description == 'SCSI Enclosure' or \
'volume' in description.lower(): (size is None and logicalname is None):
return True return True
return False return False
def get_hw_disks(self): def get_hw_disks(self):
disks = [] disks = []
for raid_card in self.get_raid_cards(filter_cards=True):
disks.extend(raid_card.get_physical_disks())
raid_devices = [
d.get('custom_fields', {}).get('vd_device')
for d in disks
if d.get('custom_fields', {}).get('vd_device')
]
for disk in self.lshw.get_hw_linux("storage"): for disk in self.lshw.get_hw_linux("storage"):
if self.is_virtual_disk(disk): if self.is_virtual_disk(disk, raid_devices):
continue continue
size =int(disk.get('size', 0)) / 1073741824
logicalname = disk.get('logicalname') d = {
description = disk.get('description') "name": "",
size = disk.get('size', 0) 'Size': '{} GB'.format(size),
product = disk.get('product') 'logicalname': disk.get('logicalname'),
serial = disk.get('serial') 'description': disk.get('description'),
'SN': disk.get('serial'),
d = {} 'Model': disk.get('product'),
d["name"] = "" 'Type': disk.get('type'),
d['Size'] = '{} GB'.format(int(size / 1024 / 1024 / 1024)) }
d['logicalname'] = logicalname
d['description'] = description
d['SN'] = serial
d['Model'] = product
if disk.get('vendor'): if disk.get('vendor'):
d['Vendor'] = disk['vendor'] d['Vendor'] = disk['vendor']
else: else:
d['Vendor'] = get_vendor(disk['product']) d['Vendor'] = get_vendor(disk['product'])
disks.append(d) disks.append(d)
for raid_card in self.get_raid_cards(filter_cards=True):
disks += raid_card.get_physical_disks()
# remove duplicate serials # remove duplicate serials
seen = set() seen = set()
uniq = [x for x in disks if x['SN'] not in seen and not seen.add(x['SN'])] uniq = [x for x in disks if x['SN'] not in seen and not seen.add(x['SN'])]
@ -361,53 +369,79 @@ class Inventory():
logicalname = disk.get('logicalname') logicalname = disk.get('logicalname')
desc = disk.get('description') desc = disk.get('description')
# nonraid disk name = '{} ({})'.format(disk['Model'], disk['Size'])
if logicalname and desc: description = disk['Type']
if type(logicalname) is list:
logicalname = logicalname[0]
name = '{} - {} ({})'.format(
desc,
logicalname,
disk.get('Size', 0))
description = 'Device {}'.format(disk.get('logicalname', 'Unknown'))
else:
name = '{} ({})'.format(disk['Model'], disk['Size'])
description = '{}'.format(disk['Type'])
_ = nb.dcim.inventory_items.create( parms = {
device=self.device_id, 'device': self.device_id,
discovered=True, 'discovered': True,
tags=[{'name': INVENTORY_TAG['disk']['name']}], 'tags': [{'name': INVENTORY_TAG['disk']['name']}],
name=name, 'name': name,
serial=disk['SN'], 'serial': disk['SN'],
part_id=disk['Model'], 'part_id': disk['Model'],
description=description, 'description': description,
manufacturer=manufacturer.id if manufacturer else None 'manufacturer': getattr(manufacturer, "id", None),
) }
if config.process_virtual_drives:
parms['custom_fields'] = disk.get("custom_fields", {})
_ = nb.dcim.inventory_items.create(**parms)
logging.info('Creating Disk {model} {serial}'.format( logging.info('Creating Disk {model} {serial}'.format(
model=disk['Model'], model=disk['Model'],
serial=disk['SN'], serial=disk['SN'],
)) ))
def dump_disks_map(self, disks):
disk_map = [d['custom_fields'] for d in disks if 'custom_fields' in d]
if config.dump_disks_map == "-":
f = sys.stdout
else:
f = open(config.dump_disks_map, "w")
f.write(
json.dumps(
disk_map,
separators=(',', ':'),
indent=4,
sort_keys=True
)
)
if config.dump_disks_map != "-":
f.close()
def do_netbox_disks(self): def do_netbox_disks(self):
nb_disks = self.get_netbox_inventory( nb_disks = self.get_netbox_inventory(
device_id=self.device_id, device_id=self.device_id,
tag=INVENTORY_TAG['disk']['slug']) tag=INVENTORY_TAG['disk']['slug']
)
disks = self.get_hw_disks() disks = self.get_hw_disks()
if config.dump_disks_map:
try:
self.dump_disks_map(disks)
except Exception as e:
logging.error("Failed to dump disks map: {}".format(e))
logging.debug(traceback.format_exc())
disk_serials = [d['SN'] for d in disks if 'SN' in d]
# delete disks that are in netbox but not locally # delete disks that are in netbox but not locally
# use the serial_number has the comparison element # use the serial_number has the comparison element
for nb_disk in nb_disks: for nb_disk in nb_disks:
if nb_disk.serial not in [x['SN'] for x in disks if x.get('SN')]: if nb_disk.serial not in disk_serials or \
config.force_disk_refresh:
logging.info('Deleting unknown locally Disk {serial}'.format( logging.info('Deleting unknown locally Disk {serial}'.format(
serial=nb_disk.serial, serial=nb_disk.serial,
)) ))
nb_disk.delete() nb_disk.delete()
if config.force_disk_refresh:
nb_disks = self.get_netbox_inventory(
device_id=self.device_id,
tag=INVENTORY_TAG['disk']['slug']
)
# create disks that are not in netbox # create disks that are not in netbox
for disk in disks: for disk in disks:
if disk.get('SN') not in [x.serial for x in nb_disks]: if disk.get('SN') not in [d.serial for d in nb_disks]:
self.create_netbox_disk(disk) self.create_netbox_disk(disk)
def create_netbox_memory(self, memory): def create_netbox_memory(self, memory):

View file

@ -1,9 +1,8 @@
import json
import logging
import subprocess
import sys
from netbox_agent.misc import is_tool from netbox_agent.misc import is_tool
import subprocess
import logging
import json
import sys
class LSHW(): class LSHW():
@ -15,7 +14,13 @@ class LSHW():
data = subprocess.getoutput( data = subprocess.getoutput(
'lshw -quiet -json' 'lshw -quiet -json'
) )
self.hw_info = json.loads(data) json_data = json.loads(data)
# Starting from version 02.18, `lshw -json` wraps its result in a list
# rather than returning directly a dictionary
if isinstance(json_data, list):
self.hw_info = json_data[0]
else:
self.hw_info = json_data
self.info = {} self.info = {}
self.memories = [] self.memories = []
self.interfaces = [] self.interfaces = []
@ -77,42 +82,41 @@ class LSHW():
def find_storage(self, obj): def find_storage(self, obj):
if "children" in obj: if "children" in obj:
for device in obj["children"]: for device in obj["children"]:
d = {} self.disks.append({
d["logicalname"] = device.get("logicalname") "logicalname": device.get("logicalname"),
d["product"] = device.get("product") "product": device.get("product"),
d["serial"] = device.get("serial") "serial": device.get("serial"),
d["version"] = device.get("version") "version": device.get("version"),
d["size"] = device.get("size") "size": device.get("size"),
d["description"] = device.get("description") "description": device.get("description"),
"type": device.get("description"),
self.disks.append(d) })
elif "nvme" in obj["configuration"]["driver"]: elif "nvme" in obj["configuration"]["driver"]:
if not is_tool('nvme'): if not is_tool('nvme'):
logging.error('nvme-cli >= 1.0 does not seem to be installed') logging.error('nvme-cli >= 1.0 does not seem to be installed')
else: return
try: try:
nvme = json.loads( nvme = json.loads(
subprocess.check_output( subprocess.check_output(
["nvme", '-list', '-o', 'json'], ["nvme", '-list', '-o', 'json'],
encoding='utf8') encoding='utf8')
) )
for device in nvme["Devices"]:
for device in nvme["Devices"]: d = {
d = {} 'logicalname': device["DevicePath"],
d['logicalname'] = device["DevicePath"] 'product': device["ModelNumber"],
d['product'] = device["ModelNumber"] 'serial': device["SerialNumber"],
d['serial'] = device["SerialNumber"] "version": device["Firmware"],
d["version"] = device["Firmware"] 'description': "NVME",
if "UsedSize" in device: 'type': "NVME",
d['size'] = device["UsedSize"] }
if "UsedBytes" in device: if "UsedSize" in device:
d['size'] = device["UsedBytes"] d['size'] = device["UsedSize"]
d['description'] = "NVME Disk" if "UsedBytes" in device:
d['size'] = device["UsedBytes"]
self.disks.append(d) self.disks.append(d)
except Exception: except Exception:
pass pass
def find_cpus(self, obj): def find_cpus(self, obj):
if "product" in obj: if "product" in obj:

View file

@ -1,10 +1,9 @@
import socket
import subprocess
from shutil import which
from slugify import slugify
from netbox_agent.config import netbox_instance as nb from netbox_agent.config import netbox_instance as nb
from slugify import slugify
from shutil import which
import subprocess
import socket
import re
def is_tool(name): def is_tool(name):
@ -74,3 +73,19 @@ def create_netbox_tags(tags):
) )
ret.append(nb_tag) ret.append(nb_tag)
return ret return ret
def get_mount_points():
mount_points = {}
output = subprocess.getoutput('mount')
for r in output.split("\n"):
if not r.startswith("/dev/"):
continue
mount_info = r.split()
device = mount_info[0]
device = re.sub(r'\d+$', '', device)
mp = mount_info[2]
mount_points.setdefault(device, []).append(mp)
return mount_points

View file

@ -325,7 +325,7 @@ class Network(object):
netbox_ips = nb.ipam.ip_addresses.filter( netbox_ips = nb.ipam.ip_addresses.filter(
address=ip, address=ip,
) )
if not len(netbox_ips): if not netbox_ips:
logging.info('Create new IP {ip} on {interface}'.format( logging.info('Create new IP {ip} on {interface}'.format(
ip=ip, interface=interface)) ip=ip, interface=interface))
query_params = { query_params = {
@ -340,7 +340,7 @@ class Network(object):
) )
return netbox_ip return netbox_ip
netbox_ip = next(netbox_ips) netbox_ip = list(netbox_ips)[0]
# If IP exists in anycast # If IP exists in anycast
if netbox_ip.role and netbox_ip.role.label == 'Anycast': if netbox_ip.role and netbox_ip.role.label == 'Anycast':
logging.debug('IP {} is Anycast..'.format(ip)) logging.debug('IP {} is Anycast..'.format(ip))

View file

@ -115,7 +115,7 @@ class PowerSupply():
voltage = [p['voltage'] for p in pwr_feeds] voltage = [p['voltage'] for p in pwr_feeds]
else: else:
logging.info('Could not find power feeds for Rack, defaulting value to 230') logging.info('Could not find power feeds for Rack, defaulting value to 230')
voltage = [230 for _ in nb_psu] voltage = [230 for _ in nb_psus]
for i, nb_psu in enumerate(nb_psus): for i, nb_psu in enumerate(nb_psus):
nb_psu.allocated_draw = int(float(psu_cons[i]) * voltage[i]) nb_psu.allocated_draw = int(float(psu_cons[i]) * voltage[i])

View file

@ -1,13 +1,20 @@
import re
import subprocess
from netbox_agent.config import config
from netbox_agent.misc import get_vendor
from netbox_agent.raid.base import Raid, RaidController 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]+)') 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): def _parse_ctrl_output(lines):
controllers = {} controllers = {}
current_ctrl = None current_ctrl = None
@ -18,11 +25,11 @@ def _parse_ctrl_output(lines):
ctrl = REGEXP_CONTROLLER_HP.search(line) ctrl = REGEXP_CONTROLLER_HP.search(line)
if ctrl is not None: if ctrl is not None:
current_ctrl = ctrl.group(1) current_ctrl = ctrl.group(1)
controllers[current_ctrl] = {"Slot": ctrl.group(2)} controllers[current_ctrl] = {'Slot': ctrl.group(2)}
if "Embedded" not in line: if 'Embedded' not in line:
controllers[current_ctrl]["External"] = True controllers[current_ctrl]['External'] = True
continue continue
attr, val = line.split(": ", 1) attr, val = line.split(': ', 1)
attr = attr.strip() attr = attr.strip()
val = val.strip() val = val.strip()
controllers[current_ctrl][attr] = val controllers[current_ctrl][attr] = val
@ -39,27 +46,54 @@ def _parse_pd_output(lines):
if not line or line.startswith('Note:'): if not line or line.startswith('Note:'):
continue continue
# Parses the Array the drives are in # Parses the Array the drives are in
if line.startswith("Array"): if line.startswith('Array'):
current_array = line.split(None, 1)[1] current_array = line.split(None, 1)[1]
# Detects new physical drive # Detects new physical drive
if line.startswith("physicaldrive"): if line.startswith('physicaldrive'):
current_drv = line.split(None, 1)[1] current_drv = line.split(None, 1)[1]
drives[current_drv] = {} drives[current_drv] = {}
if current_array is not None: if current_array is not None:
drives[current_drv]["Array"] = current_array drives[current_drv]['Array'] = current_array
continue continue
if ": " not in line: if ': ' not in line:
continue continue
attr, val = line.split(": ", 1) attr, val = line.split(': ', 1)
drives.setdefault(current_drv, {})[attr] = val drives.setdefault(current_drv, {})[attr] = val
return drives 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): class HPRaidController(RaidController):
def __init__(self, controller_name, data): def __init__(self, controller_name, data):
self.controller_name = controller_name self.controller_name = controller_name
self.data = data self.data = data
self.drives = self._get_physical_disks() self.pdrives = self._get_physical_disks()
self.ldrives = self._get_logical_drives()
self._get_virtual_drives_map()
def get_product_name(self): def get_product_name(self):
return self.controller_name return self.controller_name
@ -77,15 +111,12 @@ class HPRaidController(RaidController):
return self.data.get('External', False) return self.data.get('External', False)
def _get_physical_disks(self): def _get_physical_disks(self):
output = subprocess.getoutput( lines = ssacli('ctrl slot={} pd all show detail'.format(self.data['Slot']))
'ssacli ctrl slot={slot} pd all show detail'.format(slot=self.data['Slot']) pdrives = _parse_pd_output(lines)
) ret = {}
lines = output.split('\n')
lines = list(filter(None, lines))
drives = _parse_pd_output(lines)
ret = []
for name, attrs in drives.items(): for name, attrs in pdrives.items():
array = attrs.get('Array', '')
model = attrs.get('Model', '').strip() model = attrs.get('Model', '').strip()
vendor = None vendor = None
if model.startswith('HP'): if model.startswith('HP'):
@ -95,7 +126,8 @@ class HPRaidController(RaidController):
else: else:
vendor = get_vendor(model) vendor = get_vendor(model)
ret.append({ ret[name] = {
'Array': array,
'Model': model, 'Model': model,
'Vendor': vendor, 'Vendor': vendor,
'SN': attrs.get('Serial Number', '').strip(), 'SN': attrs.get('Serial Number', '').strip(),
@ -103,11 +135,40 @@ class HPRaidController(RaidController):
'Type': 'SSD' if attrs.get('Interface Type') == 'Solid State SATA' 'Type': 'SSD' if attrs.get('Interface Type') == 'Solid State SATA'
else 'HDD', else 'HDD',
'_src': self.__class__.__name__, '_src': self.__class__.__name__,
}) }
return ret 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): def get_physical_disks(self):
return self.drives return list(self.pdrives.values())
class HPRaid(Raid): class HPRaid(Raid):

View file

@ -1,25 +1,32 @@
import re
import subprocess
import xml.etree.ElementTree as ET # NOQA
from netbox_agent.misc import get_vendor
from netbox_agent.raid.base import Raid, RaidController from netbox_agent.raid.base import Raid, RaidController
from netbox_agent.misc import get_vendor, get_mount_points
# Inspiration from https://github.com/asciiphil/perc-status/blob/master/perc-status from netbox_agent.config import config
import subprocess
import logging
import re
def get_field(obj, fieldname): def omreport(sub_command):
f = obj.find(fieldname) command = 'omreport {}'.format(sub_command)
if f is None: output = subprocess.getoutput(command)
return None res = {}
if f.attrib['type'] in ['u32', 'u64']: section_re = re.compile('^[A-Z]')
if re.search('Mask$', fieldname): current_section = None
return int(f.text, 2) current_obj = None
else:
return int(f.text) for line in output.split('\n'):
if f.attrib['type'] == 'astring': if ': ' in line:
return f.text attr, value = line.split(': ', 1)
return f.text attr = attr.strip()
value = value.strip()
if attr == 'ID':
obj = {}
res.setdefault(current_section, []).append(obj)
current_obj = obj
current_obj[attr] = value
elif section_re.search(line) is not None:
current_section = line.strip()
return res
class OmreportController(RaidController): class OmreportController(RaidController):
@ -28,49 +35,88 @@ class OmreportController(RaidController):
self.controller_index = controller_index self.controller_index = controller_index
def get_product_name(self): def get_product_name(self):
return get_field(self.data, 'Name') return self.data['Name']
def get_manufacturer(self): def get_manufacturer(self):
return None return None
def get_serial_number(self): def get_serial_number(self):
return get_field(self.data, 'DeviceSerialNumber') return self.data.get('DeviceSerialNumber')
def get_firmware_version(self): def get_firmware_version(self):
return get_field(self.data, 'Firmware Version') return self.data.get('Firmware Version')
def _get_physical_disks(self):
pds = {}
res = omreport('storage pdisk controller={}'.format(
self.controller_index
))
for pdisk in [d for d in list(res.values())[0]]:
disk_id = pdisk['ID']
size = re.sub('B .*$', 'B', pdisk['Capacity'])
pds[disk_id] = {
'Vendor': get_vendor(pdisk['Vendor ID']),
'Model': pdisk['Product ID'],
'SN': pdisk['Serial No.'],
'Size': size,
'Type': pdisk['Media'],
'_src': self.__class__.__name__,
}
return pds
def _get_virtual_drives_map(self):
pds = {}
res = omreport('storage vdisk controller={}'.format(
self.controller_index
))
for vdisk in [d for d in list(res.values())[0]]:
vdisk_id = vdisk['ID']
device = vdisk['Device Name']
mount_points = get_mount_points()
mp = mount_points.get(device, 'n/a')
size = re.sub('B .*$', 'B', vdisk['Size'])
vd = {
'vd_array': vdisk_id,
'vd_size': size,
'vd_consistency': vdisk['State'],
'vd_raid_type': vdisk['Layout'],
'vd_device': vdisk['Device Name'],
'mount_point': ', '.join(sorted(mp)),
}
drives_res = omreport(
'storage pdisk controller={} vdisk={}'.format(
self.controller_index, vdisk_id
))
for pdisk in [d for d in list(drives_res.values())[0]]:
pds[pdisk['ID']] = vd
return pds
def get_physical_disks(self): def get_physical_disks(self):
ret = [] pds = self._get_physical_disks()
output = subprocess.getoutput( vds = self._get_virtual_drives_map()
'omreport storage controller controller={} -fmt xml'.format(self.controller_index) for pd_identifier, vd in vds.items():
) if pd_identifier not in pds:
root = ET.fromstring(output) logging.error(
et_array_disks = root.find('ArrayDisks') 'Physical drive {} listed in virtual drive {} not '
if et_array_disks is not None: 'found in drives list'.format(
for obj in et_array_disks.findall('DCStorageObject'): pd_identifier, vd['vd_array']
ret.append({ )
'Vendor': get_vendor(get_field(obj, 'Vendor')), )
'Model': get_field(obj, 'ProductID'), continue
'SN': get_field(obj, 'DeviceSerialNumber'), pds[pd_identifier].setdefault('custom_fields', {}).update(vd)
'Size': '{:.0f}GB'.format( pds[pd_identifier]['custom_fields']['pd_identifier'] = pd_identifier
int(get_field(obj, 'Length')) / 1024 / 1024 / 1024 return list(pds.values())
),
'Type': 'HDD' if int(get_field(obj, 'MediaType')) == 1 else 'SSD',
'_src': self.__class__.__name__,
})
return ret
class OmreportRaid(Raid): class OmreportRaid(Raid):
def __init__(self): def __init__(self):
output = subprocess.getoutput('omreport storage controller -fmt xml')
controller_xml = ET.fromstring(output)
self.controllers = [] self.controllers = []
res = omreport('storage controller')
for obj in controller_xml.find('Controllers').findall('DCStorageObject'): for controller in res['Controller']:
ctrl_index = get_field(obj, 'ControllerNum') ctrl_index = controller['ID']
self.controllers.append( self.controllers.append(
OmreportController(ctrl_index, obj) OmreportController(ctrl_index, controller)
) )
def get_controllers(self): def get_controllers(self):

View file

@ -1,8 +1,31 @@
import json
import subprocess
from netbox_agent.misc import get_vendor
from netbox_agent.raid.base import Raid, RaidController from netbox_agent.raid.base import Raid, RaidController
from netbox_agent.misc import get_vendor, get_mount_points
from netbox_agent.config import config
import subprocess
import logging
import json
import re
import os
def storecli(sub_command):
command = 'storcli {} J'.format(sub_command)
output = subprocess.getoutput(command)
data = json.loads(output)
controllers = dict([
(
c['Command Status']['Controller'],
c['Response Data']
) for c in data['Controllers']
if c['Command Status']['Status'] == 'Success'
])
if not controllers:
logging.error(
"Failed to execute command '{}'. "
"Ignoring data.".format(command)
)
return {}
return controllers
class StorcliController(RaidController): class StorcliController(RaidController):
@ -22,52 +45,101 @@ class StorcliController(RaidController):
def get_firmware_version(self): def get_firmware_version(self):
return self.data['FW Package Build'] return self.data['FW Package Build']
def get_physical_disks(self): def _get_physical_disks(self):
ret = [] pds = {}
output = subprocess.getoutput( cmd = '/c{}/eall/sall show all'.format(self.controller_index)
'storcli /c{}/eall/sall show all J'.format(self.controller_index) controllers = storecli(cmd)
) pd_info = controllers[self.controller_index]
drive_infos = json.loads(output)['Controllers'][self.controller_index]['Response Data'] pd_re = re.compile(r'^Drive (/c\d+/e\d+/s\d+)$')
for physical_drive in self.data['PD LIST']: for section, attrs in pd_info.items():
enclosure = physical_drive.get('EID:Slt').split(':')[0] reg = pd_re.search(section)
slot = physical_drive.get('EID:Slt').split(':')[1] if reg is None:
size = physical_drive.get('Size').strip() continue
media_type = physical_drive.get('Med').strip() pd_name = reg.group(1)
drive_identifier = 'Drive /c{}/e{}/s{}'.format( pd_attr = attrs[0]
str(self.controller_index), str(enclosure), str(slot) pd_identifier = pd_attr['EID:Slt']
) size = pd_attr.get('Size', '').strip()
drive_attr = drive_infos['{} - Detailed Information'.format(drive_identifier)][ media_type = pd_attr.get('Med', '').strip()
'{} Device attributes'.format(drive_identifier)] pd_details = pd_info['{} - Detailed Information'.format(section)]
model = drive_attr.get('Model Number', '').strip() pd_dev_attr = pd_details['{} Device attributes'.format(section)]
ret.append({ model = pd_dev_attr.get('Model Number', '').strip()
pd = {
'Model': model, 'Model': model,
'Vendor': get_vendor(model), 'Vendor': get_vendor(model),
'SN': drive_attr.get('SN', '').strip(), 'SN': pd_dev_attr.get('SN', '').strip(),
'Size': size, 'Size': size,
'Type': media_type, 'Type': media_type,
'_src': self.__class__.__name__, '_src': self.__class__.__name__,
}) }
return ret if config.process_virtual_drives:
pd.setdefault('custom_fields', {})['pd_identifier'] = pd_name
pds[pd_identifier] = pd
return pds
def _get_virtual_drives_map(self):
vds = {}
cmd = '/c{}/vall show all'.format(self.controller_index)
controllers = storecli(cmd)
vd_info = controllers[self.controller_index]
mount_points = get_mount_points()
for vd_identifier, vd_attrs in vd_info.items():
if not vd_identifier.startswith("/c{}/v".format(self.controller_index)):
continue
volume = vd_identifier.split("/")[-1].lstrip("v")
vd_attr = vd_attrs[0]
vd_pd_identifier = 'PDs for VD {}'.format(volume)
vd_pds = vd_info[vd_pd_identifier]
vd_prop_identifier = 'VD{} Properties'.format(volume)
vd_properties = vd_info[vd_prop_identifier]
for pd in vd_pds:
pd_identifier = pd["EID:Slt"]
wwn = vd_properties["SCSI NAA Id"]
wwn_path = "/dev/disk/by-id/wwn-0x{}".format(wwn)
device = os.path.realpath(wwn_path)
mp = mount_points.get(device, "n/a")
vds[pd_identifier] = {
"vd_array": vd_identifier,
"vd_size": vd_attr["Size"],
"vd_consistency": vd_attr["Consist"],
"vd_raid_type": vd_attr["TYPE"],
"vd_device": device,
"mount_point": ", ".join(sorted(mp))
}
return vds
def get_physical_disks(self):
# Parses physical disks information
pds = self._get_physical_disks()
# Parses virtual drives information and maps them to physical disks
vds = self._get_virtual_drives_map()
for pd_identifier, vd in vds.items():
if pd_identifier not in pds:
logging.error(
"Physical drive {} listed in virtual drive {} not "
"found in drives list".format(
pd_identifier, vd["vd_array"]
)
)
continue
pds[pd_identifier].setdefault("custom_fields", {}).update(vd)
return list(pds.values())
class StorcliRaid(Raid): class StorcliRaid(Raid):
def __init__(self): def __init__(self):
self.output = subprocess.getoutput('storcli /call show J')
self.data = json.loads(self.output)
self.controllers = [] self.controllers = []
controllers = storecli('/call show')
if len([ for controller_id, controller_data in controllers.items():
x for x in self.data['Controllers'] self.controllers.append(
if x['Command Status']['Status'] == 'Success' StorcliController(
]) > 0: controller_id,
for controller in self.data['Controllers']: controller_data
self.controllers.append(
StorcliController(
controller['Command Status']['Controller'],
controller['Response Data']
)
) )
)
def get_controllers(self): def get_controllers(self):
return self.controllers return self.controllers

View file

@ -1,9 +1,3 @@
import logging
import socket
import subprocess
import sys
from pprint import pprint
import netbox_agent.dmidecode as dmidecode import netbox_agent.dmidecode as dmidecode
from netbox_agent.config import config from netbox_agent.config import config
from netbox_agent.config import netbox_instance as nb from netbox_agent.config import netbox_instance as nb
@ -12,6 +6,11 @@ from netbox_agent.location import Datacenter, Rack, Tenant
from netbox_agent.misc import create_netbox_tags, get_device_role, get_device_type from netbox_agent.misc import create_netbox_tags, get_device_role, get_device_type
from netbox_agent.network import ServerNetwork from netbox_agent.network import ServerNetwork
from netbox_agent.power import PowerSupply from netbox_agent.power import PowerSupply
from pprint import pprint
import subprocess
import logging
import socket
import sys
class ServerBase(): class ServerBase():
@ -493,3 +492,15 @@ class ServerBase():
Indicates if the device hosts an expansion card Indicates if the device hosts an expansion card
""" """
return False return False
def own_gpu_expansion_slot(self):
"""
Indicates if the device hosts a GPU expansion card
"""
return False
def own_drive_expansion_slot(self):
"""
Indicates if the device hosts a drive expansion bay
"""
return False

View file

@ -8,7 +8,7 @@ class GenericHost(ServerBase):
self.manufacturer = dmidecode.get_by_type(self.dmi, 'Baseboard')[0].get('Manufacturer') self.manufacturer = dmidecode.get_by_type(self.dmi, 'Baseboard')[0].get('Manufacturer')
def is_blade(self): def is_blade(self):
return None return False
def get_blade_slot(self): def get_blade_slot(self):
return None return None