Skip to content

Instantly share code, notes, and snippets.

@andrewladlow
Created April 16, 2024 19:48
Show Gist options
  • Save andrewladlow/9f4d03aab8ef0e957343b65ee6638c3a to your computer and use it in GitHub Desktop.
Save andrewladlow/9f4d03aab8ef0e957343b65ee6638c3a to your computer and use it in GitHub Desktop.
Adjustments to https://github.com/eLvErDe/hwraid to support Dell H750 (and possibly other cards) using perccli
#!/usr/bin/python3
# $Id: megaclisas-status,v 1.78 2018/10/01 03:52:57 root Exp root $
#
# Written by Adam Cecile <gandalf@NOSPAM.le-vert.net>
# Modified by Vincent S. Cojot <vincent@NOSPAM.cojot.name>
#
import os
import re
import sys
import pdb
import inspect
import argparse
if sys.platform == "win32":
import ctypes
def_megaclipath = "/opt/MegaRAID/MegaCli/MegaCli64"
# Non-Nagios Mode defaults
nagiosmode = False
nagiosoutput = ""
nagiosgoodarray = 0
nagiosbadarray = 0
nagiosgooddisk = 0
nagiosbaddisk = 0
# Sane defaults
printarray = True
printcontroller = True
debugmode = False
notempmode = False
totaldrivenumber = 0
totalconfdrivenumber = 0
totalunconfdrivenumber = 0
# Hardcode a max of 16 HBA and 128 LDs for now. LDTable must be initialized to accept populating list of LD's into each ctlr's list.
MaxNumHBA = 16
MaxNumLD = 128
LDTable = [[] * MaxNumHBA for i in range(MaxNumLD)]
NestedLDTable = [[False for i in range(MaxNumLD)] for j in range(MaxNumHBA)]
# Outputs is a 'dict' of all MegaCLI outputs so we can re-use them during loops..
Outputs = {}
ConfDisks = {}
NagiosBadDisks = {}
NagiosGoodDisks = {}
# We need root access to query
if __name__ == "__main__":
# deal with command line options
parser = argparse.ArgumentParser()
parser.add_argument("--nagios", help="enable nagios support", action="store_true")
parser.add_argument("--debug", help="enable debugging output", action="store_true")
parser.add_argument("--notemp", help="disable temperature reporting", action="store_true")
args = parser.parse_args()
nagiosmode = args.nagios
debugmode = args.debug
notempmode = args.notemp
try:
root_or_admin = os.geteuid() == 0
except AttributeError:
root_or_admin = ctypes.windll.shell32.IsUserAnAdmin() != 0
if not root_or_admin:
print("# This script requires Administrator privileges")
sys.exit(5)
# Functions
def dbgprint(msg):
if debugmode:
sys.stderr.write(str("# DEBUG (" + str(inspect.currentframe().f_back.f_lineno) + ") : " + msg + "\n"))
def is_exe(fpath):
return os.path.isfile(fpath) and os.access(fpath, os.X_OK)
def which(program):
import os
fpath, fname = os.path.split(program)
if fpath:
if is_exe(program):
return program
else:
# Add some defaults
os.environ["PATH"] += os.pathsep + "/opt/MegaRAID/MegaCli"
os.environ["PATH"] += os.pathsep + "/ms/dist/hwmgmt/bin"
os.environ["PATH"] += os.pathsep + "/opt/MegaRAID/perccli"
os.environ["PATH"] += os.pathsep + "/opt/MegaRAID/storcli"
os.environ["PATH"] += os.pathsep + "/opt/lsi/storcli"
os.environ["PATH"] += os.pathsep + os.path.dirname(os.path.realpath(sys.argv[0]))
for path in os.environ["PATH"].split(os.pathsep):
dbgprint("Looking in PATH " + str(path))
path = path.strip('"')
exe_file = os.path.join(path, program)
if is_exe(exe_file):
dbgprint('Found "' + program + '" at ' + exe_file)
return exe_file
return None
# Find MegaCli
for megabin in "MegaCli64", "MegaCli", "megacli", "MegaCli.exe", "perccli64", "perccli", "storcli64", "storcli":
dbgprint("Looking for " + str(megabin) + " in PATH...")
megaclipath = which(megabin)
if megaclipath != None:
dbgprint("Will use this executable: " + str(megaclipath))
break
# Check binary exists (and +x), if not print an error message
if megaclipath != None:
if os.path.exists(megaclipath) and os.access(megaclipath, os.X_OK):
pass
else:
if nagiosmode:
print("UNKNOWN - Cannot find " + megaclipath)
else:
print("Cannot find " + megaclipath + "in your PATH. Please install it.")
sys.exit(3)
else:
print('Cannot find "MegaCli{64,}", "megacli{64,}", "perccli{64,}" or "storcli{64,}" in your PATH. Please install one of them.')
sys.exit(3)
#### pdb.set_trace()
def returnWdthFromArrayCol(glarray, idx):
maxwdth = 0
for glrow in glarray:
if len(glrow[idx]) > maxwdth:
maxwdth = len(glrow[idx])
return maxwdth
# Get and cache command output
def getOutput(cmd):
lines = []
if cmd in Outputs:
dbgprint("Got Cached value: " + str(cmd))
lines = Outputs[cmd]
else:
dbgprint("Not a Cached value: " + str(cmd))
output = os.popen(cmd)
for line in output:
if not re.match(r"^$", line.strip()):
lines.append(line.strip())
Outputs[cmd] = lines
return lines
# Get and cache disks, make sure we don't count the same disk twice
def AddDisk(mytable, disk):
lines = []
if disk in mytable:
dbgprint("Disk: " + str(disk) + " Already present in Disk Table")
return False
else:
dbgprint("Confed " + str(nagiosgooddisk) + "/" + str(nagiosbaddisk) + "Disk: " + str(disk) + " Not already present in Disk Table, adding")
mytable[disk] = True
return True
def returnControllerNumber(output):
for line in output:
if re.match(r"^Controller Count.*$", line.strip()):
return int(line.split(":")[1].strip().strip("."))
def returnTotalDriveNumber(output):
for line in output:
if re.match(r"Number of Physical Drives on Adapter.*$", line.strip()):
return int(line.split(":")[1].strip())
def returnRebuildProgress(output):
percent = 0
tmpstr = ""
for line in output:
if re.match(r"^Rebuild Progress on Device at Enclosure.*, Slot .* Completed ", line.strip()):
tmpstr = line.split("Completed")[1].strip()
percent = int(tmpstr.split("%")[0].strip())
return percent
def returnConfDriveNumber(controllerid, output):
# Count the configured drives
confdrives = 0
enclid = "N/A"
slotid = "N/A"
for line in output:
if re.match(r"Enclosure Device ID: .*$", line.strip()):
# We match here early in the analysis so reset the vars if this is a new disk we're reading..
enclid = line.split(":")[1].strip()
elif re.match(r"Slot Number: .*$", line.strip()):
slotid = line.split(":")[1].strip()
if AddDisk(ConfDisks, str(controllerid) + enclid + slotid):
confdrives += 1
return int(confdrives)
def returnUnConfDriveNumber(output):
# Count the un-configured/Hotspare drives
unconfdrives = 0
for line in output:
if re.match(r"^Firmware state: Unconfigured.*$", line.strip()):
unconfdrives += 1
elif re.match(r"^Firmware state: Hotspare.*$", line.strip()):
unconfdrives += 1
return int(unconfdrives)
def returnControllerModel(output):
for line in output:
if re.match(r"^Model.*$", line.strip()):
return line.split("=")[1].strip()
def returnMemorySize(output):
for line in output:
if re.match(r"^On Board Memory Size.*$", line.strip()):
return line.split("=")[1].strip()
def returnFirmwareVersion(output):
for line in output:
if re.match(r"^Firmware Package Build.*$", line.strip()):
return line.split("=")[1].strip()
def returnROCTemp(output):
ROCtemp = ""
tmpstr = ""
if notempmode:
return str("N/A")
else:
for line in output:
if re.match(r"^ROC temperature", line.strip()):
tmpstr = line.split("=")[1].strip()
ROCtemp = re.sub(" +.*$", "", tmpstr)
if ROCtemp != "":
return str(str(ROCtemp) + "C")
else:
return str("N/A")
def returnBBUPresence(output):
BBU = ""
tmpstr = ""
for line in output:
if re.match(r"^BBU = .*$", line.strip()):
tmpstr = line.split("=")[1].strip()
BBU = re.sub(" +.*$", "", tmpstr)
break
if BBU != "":
return str(BBU)
else:
return str("N/A")
def returnBBUStatus(output):
BBUStatus = ""
tmpstr = ""
for line in output:
if re.match(r"^Replacement required .*[A-Za-z].*$", line.strip()):
tmpstr = line.split(" ")[-1].strip()
BBUStatus = re.sub(" +.*$", "", tmpstr)
break
if BBUStatus == "Yes":
return str("REPL")
else:
return str("Good")
def returnArrayNumber(output):
i = 0
for line in output:
if re.match(r"^(CacheCade )?Virtual Drive:.*$", line.strip()):
i += 1
return i
def returnHBAPCIInfo(output):
busprefix = "0000"
busid = ""
devid = ""
functionid = ""
pcipath = ""
for line in output:
if re.match(r"^Bus Number.*:.*$", line.strip()):
busid = str(line.strip().split(":")[1].strip()).zfill(2)
if re.match(r"^Device Number.*:.*$", line.strip()):
devid = str(line.strip().split(":")[1].strip()).zfill(2)
if re.match(r"^Function Number.*:.*$", line.strip()):
functionid = str(line.strip().split(":")[1].strip()).zfill(1)
if busid:
pcipath = str(busprefix + ":" + busid + ":" + devid + "." + functionid)
dbgprint("Array PCI path : " + pcipath)
return str(pcipath)
else:
return None
def returnHBAInfo(table, output, controllerid):
controllermodel = "Unknown"
controllerram = "Unknown"
controllerrev = "Unknown"
controllertemp = ""
controllermodel = returnControllerModel(output)
controllerram = returnMemorySize(output)
controllerrev = returnFirmwareVersion(output)
controllertemp = returnROCTemp(output)
controllerbbu = returnBBUPresence(output)
if controllerbbu == "Present":
#cmd = "%s -AdpBbuCmd -GetBbuStatus -a%d -NoLog" % (megaclipath, controllerid)
cmd = "%s /c%d /bbu show status" % (megaclipath, controllerid)
output = getOutput(cmd)
controllerbbu = returnBBUStatus(output)
if controllermodel != "Unknown":
table.append(["c" + str(controllerid), controllermodel, controllerram, str(controllertemp), str(controllerbbu), str("FW: " + controllerrev)])
def returnArrayInfo(output, controllerid, arrayid, arrayindex):
id = "c" + str(controllerid) + "u" + str(arrayid)
operationlinennumber = False
linenumber = 0
targetid = ""
raidtype = ""
raidlvl = ""
size = ""
state = "N/A"
strpsz = ""
dskcache = "N/A"
properties = ""
spandepth = 0
diskperspan = 0
cachecade_info = "None"
for line in output:
if re.match(r"^(CacheCade )?Virtual Drive:.*(Target Id: [0-9]+).*$", line.strip()):
# Extract the SCSI Target ID
targetid = line.strip().split(":")[2].split(")")[0].strip()
elif re.match(r"^RAID Level.*?:.*$", line.strip()):
# Extract the primary raid type, decide on X0 RAID level later when we hit Span Depth
raidlvl = int(line.strip().split(":")[1].split(",")[0].split("-")[1].strip())
elif re.match(r"^Size.*?:.*$", line.strip()):
# Size reported in MB
if re.match(r"^.*MB$", line.strip().split(":")[1]):
size = line.strip().split(":")[1].strip("MB").strip()
if float(size) > 1000:
size = str(int(round((float(size) / 1000)))) + "G"
else:
size = str(int(round(float(size)))) + "M"
# Size reported in TB
elif re.match(r"^.*TB$", line.strip().split(":")[1]):
size = line.strip().split(":")[1].strip("TB").strip()
size = str(int(round((float(size) * 1000)))) + "G"
# Size reported in GB (default)
else:
size = line.strip().split(":")[1].strip("GB").strip()
size = str(int(round((float(size))))) + "G"
elif re.match(r"^Span Depth.*?:.*$", line.strip()):
# If Span Depth is greater than 1 chances are we have a RAID 10, 50 or 60
spandepth = line.strip().split(":")[1].strip()
elif re.match(r"^State.*?:.*$", line.strip()):
state = line.strip().split(":")[1].strip()
elif re.match(r"^Strip Size.*?:.*$", line.strip()):
strpsz = line.strip().split(":")[1].strip()
elif re.match(r"^Number Of Drives per span.*:.*$", line.strip()):
diskperspan = int(line.strip().split(":")[1].strip())
elif re.match(r"^Current Cache Policy.*?:.*$", line.strip()):
props = line.strip().split(":")[1].strip()
if re.search("ReadAdaptive", props):
properties += "ADRA"
if re.search("ReadAhead", props):
properties += "RA"
if re.match("ReadAheadNone", props):
properties += "NORA"
if re.search("WriteBack", props):
properties += ",WB"
if re.match("WriteThrough", props):
properties += ",WT"
elif re.match(r"^Disk Cache Policy.*?:.*$", line.strip()):
props = line.strip().split(":")[1].strip()
if re.search("Disabled", props):
dskcache = "Disabled"
if re.search("Disk.s Default", props):
dskcache = "Default"
if re.search("Enabled", props):
dskcache = "Enabled"
elif re.match(r"^Ongoing Progresses.*?:.*$", line.strip()):
operationlinennumber = linenumber
elif re.match(r"Cache Cade Type\s*:.*$", line):
cachecade_info = "Type : " + line.strip().split(":")[1].strip()
elif re.match(r"^Target Id of the Associated LDs\s*:.*$", line):
associated = []
for array in line.split(":")[1].strip().split(","):
if array.isdigit():
associated.append("c%du%d" % (controllerid, int(array)))
if len(associated) >= 1:
cachecade_info = "Associated : %s" % (", ".join(associated))
linenumber += 1
# If there was an ongoing operation, find the relevant line in the previous output
if operationlinennumber:
inprogress = str(output[operationlinennumber + 1])
# some ugly output fix..
str1 = inprogress.split(":")[0].strip()
str2 = inprogress.split(":")[1].strip()
inprogress = str1 + " : " + str2
else:
inprogress = "None"
# Compute the RAID level
NestedLDTable[int(controllerid)][int(arrayindex)] = False
if raidlvl == "":
raidtype = str("N/A")
else:
if int(spandepth) >= 2:
raidtype = str("RAID-" + str(raidlvl) + "0")
NestedLDTable[controllerid][int(arrayindex)] = True
else:
if raidlvl == 1:
if diskperspan > 2:
raidtype = str("RAID-10")
NestedLDTable[controllerid][int(arrayindex)] = True
else:
raidtype = str("RAID-" + str(raidlvl))
else:
raidtype = str("RAID-" + str(raidlvl))
dbgprint("RAID Level: " + str(raidlvl) + " Span Depth: " + str(spandepth) + " Disk Per Span: " + str(diskperspan) + " Raid Type: " + str(raidtype))
return [id, raidtype, size, strpsz, properties, dskcache, state, targetid, cachecade_info, inprogress]
def returnDiskInfo(output, controllerid):
arrayid = False
arrayindex = -1
sarrayid = "Unknown"
diskid = False
oldenclid = False
enclid = False
spanid = False
slotid = False
lsidid = "Unknown"
table = []
fstate = "Offline"
substate = "Unknown"
model = "Unknown"
speed = "Unknown"
dsize = "Unknown"
temp = "Unk0C"
percent = 0
for line in output:
if re.match(r"^Span: [0-9]+ - Number of PDs:", line.strip()):
spanid = line.split(":")[1].strip()
spanid = re.sub(" - Number of PDs.*", "", spanid)
elif re.match(r"Enclosure Device ID: .*$", line.strip()):
# We match here early in the analysis so reset the vars if this is a new disk we're reading..
oldenclid = enclid
enclid = line.split(":")[1].strip().replace("N/A", "")
if oldenclid != False:
fstate = "Offline"
model = "Unknown"
speed = "Unknown"
temp = "Unk0C"
slotid = False
lsidid = "Unknown"
elif re.match(r"^Coerced Size: ", line.strip()):
dsize = line.split(":")[1].strip()
dsize = re.sub(" \[.*\.*$", "", dsize)
dsize = re.sub("[0-9][0-9] GB", " Gb", dsize)
elif re.match(r"^(CacheCade )?Virtual (Disk|Drive): [0-9]+.*$", line.strip()):
arrayindex += 1
arrayid = line.split("(")[0].split(":")[1].strip()
elif re.match(r"^Drive.s posi*tion: DiskGroup: [0-9]+,.*$", line.strip()):
notarrayid = line.split(",")[1].split(":")[1].strip()
elif re.match(r"PD: [0-9]+ Information.*$", line.strip()):
diskid = line.split()[1].strip()
elif re.match(r"^Device Id: .*$", line.strip()):
lsidid = line.split(":")[1].strip()
elif re.match(r"Slot Number: .*$", line.strip()):
slotid = line.split(":")[1].strip()
elif re.match(r"Firmware state: .*$", line.strip()):
fstate = line.split(":")[1].strip()
subfstate = re.sub("\(.*", "", fstate)
dbgprint("Firmware State: " + str(fstate) + " " + str(subfstate))
elif re.match(r"Inquiry Data: .*$", line.strip()):
model = line.split(":")[1].strip()
model = re.sub(" +", " ", model)
# re-define our "sub-code"
# our seagate drives have an ID string of
# 'Z1E19S2QST2000DM001-1CH164 CC43'
# or
# '6XW02738ST32000542AS CC32'
m = re.match(r"(\w{8})(ST\w+)(?:-(\w{6}))?(?:\s+(\w+))", model)
if m:
if m.group(3):
model = "{0}-{1} {2} {3}".format(m.group(2), m.group(3), m.group(4), m.group(1))
else:
model = "{0} {1:>10} {2}".format(m.group(2), m.group(4), m.group(1))
continue
# Sub code
manuf = re.sub(" .*", "", model)
dtype = re.sub(manuf + " ", "", model)
dtype = re.sub(" .*", "", dtype)
hwserial = re.sub(".*" + dtype + " *", "", model)
elif re.match(r"^Media Type: .*$", line.strip()):
mtype = line.split(":")[1].strip()
if mtype == "Hard Disk Device":
mtype = "HDD"
else:
if mtype == "Solid State Device":
mtype = "SSD"
else:
mtype = "N/A"
elif re.match(r"Device Speed: .*$", line.strip()):
speed = line.split(":")[1].strip()
elif re.match(r"Drive Temperature :.*$", line.strip()):
if notempmode:
temp = "N/A"
else:
# Drive temp is amongst the last few lines matched, decide here if we add information to the table..
temp = line.split(":")[1].strip()
temp = re.sub(" \(.*\)", "", temp)
if model != "Unknown":
dbgprint("Disk Info: " + str(arrayid) + " " + str(diskid) + " " + str(oldenclid))
if subfstate == "Rebuild":
cmd = "%s pdrbld -showprog -physdrv\[%s:%s\] -a%d -NoLog" % (megaclipath, enclid, slotid, controllerid)
output = getOutput(cmd)
percent = returnRebuildProgress(output)
fstate = str("Rebuilding (%d%%)" % (percent))
if (NestedLDTable[controllerid][int(arrayindex)] == True) and (spanid != False):
sarrayid = str(arrayid) + "s" + spanid
else:
sarrayid = str(arrayid)
table.append([sarrayid, str(diskid), mtype, model, dsize, fstate, speed, temp, enclid, slotid, lsidid])
return table
def returnUnconfDiskInfo(output, controllerid):
arrayid = False
diskid = False
olddiskid = False
enclid = False
slotid = False
lsidid = "Unknown"
table = []
fstate = "Offline"
substate = "Unknown"
model = "Unknown"
speed = "Unknown"
mtype = "Unknown"
dsize = "Unknown"
temp = "Unk0C"
ospath = "N/A"
for line in output:
if re.match(r"Enclosure Device ID: .*$", line.strip()):
# We match here early in the analysis so reset the vars if this is a new disk we're reading..
oldenclid = enclid
enclid = line.split(":")[1].strip().replace("N/A", "")
if oldenclid != False:
arrayid = False
fstate = "Offline"
model = "Unknown"
speed = "Unknown"
temp = "Unk0C"
slotid = False
lsidid = "Unknown"
elif re.match(r"^Coerced Size: ", line.strip()):
dsize = line.split(":")[1].strip()
dsize = re.sub(" \[.*\.*$", "", dsize)
dsize = re.sub("[0-9][0-9] GB", " Gb", dsize)
elif re.match(r"^Drive.s posi*tion: DiskGroup: [0-9]+,.*$", line.strip()):
arrayid = line.split(",")[1].split(":")[1].strip()
elif re.match(r"^Device Id: [0-9]+.*$", line.strip()):
diskid = line.split(":")[1].strip()
elif re.match(r"Slot Number: .*$", line.strip()):
slotid = line.split(":")[1].strip()
elif re.match(r"Firmware state: .*$", line.strip()):
fstate = line.split(":")[1].strip()
subfstate = re.sub("\(.*", "", fstate)
dbgprint("Firmware State: " + str(fstate) + " " + str(subfstate))
elif re.match(r"Inquiry Data: .*$", line.strip()):
model = line.split(":")[1].strip()
model = re.sub(" +", " ", model)
# re-define our "sub-code"
# our seagate drives have an ID string of
# 'Z1E19S2QST2000DM001-1CH164 CC43'
# or
# '6XW02738ST32000542AS CC32'
m = re.match(r"(\w{8})(ST\w+)(?:-(\w{6}))?(?:\s+(\w+))", model)
if m:
if m.group(3):
model = "{0}-{1} {2} {3}".format(m.group(2), m.group(3), m.group(4), m.group(1))
else:
model = "{0} {1:>10} {2}".format(m.group(2), m.group(4), m.group(1))
continue
manuf = re.sub(" .*", "", model)
dtype = re.sub(manuf + " ", "", model)
dtype = re.sub(" .*", "", dtype)
hwserial = re.sub(".*" + dtype + " *", "", model)
elif re.match(r"^Media Type: .*$", line.strip()):
mtype = line.split(":")[1].strip()
if mtype == "Hard Disk Device":
mtype = "HDD"
else:
if mtype == "Solid State Device":
mtype = "SSD"
else:
mtype = "N/A"
elif re.match(r"Device Speed: .*$", line.strip()):
speed = line.split(":")[1].strip()
elif re.match(r"Drive Temperature :.*$", line.strip()):
# Drive temp is amongst the last few lines matched, decide here if we add information to the table..
if notempmode:
temp = "N/A"
else:
temp = line.split(":")[1].strip()
temp = re.sub("\(.*\)", "", temp)
if arrayid == False:
if subfstate == "Unconfigured":
dbgprint("Unconfigured Disk: Arrayid: " + str(arrayid) + " DiskId: " + str(diskid) + " " + str(olddiskid) + " " + str(fstate))
elif subfstate == "Online, Spun Up":
dbgprint("Online Unconfed Disk: Arrayid: " + str(arrayid) + " DiskId: " + str(diskid) + " " + str(olddiskid) + " " + str(fstate))
table.append([mtype, model, dsize, fstate, speed, temp, enclid, slotid, diskid, ospath])
return table
cmd = "%s -adpCount -NoLog" % (megaclipath)
output = getOutput(cmd)
controllernumber = returnControllerNumber(output)
bad = False
# List available controller
if printcontroller:
if controllernumber:
if not nagiosmode:
print("-- Controller information --")
i = 0
controllerid = 0
mlen = 0
hbainfo = []
while controllerid < controllernumber:
#cmd = "%s -AdpAllInfo -a%d -NoLog" % (megaclipath, controllerid)
cmd = "%s /c%d show all" % (megaclipath, controllerid)
output = getOutput(cmd)
returnHBAInfo(hbainfo, output, controllerid)
controllerid += 1
mlen = returnWdthFromArrayCol(hbainfo, 1)
controllerid = 0
for hba in hbainfo:
hbafmt = str("%-5s | %-" + str(mlen) + "s | %-6s | %-4s | %-6s | %-12s ")
# Header
if i == 0:
if not nagiosmode:
print(hbafmt % ("-- ID", "H/W Model", "RAM", "Temp", "BBU", "Firmware"))
if not nagiosmode:
print(hbafmt % (hba[0], hba[1], hba[2], hba[3], hba[4], hba[5]))
i += 1
if not nagiosmode:
print("")
else:
print("No MegaRAID or PERC adapter detected on your system!")
exit(1)
if printarray:
if not nagiosmode:
print("-- Array information --")
controllerid = 0
pcipath = ""
diskpath = ""
i = 0
j = 0
mlen = 0
rlen = 0
clen = 0
while controllerid < controllernumber:
arrayindex = 0
cmd = "%s -LDInfo -lall -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
arraynumber = returnArrayNumber(output)
# We need to explore each HBA to look for gaps in LD's
ldid = 0
ldcount = 0
while ldcount < arraynumber:
cmd = "%s -LDInfo -l%d -a%d -NoLog" % (megaclipath, ldid, controllerid)
output = getOutput(cmd)
for line in output:
if re.match(r"^Adapter.*Virtual Drive .* Does not Exist", line.strip()):
ldid += 1
elif re.match(r"^(CacheCade )?Virtual Drive:", line.strip()):
LDTable[controllerid].append(ldid)
# NestedLDTable[controllerid][int(arrayindex)] = False
ldcount += 1
ldid += 1
while arrayindex < arraynumber:
ldid = LDTable[controllerid][arrayindex]
cmd = "%s -LDInfo -l%d -a%d -NoLog" % (megaclipath, ldid, controllerid)
output = getOutput(cmd)
arrayinfo = returnArrayInfo(output, controllerid, ldid, arrayindex)
if len(arrayinfo[1]) > rlen:
rlen = len(arrayinfo[1])
if len(arrayinfo[4]) > mlen:
mlen = len(arrayinfo[4])
if len(arrayinfo[8]) > clen:
clen = len(arrayinfo[8])
arrayindex += 1
controllerid += 1
controllerid = 0
while controllerid < controllernumber:
arrayindex = 0
cmd = "%s -AdpGetPciInfo -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
pcipath = returnHBAPCIInfo(output)
cmd = "%s -LDInfo -lall -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
arraynumber = returnArrayNumber(output)
while arrayindex < arraynumber:
ldid = LDTable[controllerid][arrayindex]
cmd = "%s -LDInfo -l%d -a%d -NoLog" % (megaclipath, ldid, controllerid)
output = getOutput(cmd)
arrayinfo = returnArrayInfo(output, controllerid, ldid, arrayindex)
if pcipath:
diskprefix = str("/dev/disk/by-path/pci-" + pcipath + "-scsi-0:")
dbgprint("Will look for DISKprefix : " + diskprefix)
# RAID disks are usually with a channel of '2', JBOD disks with a channel of '0'
for j in range(1, 8):
diskpath = diskprefix + str(j) + ":" + str(arrayinfo[7]) + ":0"
dbgprint("Looking for DISKpath : " + diskpath)
if os.path.exists(diskpath):
arrayinfo[7] = os.path.realpath(diskpath)
dbgprint("Found DISK match: " + diskpath + " -> " + arrayinfo[7])
break
else:
arrayinfo[7] = "N/A"
# Pad the string length, just to make sure it's aligned with the headers...
if rlen < len("Type"):
rlen = len("Type")
if mlen < len("Flags"):
mlen = len("Flags")
if clen < len("CacheCade"):
clen = len("CacheCade")
ldfmt = str("%-6s | %-" + str(rlen) + "s | %7s | %7s | %" + str(mlen) + "s | %8s | %8s | %8s | %-" + str(clen) + "s |%-12s ")
# Header
if i == 0:
if not nagiosmode:
print(ldfmt % ("-- ID", "Type", "Size", "Strpsz", "Flags", "DskCache", "Status", "OS Path", "CacheCade", "InProgress"))
if not nagiosmode:
print(
ldfmt
% (
arrayinfo[0],
arrayinfo[1],
arrayinfo[2],
arrayinfo[3],
arrayinfo[4],
arrayinfo[5],
arrayinfo[6],
arrayinfo[7],
arrayinfo[8],
arrayinfo[9],
)
)
dbgprint("Array state : LD " + arrayinfo[0] + ", status : " + arrayinfo[6])
if arrayinfo[6] not in ["Optimal", "N/A"]:
bad = True
nagiosbadarray += 1
else:
nagiosgoodarray += 1
arrayindex += 1
i += 1
controllerid += 1
if not nagiosmode:
print("")
controllerid = 0
while controllerid < controllernumber:
cmd = "%s -PDGetNum -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
totaldrivenumber += returnTotalDriveNumber(output)
controllerid += 1
if totaldrivenumber:
if not nagiosmode:
print("-- Disk information --")
i = 0
dlen = 0
mlen = 0
flen = 0
controllerid = 0
while controllerid < controllernumber:
arrayid = 0
cmd = "%s -LDInfo -lall -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
arraynumber = returnArrayNumber(output)
#### BUG: -LdPdInfo shows all PD on the adapter, not just for the LD we wanted..
#### while arrayid <= arraynumber:
cmd = "%s -LdPdInfo -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
arraydisk = returnDiskInfo(output, controllerid)
for array in arraydisk:
diskname = str(controllerid) + array[8] + array[9]
dbgprint("Disk c" + diskname + " status : " + array[5])
if re.match("|".join(["^Online$", "^Online, Spun Up$", "^Rebuilding \(.*"]), array[5]):
if AddDisk(NagiosGoodDisks, diskname):
nagiosgooddisk += 1
else:
bad = True
if AddDisk(NagiosBadDisks, diskname):
nagiosbaddisk += 1
if returnWdthFromArrayCol(arraydisk, 0) > dlen:
dlen = returnWdthFromArrayCol(arraydisk, 0)
if returnWdthFromArrayCol(arraydisk, 3) > mlen:
mlen = returnWdthFromArrayCol(arraydisk, 3)
if returnWdthFromArrayCol(arraydisk, 5) > flen:
flen = returnWdthFromArrayCol(arraydisk, 5)
controllerid += 1
controllerid = 0
while controllerid < controllernumber:
arrayid = 0
cmd = "%s -LDInfo -lall -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
arraynumber = returnArrayNumber(output)
#### BUG: -LdPdInfo shows all PD on the adapter, not just for said LD..
#### while arrayid <= arraynumber:
cmd = "%s -LdPdInfo -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
arraydisk = returnDiskInfo(output, controllerid)
# Adjust print format with width computed above
drvfmt = "%-" + str(dlen + 6) + "s | %-4s | %-" + str(mlen) + "s | %-8s | %-" + str(flen) + "s | %-8s | %-4s | %-8s | %-8s"
for array in arraydisk:
# Header
if i == 0:
if not nagiosmode:
print(drvfmt % ("-- ID", "Type", "Drive Model", "Size", "Status", "Speed", "Temp", "Slot ID", "LSI ID"))
# Drive information
if not nagiosmode:
print(
drvfmt
% (
str("c" + str(controllerid) + "u" + array[0] + "p" + array[1]), # c0p0
array[2], # HDD/SDD
array[3], # Model Information (Variable len)
array[4], # Size
array[5], # Status (Variable len)
array[6], # Speed
array[7], # Temp
str("[" + array[8] + ":" + array[9] + "]"), # Slot ID
array[10],
)
) # LSI ID
i = i + 1
controllerid += 1
if not nagiosmode:
print("")
controllerid = 0
totalconfdrivenumber = 0
totalunconfdrivenumber = 0
totaldrivenumber = 0
while controllerid < controllernumber:
cmd = "%s -LdPdInfo -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
totalconfdrivenumber += returnConfDriveNumber(controllerid, output)
cmd = "%s -PDGetNum -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
totaldrivenumber += returnTotalDriveNumber(output)
cmd = "%s -PDList -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
# Sometimes a drive will be reconfiguring without any info on that it is going through a rebuild process.
# This happens when expanding an R{5,6,50,60} array, for example. In that case, totaldrivenumber will still be
# greater than totalconfdrivenumber while returnUnConfDriveNumber(output) will be zero. The math below attempts to solve this.
totalunconfdrivenumber += max(returnUnConfDriveNumber(output), totaldrivenumber - totalconfdrivenumber)
controllerid += 1
dbgprint("Total Drives in system : " + str(totaldrivenumber))
dbgprint("Total Configured Drives : " + str(totalconfdrivenumber))
dbgprint("Total Unconfigured Drives : " + str(totalunconfdrivenumber))
if totalunconfdrivenumber:
if not nagiosmode:
print("-- Unconfigured Disk information --")
controllerid = 0
pcipath = ""
while controllerid < controllernumber:
arrayid = 0
cmd = "%s -LDInfo -lall -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
arraynumber = returnArrayNumber(output)
cmd = "%s -AdpGetPciInfo -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
pcipath = returnHBAPCIInfo(output)
#### BUG: -LdPdInfo shows all PD on the adapter, not just for given LD..
#### while arrayid <= arraynumber:
cmd = "%s -PDList -a%d -NoLog" % (megaclipath, controllerid)
output = getOutput(cmd)
arraydisk = returnUnconfDiskInfo(output, controllerid)
for array in arraydisk:
dbgprint("Unconfed " + str(nagiosgooddisk) + "/" + str(nagiosbaddisk) + " Disk c" + str(controllerid) + "uXpY status : " + array[3])
if array[3] in [
"Online",
"Unconfigured(good), Spun Up",
"Unconfigured(good), Spun down",
"JBOD",
"Hotspare, Spun Up",
"Hotspare, Spun down",
"Online, Spun Up",
]:
nagiosgooddisk += 1
else:
bad = True
nagiosbaddisk += 1
# JBOD disks has a real device path and are not masked. Try to find a device name here, if possible.
if pcipath:
if array[3] in ["JBOD"]:
diskprefix = str("/dev/disk/by-path/pci-" + pcipath + "-scsi-0:0:")
dbgprint("Will look for DISKprefix : " + diskprefix)
# RAID disks are usually with a channel of '2', JBOD disks with a channel of '0'
diskpath = diskprefix + str(array[8]) + ":0"
dbgprint("Looking for DISKpath : " + diskpath)
if os.path.exists(diskpath):
dbgprint("Found DISK match: " + diskpath + " -> " + array[9])
array[9] = os.path.realpath(diskpath)
else:
dbgprint("DISK NOT present: " + diskpath)
array[9] = "N/A"
mlen = returnWdthFromArrayCol(arraydisk, 1)
flen = returnWdthFromArrayCol(arraydisk, 3)
# Adjust print format with widths computed above
drvfmt = "%-7s | %-4s | %-" + str(mlen) + "s | %-8s | %-" + str(flen + 2) + "s | %-8s | %-4s | %-8s | %-6s | %-8s"
i = 0
for array in arraydisk:
# Header
if i == 0:
if not nagiosmode:
print(drvfmt % ("-- ID", "Type", "Drive Model", "Size", "Status", "Speed", "Temp", "Slot ID", "LSI ID", "Path"))
# Drive information
if not nagiosmode:
print(
drvfmt
% (
str("c" + str(controllerid) + "uXpY"), # cXpY
array[0], # HDD/SDD
array[1], # Model Information (Variable len)
array[2], # Size
array[3], # Status (Variable len)
array[4], # Speed
array[5], # Temp
str("[" + array[6] + ":" + array[7] + "]"), # Slot ID
array[8], # LSI ID
array[9],
)
) # OS path, if any
i += 1
controllerid += 1
if not nagiosmode:
print("")
if debugmode:
dbgprint("Printing Outputs[][]")
for myl in Outputs:
dbgprint(myl + "\n")
sys.stderr.write("\n".join("".join(map(str, myd)) for myd in Outputs[myl]) + "\n")
dbgprint("Printing arraydisk[]")
sys.stderr.write("\n".join(" | ".join(map(str, myd)) for myd in arraydisk) + "\n")
dbgprint("Printing ConfDisks[]")
sys.stderr.write("\n".join("".join(map(str, myd)) for myd in ConfDisks) + "\n")
dbgprint("Printing NagiosGoodDisks[]")
sys.stderr.write("\n".join("".join(map(str, myd)) for myd in NagiosGoodDisks) + "\n")
dbgprint("Printing NagiosBadDisks[]")
sys.stderr.write("\n".join("".join(map(str, myd)) for myd in NagiosBadDisks) + "\n")
if nagiosmode:
if bad:
print(
"RAID ERROR - Arrays: OK:"
+ str(nagiosgoodarray)
+ " Bad:"
+ str(nagiosbadarray)
+ " - Disks: OK:"
+ str(nagiosgooddisk)
+ " Bad:"
+ str(nagiosbaddisk)
)
sys.exit(2)
else:
print(
"RAID OK - Arrays: OK:"
+ str(nagiosgoodarray)
+ " Bad:"
+ str(nagiosbadarray)
+ " - Disks: OK:"
+ str(nagiosgooddisk)
+ " Bad:"
+ str(nagiosbaddisk)
)
else:
if bad:
# DO NOT MODIFY OUTPUT BELOW
# Scripts may relies on it
# https://github.com/eLvErDe/hwraid/issues/99
print("\nThere is at least one disk/array in a NOT OPTIMAL state.")
print(
"RAID ERROR - Arrays: OK:"
+ str(nagiosgoodarray)
+ " Bad:"
+ str(nagiosbadarray)
+ " - Disks: OK:"
+ str(nagiosgooddisk)
+ " Bad:"
+ str(nagiosbaddisk)
)
sys.exit(1)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment