# Copyright (c) 2014 The Foundry Visionmongers Ltd.  All Rights Reserved.
# Warning: the code below is extracted from an internal test tool (Tests/Common/FnSysInfo.py) and is not recommended for
# use in your own scripts.
import os, sys, glob, math
import xml.sax.saxutils
#Helpers
#Temporary until we fix system profiler.
[docs]def RunCmdWithTimeout(cmd, timeout):
  import subprocess, time
  p = subprocess.Popen(cmd, stderr=subprocess.PIPE, stdout=subprocess.PIPE, shell=True)
  timerStep = 0.5
  runTime = 0
  while True:
    if p.poll() is not None:
      break
    runTime += timerStep
    if runTime > timeout:
      p.terminate()
      return -1
      sys.exit(1)
    time.sleep(timerStep)
  return 0 
#End temporary
[docs]def ConvertUnits(s, token, multiplier):
  if s.find(token) != -1:
    realSpeed = s[:s.find(token)]
    realSpeedFlt = float(realSpeed)
    return str(int(realSpeedFlt * multiplier)) 
[docs]def ConvertSpeedUnitsToMhZ(s):
  p = ConvertUnits(s, "GHz", 1000)
  if p != None:
    return p
  return str(int(math.ceil(float(s)))) 
[docs]def ConvertMemSizeToKb(s):
  p = ConvertUnits(s, "MB", 1000)
  if p != None:
    return p
  p = ConvertUnits(s, "GB", 1000000)
  if p != None:
    return p
  p = ConvertUnits(s, "Gb", 1000000)
  if p != None:
    return p
  p = ConvertUnits(s, "kB", 1)
  if p != None:
    return p
  p = ConvertUnits(s, "KB", 1)
  if p != None:
    return p
  return s 
#Standard Identifiers
gCPUSpeed = "CPUSpeed_Mhz"
gMachineName = "MachineName"
gNumCPUs = "NumCPUs"
gNumCores = "NumCores"
gRAM = "RAM_Kb"
gCPUType = "CPUType"
gOS = "OS"
gBusSpeed = "BusSpeed"
gL2Cache = "L2Cache_Kb"
gOSVersion = "OSVersion"
gKernelVersion = "KernelVersion"
[docs]class HardwareInfo:
  def __init__(self):
    self._standardDict = {}
    self.gLogStr = ""
    if sys.platform.lower() == "darwin":
      self.SafeCall(self.initMac)
    elif (sys.platform.lower() == "linux") | (sys.platform.lower() == "linux2"):
      self.SafeCall(self.initLinux)
    elif (sys.platform.lower().find("win") == 0):
      self.SafeCall(self.initWin)
  def SafeCall(self, f, *positional, **keyword):
    try:
      return f(*positional, **keyword)
    except Exception as e:
      self.gLogStr += "Function: calling " + f.__name__ + "\n"
      self.gLogStr += "\tDescription:" + str(f.__doc__) + "\n"
      self.gLogStr += "\tError:" + str(e) + "\n"
  def testCatCommand(self, info):
    if not os.path.exists(info):
      self.gLogStr += "Failed to find " + info
      return False
    return True
  def parseProcInfo(self, info):
    success = self.testCatCommand(info)
    if not success:
      return ([{}], False)
    import subprocess
    cpuinfo = subprocess.getoutput("cat " + info)
    return self.parseProcInfoStr(cpuinfo)
  def parseProcInfoStr(self, cpuinfo):
    import copy
    values = []
    currentDict = {}
    for i in cpuinfo.split("\n"):
      if i.find(":") == -1:
        values.append(copy.deepcopy(currentDict))
        currentDict = {}
        continue
      tokens = i.split(":")
      name = tokens[0]
      value = ":".join(tokens[1:])
      currentDict[name.strip()] = value.strip()
    if len(currentDict) != 0:
      values.append(copy.deepcopy(currentDict))
    return (values, True)
  def parseCPUInfo(self):
    (itemDict, success) = self.parseProcInfo("/proc/cpuinfo")
    if success:
      mapping = [["cpu MHz", gCPUSpeed],
                 ["model name", gCPUType],
                 ["cache size", gL2Cache]]
      self.MapDictionaries(itemDict, self._standardDict, mapping, "/proc/cpuinfo")
      self._standardDict[gL2Cache] =  ConvertMemSizeToKb(self._standardDict[gL2Cache])
      self._standardDict[gCPUSpeed] = ConvertSpeedUnitsToMhZ(self._standardDict[gCPUSpeed])
      self._standardDict[gNumCores] = len(itemDict)
  def ParseProcVersion(self):
    if self.testCatCommand("/etc/redhat-release"):
      import subprocess
      osVersion = subprocess.getoutput("cat /etc/redhat-release")
      self._standardDict[gOSVersion] = osVersion
    elif self.testCatCommand("/etc/lsb-release"):
      import subprocess
      lsbInfo = subprocess.getoutput("cat /etc/lsb-release")
      for i in lsbInfo.split("\n"):
        if i.find("DISTRIB_DESCRIPTION") != -1:
          self._standardDict[gOSVersion] = i.split("=")[1].replace("\"","")
    elif self.testCatCommand("/proc/version"):
      import subprocess
      osVersion = subprocess.getoutput("cat /proc/version")
      start = osVersion.find(" #1 ")
      osVersionShort = osVersion[:start]
      token = osVersionShort.split("(")[-1]
      self._standardDict[gOSVersion] = token.replace(")", "")
  def parseProcSimple(self, file, entry):
    import subprocess
    if os.path.exists(file):
      self._standardDict[entry] = subprocess.getoutput("cat " + file)
    else:
      self.gLogStr += file + " doesn't exist.\n"
      self._standardDict[entry] = "Unknown"
  def parseMemInfo(self):
    (itemDict, success) = self.parseProcInfo("/proc/meminfo")
    if success:
      self.MapDictionaries(itemDict, self._standardDict, [["MemTotal", gRAM]], "/proc/meminfo")
      self._standardDict[gRAM] =  ConvertMemSizeToKb(self._standardDict[gRAM])
  def initLinux(self):
    self._standardDict[gOS] = "linux"
    self.SafeCall(self.parseCPUInfo)
    self.SafeCall(self.ParseProcVersion)
    self.SafeCall(self.parseProcSimple, "/proc/sys/kernel/osrelease", gKernelVersion)
    self.SafeCall(self.parseProcSimple,	"/proc/sys/kernel/hostname", gMachineName)
    self.SafeCall(self.parseMemInfo)
  def getRegistryNumSubKeys(self, key, subkey):
    try:
      import winreg
      key = getattr(_winreg, key)
      handle = winreg.OpenKey(key, subkey)
      return winreg.QueryInfoKey(handle)[0]
    except:
      self.gLogStr += "Failed to find " + str(key) + ", " + str(subkey) + ", " + str(value) + "\n"
      return "Unknown"
  def getRegistryValue(self, key, subkey, value):
    try:
      import winreg
      key = getattr(_winreg, key)
      handle = winreg.OpenKey(key, subkey)
      (result, type) = winreg.QueryValueEx(handle, value)
      return result
    except:
      self.gLogStr += "Failed to find " + str(key) + ", " + str(subkey) + ", " + str(value) + "\n"
      return "Unknown"
  def getWindowsRam(self):
    try:
      from psutil import virtual_memory
      mem = virtual_memory()
      memGb =  mem.total * 0.000000001
      return str(memGb) + " Gb"
    except:
      return "unable to import psutil - memory information disabled"
  def RunCmdWin(cmd):
    import subprocess
    process = subprocess.Popen(cmd, stderr=subprocess.STDOUT, stdin=subprocess.PIPE, stdout=subprocess.PIPE, shell=True)
    process.stdin.close()
    output = ""
    while line != "":
      line = process.stdout.readline()
      output += line
    return output
  def getWindowsL2Cache(self):
    output = RunCmdWin("wmic cpu get L2CacheSize")
    return output.split("\n")[-1]
  def getWindowsOSVersion(self):
    def get(key):
      return self.getRegistryValue("HKEY_LOCAL_MACHINE", "SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion", key)
    os = get("ProductName")
    sp = get("CSDVersion")
    build = get("CurrentBuildNumber")
    return "%s %s (build %s)" % (os, sp, build)
  def getWindowsMachineName(self):
   return self.getRegistryValue("HKEY_LOCAL_MACHINE",
                                "SYSTEM\\CurrentControlSet\\Control\\ComputerName\\ComputerName",
                                "ComputerName")
  def initWin(self):
    import ctypes
    self._standardDict[gOS] = "win"
    self._standardDict[gCPUType] = self.getRegistryValue("HKEY_LOCAL_MACHINE", "HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\0", "ProcessorNameString")
    self._standardDict[gCPUSpeed] = self.getRegistryValue("HKEY_LOCAL_MACHINE", "HARDWARE\\DESCRIPTION\\System\\CentralProcessor\\0", "~MHz")
    self._standardDict[gNumCores] = self.getRegistryNumSubKeys("HKEY_LOCAL_MACHINE", "HARDWARE\\DESCRIPTION\\System\\CentralProcessor")
    self._standardDict[gRAM] = self.SafeCall(self.getWindowsRam)
    self._standardDict[gRAM] =  ConvertMemSizeToKb(self._standardDict[gRAM])
    self._standardDict[gOSVersion] = self.SafeCall(self.getWindowsOSVersion)
    self._standardDict[gL2Cache] = self.SafeCall(self.getWindowsL2Cache)
    self._standardDict[gMachineName] = self.SafeCall(self.getWindowsMachineName)
  def MapDictionaries(self, originalDict, addTo, mapping, name):
    for i in mapping:
      try:
        addTo[i[1]] = originalDict[0][i[0]]
      except:
        self.gLogStr += "Failed to find key: " + i[0] + " in " + name + "\n"
        addTo[i[1]] = "Unknown"
[docs]  def initMac(self):
    """Initialises the object for mac - relies on system_profiler being in the path"""
    self._standardDict[gOS] = "mac"
    import tempfile
    (handle, tmpPList) = tempfile.mkstemp()
    try:
      os.close(handle)
      status = RunCmdWithTimeout("system_profiler -xml > " + tmpPList, 300.0)
      self.SafeCall(self.initMacFromFile, tmpPList)
    finally:
      os.remove(tmpPList) 
  def initMacHardware(self, itemDicts):
    mapping = [["current_processor_speed", gCPUSpeed],
               ["physical_memory", gRAM],
               ["number_processors", gNumCores],
               ["cpu_type", gCPUType],
               ["l2_cache", gL2Cache]]
    self.MapDictionaries(itemDicts, self._standardDict, mapping, "SPHardwareDataType")
    if self._standardDict[gL2Cache] == "Unknown":
      fixMacChangingThingsMapping = [["l2_cache_share", gL2Cache]]
      self.MapDictionaries(itemDicts, self._standardDict, fixMacChangingThingsMapping, "SPHardwareDataType")
    self._standardDict[gCPUSpeed] = ConvertSpeedUnitsToMhZ(self._standardDict[gCPUSpeed])
    self._standardDict[gL2Cache] =  ConvertMemSizeToKb(self._standardDict[gL2Cache])
    self._standardDict[gRAM] =  ConvertMemSizeToKb(self._standardDict[gRAM])
    extendedMapping = [["boot_rom_version", gBootROMVersion],
                       ["machine_model", gMachineName]]
    self.MapDictionaries(itemDicts, self._extendedDict, extendedMapping, "SPHardwareDataType")
  def initMacSoftware(self, itemDicts):
    mapping = [["os_version", gOSVersion],
               ["kernel_version", gKernelVersion],
               ["local_host_name", gMachineName]]
    self.MapDictionaries(itemDicts, self._standardDict, mapping, "SPSoftwareDataType")
[docs]  def printDict(self, pDict, handle, indentLevel = 1):
    """
    outputs the elements from pDict in XML structure, nesting when an
    element contains another dict
    """
    for key in sorted(pDict):
      elem = pDict[key]
      handle.write("  " * indentLevel)
      if type(elem) == dict: #nest
        handle.write("<%s>\n" % key)
        self.printDict(elem, handle, indentLevel +1, isEnv=True)
        handle.write("  " * indentLevel)
        handle.write("</%s>" % key)
      else: #bottom level, just write the value of the key
        if type(elem) == str or type(elem) == str:
          elem = elem.encode('UTF-8')
        #if isinstance(elem, str):
        #  elem = elem.replace(u"\u001B", "")
        try:
          for badch in "<>&": #make sure anything with these is escaped, so XML doesn't screw up
            if badch in str(elem) and not str(elem).startswith("<![CDATA"):
              elem = "<![CDATA[\n%s\n]]>" % elem
            for c in str(elem):
              if ord(c) > 128:
                raise UnicodeDecodeError
          handle.write("<%(key)s>%(elem)s</%(key)s>" % {
            "key": xml.sax.saxutils.escape(key),
            "elem": elem
            })
        except UnicodeDecodeError as e:
            handle.write("<%(key)s>[--- unicode decoding problem ---]</%(key)s>" % {"key": key})
      handle.write("\n") 
  def printXML(self, file, indentLevel = 1):
    self.printDict(self._standardDict, file, indentLevel); 
[docs]def PrintMachineInfoToFile(file, indentLevel = 1):
  info = HardwareInfo()
  info.printXML(file, indentLevel)