# -*- coding: utf-8 -*-
"""
sysinfo.py [host1] [host2] [options]

obtain system info from remote machine.

(c) Holger Krekel, MIT license
"""
import optparse
import re
import sys

import execnet
import py


parser = optparse.OptionParser(usage=__doc__)
parser.add_option(
    "-f",
    "--sshconfig",
    action="store",
    dest="ssh_config",
    default=None,
    help="use given ssh config file,"
    " and add info all contained hosts for getting info",
)
parser.add_option(
    "-i",
    "--ignore",
    action="store",
    dest="ignores",
    default=None,
    help="ignore hosts " "(useful if the list of hostnames come from a file list)",
)


def parsehosts(path):
    path = py.path.local(path)
    l = []
    rex = re.compile(r"Host\s*(\S+)")
    for line in path.readlines():
        m = rex.match(line)
        if m is not None:
            (sshname,) = m.groups()
            l.append(sshname)
    return l


class RemoteInfo:
    def __init__(self, gateway):
        self.gw = gateway
        self._cache = {}

    def exreceive(self, execstring):
        if execstring not in self._cache:
            channel = self.gw.remote_exec(execstring)
            self._cache[execstring] = channel.receive()
        return self._cache[execstring]

    def getmodattr(self, modpath):
        module = modpath.split(".")[0]
        return self.exreceive(
            """
            import %s
            channel.send(%s)
        """
            % (module, modpath)
        )

    def islinux(self):
        return self.getmodattr("sys.platform").find("linux") != -1

    def getfqdn(self):
        return self.exreceive(
            """
            import socket
            channel.send(socket.getfqdn())
        """
        )

    def getmemswap(self):
        if self.islinux():
            return self.exreceive(
                r"""
            import commands, re
            out = commands.getoutput("free")
            mem = re.search(r"Mem:\s+(\S*)", out).group(1)
            swap = re.search(r"Swap:\s+(\S*)", out).group(1)
            channel.send((mem, swap))
            """
            )

    def getcpuinfo(self):
        if self.islinux():
            return self.exreceive(
                """
                # a hyperthreaded cpu core only counts as 1, although it
                # is present as 2 in /proc/cpuinfo.  Counting it as 2 is
                # misleading because it is *by far* not as efficient as
                # two independent cores.
                cpus = {}
                cpuinfo = {}
                f = open("/proc/cpuinfo")
                lines = f.readlines()
                f.close()
                for line in lines + ['']:
                    if line.strip():
                        key, value = line.split(":", 1)
                        cpuinfo[key.strip()] = value.strip()
                    else:
                        corekey = (cpuinfo.get("physical id"),
                                   cpuinfo.get("core id"))
                        cpus[corekey] = 1
                numcpus = len(cpus)
                model = cpuinfo.get("model name")
                channel.send((numcpus, model))
            """
            )


def debug(*args):
    print >>sys.stderr, " ".join(map(str, args))


def error(*args):
    debug("ERROR", args[0] + ":", *args[1:])


def getinfo(sshname, ssh_config=None, loginfo=sys.stdout):
    if ssh_config:
        spec = "ssh=-F {} {}".format(ssh_config, sshname)
    else:
        spec += "ssh=%s" % sshname
    debug("connecting to", repr(spec))
    try:
        gw = execnet.makegateway(spec)
    except IOError:
        error("could not get sshgatway", sshname)
    else:
        ri = RemoteInfo(gw)
        # print "%s info:" % sshname
        prefix = sshname.upper() + " "
        print >> loginfo, prefix, "fqdn:", ri.getfqdn()
        for attr in ("sys.platform", "sys.version_info"):
            loginfo.write("{} {}: ".format(prefix, attr))
            loginfo.flush()
            value = ri.getmodattr(attr)
            loginfo.write(str(value))
            loginfo.write("\n")
            loginfo.flush()
        memswap = ri.getmemswap()
        if memswap:
            mem, swap = memswap
            print >> loginfo, prefix, "Memory:", mem, "Swap:", swap
        cpuinfo = ri.getcpuinfo()
        if cpuinfo:
            numcpu, model = cpuinfo
            print >> loginfo, prefix, "number of cpus:", numcpu
            print >> loginfo, prefix, "cpu model", model
        return ri


if __name__ == "__main__":
    options, args = parser.parse_args()
    hosts = list(args)
    ssh_config = options.ssh_config
    if ssh_config:
        hosts.extend(parsehosts(ssh_config))
    ignores = options.ignores or ()
    if ignores:
        ignores = ignores.split(",")
    for host in hosts:
        if host not in ignores:
            getinfo(host, ssh_config=ssh_config)
