CMDB-Linux下收集数据(五)

Linux下收集数据就有很多命令和工具了,比Windows方便多了。

但是要在Python的进程中运行操作系统级别的命令,我们通常需要使用subprocess模块。这个模块的具体用法,请查看Python教程中相关部分的内容。

下面,我们在Client/plugins下创建一个linux包,再到包里创建一个sys_info.py文件,写入下面的代码:

前提需要现在被收集的虚机上面安装必须的组件:

yum install -y lsb
关于disk的获取与原著有差别,更容易理解:
##获取厂商:
[root@python_master pythontest]# dmidecode -s system-manufacturer
VMware, Inc.

##获取型号:
[root@python_master pythontest]# dmidecode -s system-product-name
VMware Virtual Platform

##获取sn:
[root@python_master pythontest]# dmidecode -s system-serial-number
VMware-56 4d 25 68 5c ee dc fb-a1 5e 77 6a 5f e7 66 60

#!/usr/bin/env python
# -*- coding: utf-8 -*-
# @Time    : 2019-3-3 13:16
# @Author  : zhdya@zhdya.cn
# @File    : sys_info.py

import subprocess


def collect():
    filter_keys = ['Manufacturer', 'Serial Number', 'Product Name', 'UUID', 'Wake-up Type']
    raw_data = {}

    for key in filter_keys:
        try:
            res = subprocess.Popen("sudo dmidecode -t system|grep '%s'" % key,
                                   stdout=subprocess.PIPE, shell=True)
            result = res.stdout.read().decode()
            data_list = result.split(':')

            if len(data_list) > 1:
                raw_data[key] = data_list[1].strip()
            else:
                raw_data[key] = -1
        except Exception as e:
            print(e)
            raw_data[key] = -2

    data = dict()
    data['asset_type'] = 'server'
    data['manufacturer'] = raw_data['Manufacturer']
    data['sn'] = raw_data['Serial Number']
    data['model'] = raw_data['Product Name']
    data['uuid'] = raw_data['UUID']
    data['wake_up_type'] = raw_data['Wake-up Type']

    data.update(get_os_info())
    data.update(get_cpu_info())
    data.update(get_ram_info())
    data.update(get_nic_info())
    data.update(get_disk_info())
    return data


def get_os_info():
    """
    获取操作系统信息
    :return:
    """
    distributor = subprocess.Popen("lsb_release -a|grep 'Distributor ID'",
                                   stdout=subprocess.PIPE, shell=True)
    distributor = distributor.stdout.read().decode().split(":")

    release = subprocess.Popen("lsb_release -a|grep 'Description'",
                               stdout=subprocess.PIPE, shell=True)

    release = release.stdout.read().decode().split(":")
    data_dic = {
        "os_distribution": distributor[1].strip() if len(distributor) > 1 else "",
        "os_release": release[1].strip() if len(release) > 1 else "",
        "os_type": "Linux",
    }
    return data_dic


def get_cpu_info():
    """
    获取cpu信息
    :return:
    """
    base_cmd = 'cat /proc/cpuinfo'

    raw_data = {
        'cpu_model': "%s |grep 'model name' |head -1 " % base_cmd,
        'cpu_count':  "%s |grep  'processor'|wc -l " % base_cmd,
        'cpu_core_count': "%s |grep 'cpu cores' |awk -F: '{SUM +=$2} END {print SUM}'" % base_cmd,
    }

    for key, cmd in raw_data.items():
        try:
            cmd_res = subprocess.Popen(cmd, stdout=subprocess.PIPE, shell=True)
            raw_data[key] = cmd_res.stdout.read().decode().strip()
        except ValueError as e:
            print(e)
            raw_data[key] = ""

    data = {
        "cpu_count": raw_data["cpu_count"],
        "cpu_core_count": raw_data["cpu_core_count"]
        }

    cpu_model = raw_data["cpu_model"].split(":")

    if len(cpu_model) > 1:
        data["cpu_model"] = cpu_model[1].strip()
    else:
        data["cpu_model"] = -1

    return data


def get_ram_info():
    """
    获取内存信息
    :return:
    """
    raw_data = subprocess.Popen("sudo dmidecode -t memory", stdout=subprocess.PIPE, shell=True)
    raw_list = raw_data.stdout.read().decode().split("\n")
    raw_ram_list = []
    item_list = []
    for line in raw_list:
        if line.startswith("Memory Device"):
            raw_ram_list.append(item_list)
            item_list = []
        else:
            item_list.append(line.strip())

    ram_list = []
    for item in raw_ram_list:
        item_ram_size = 0
        ram_item_to_dic = {}
        for i in item:
            data = i.split(":")
            if len(data) == 2:
                key, v = data
                if key == 'Size':
                    if v.strip() != "No Module Installed":
                        ram_item_to_dic['capacity'] = v.split()[0].strip()
                        item_ram_size = round(float(v.split()[0]))
                    else:
                        ram_item_to_dic['capacity'] = 0

                if key == 'Type':
                    ram_item_to_dic['model'] = v.strip()
                if key == 'Manufacturer':
                    ram_item_to_dic['manufacturer'] = v.strip()
                if key == 'Serial Number':
                    ram_item_to_dic['sn'] = v.strip()
                if key == 'Asset Tag':
                    ram_item_to_dic['asset_tag'] = v.strip()
                if key == 'Locator':
                    ram_item_to_dic['slot'] = v.strip()

        if item_ram_size == 0:
            pass
        else:
            ram_list.append(ram_item_to_dic)

    raw_total_size = subprocess.Popen("cat /proc/meminfo|grep MemTotal ", stdout=subprocess.PIPE, shell=True)
    raw_total_size = raw_total_size.stdout.read().decode().split(":")
    ram_data = {'ram': ram_list}
    if len(raw_total_size) == 2:
        total_gb_size = int(raw_total_size[1].split()[0]) / 1024**2
        ram_data['ram_size'] = total_gb_size

    return ram_data


def get_nic_info():
    """
    获取网卡信息
    :return:
    """
    raw_data = subprocess.Popen("ifconfig -a", stdout=subprocess.PIPE, shell=True)

    raw_data = raw_data.stdout.read().decode().split("\n")

    nic_dic = dict()
    next_ip_line = False
    last_mac_addr = None

    for line in raw_data:
        if next_ip_line:
            next_ip_line = False
            nic_name = last_mac_addr.split()[0]
            mac_addr = last_mac_addr.split("HWaddr")[1].strip()
            raw_ip_addr = line.split("inet addr:")
            raw_bcast = line.split("Bcast:")
            raw_netmask = line.split("Mask:")
            if len(raw_ip_addr) > 1:
                ip_addr = raw_ip_addr[1].split()[0]
                network = raw_bcast[1].split()[0]
                netmask = raw_netmask[1].split()[0]
            else:
                ip_addr = None
                network = None
                netmask = None
            if mac_addr not in nic_dic:
                nic_dic[mac_addr] = {'name': nic_name,
                                     'mac': mac_addr,
                                     'net_mask': netmask,
                                     'network': network,
                                     'bonding': 0,
                                     'model': 'unknown',
                                     'ip_address': ip_addr,
                                     }
            else:
                if '%s_bonding_addr' % (mac_addr,) not in nic_dic:
                    random_mac_addr = '%s_bonding_addr' % (mac_addr,)
                else:
                    random_mac_addr = '%s_bonding_addr2' % (mac_addr,)

                nic_dic[random_mac_addr] = {'name': nic_name,
                                            'mac': random_mac_addr,
                                            'net_mask': netmask,
                                            'network': network,
                                            'bonding': 1,
                                            'model': 'unknown',
                                            'ip_address': ip_addr,
                                            }

        if "HWaddr" in line:
            next_ip_line = True
            last_mac_addr = line
    nic_list = []
    for k, v in nic_dic.items():
        nic_list.append(v)

    return {'nic': nic_list}


def get_disk_info():
    """
    获取存储信息。
    本脚本只针对centos7.6中使用sda2,且只有一块虚拟硬盘的情况。
    具体查看硬盘信息的命令,请根据实际情况,实际调整。
    如果需要查看Raid信息,可以尝试MegaCli工具。
    :return:
    """
    sn_raw_data = subprocess.Popen("sudo dmidecode -s system-serial-number", stdout=subprocess.PIPE, shell=True)
    sn = sn_raw_data.stdout.read().decode()
    model_raw_data = subprocess.Popen("sudo dmidecode -s system-product-name", stdout=subprocess.PIPE, shell=True)
    model = model_raw_data.stdout.read().decode()

    #size_data = subprocess.Popen("sudo fdisk -l /dev/sda2 | grep Disk|head -1", stdout=subprocess.PIPE, shell=True)
    #size_data = size_data.stdout.read().decode()
    #size = size_data.split(":")[1].strip().split(" ")[0]

    size_raw_data = subprocess.Popen("sudo smartctl -a /dev/sda2 |grep Capacity", stdout=subprocess.PIPE, shell=True)
    raw_data = size_raw_data.stdout.read().decode()
    data_list = raw_data.split()[4]
    size = data_list.split('[')[1]


    result = {'physical_disk_driver': []}
    disk_dict = dict()
    disk_dict["model"] = model
    disk_dict["size"] = size
    disk_dict["sn"] = sn
    result['physical_disk_driver'].append(disk_dict)

    return result


if __name__ == "__main__":
    # 收集信息功能测试
    d = collect()
    print(d)

先来个输出在 centos7.6虚机上面的测试(可以读出所有数据):

{'asset_type': 'server', 'manufacturer': 'VMware, Inc.', 'sn': 'VMware-56 4d 25 68 5c ee dc fb-a1 5e 77 6a 5f e7 66 60', 'model': 'VMware Virtual Platform', 'uuid': '68254d56-ee5c-fbdc-a15e-776a5fe76660', 'wake_up_type': 'Power Switch', 'os_distribution': 'CentOS', 'os_release': 'CentOS Linux release 7.6.1810 (Core)', 'os_type': 'Linux', 'cpu_count': '1', 'cpu_core_count': '1', 'cpu_model': 'Intel(R) Core(TM) i5-7300HQ CPU @ 2.50GHz', 'ram': [{'capacity': '1024', 'slot': 'RAM slot #0', 'model': 'DRAM', 'manufacturer': 'Not Specified', 'sn': 'Not Specified', 'asset_tag': 'Not Specified'}], 'ram_size': 0.9497604370117188, 'nic': [], 'physical_disk_driver': [{'model': 'VMware Virtual Platform\n', 'size': '32.2', 'sn': 'VMware-56 4d 25 68 5c ee dc fb-a1 5e 77 6a 5f e7 66 60\n'}]}

代码整体没有什么难点,无非就是使用subprocess.Popen()方法执行Linux的命令,然后获取返回值,并以规定的格式打包到data字典里。

需要说明的问题有:

  • 当Linux中存在好几个Python解释器版本时,要注意调用方式,前面已经强调过了;
  • 不同的Linux发行版,有些命令可能没有,需要额外安装
  • 所使用的查看硬件信息的命令并不一定必须和这里的一样,只要能获得数据就行;
  • 有一些命令在ubuntu中涉及sudo的问题,需要特别对待;
  • 最终数据字典的格式一定要正确。
  • 可以在Linux下配置cronb或其它定时服务,设置定期的数据收集、报告任务。 下面,我们在Linux虚拟机上,测试一下客户端。

将Pycharm中的Client客户端文件夹,拷贝到Linux虚拟机中,我这里是centos7.6

进入bin目录,运行:

python3 main.py report_data
一切顺利的话应该能得到如下的反馈:
正在将数据发送至: [http://10.101.120.34:8000/assets/report/]  ......
handler_data_encode-->> b'asset_data=%7B%22asset_type%22%3A+%22server%22%2C+%22manufacturer%22%3A+%22VMware%2C+Inc.%22%2C+%22sn%22%3A+%22VMware-56+4d+25+68+5c+ee+dc+fb-a1+5e+77+6a+5f+e7+66+60%22%2C+%22model%22%3A+%22VMware+Virtual+Platform%22%2C+%22uuid%22%3A+%2268254d56-ee5c-fbdc-a15e-776a5fe76660%22%2C+%22wake_up_type%22%3A+%22Power+Switch%22%2C+%22os_distribution%22%3A+%22CentOS%22%2C+%22os_release%22%3A+%22CentOS+Linux+release+7.6.1810+%28Core%29%22%2C+%22os_type%22%3A+%22Linux%22%2C+%22cpu_count%22%3A+%221%22%2C+%22cpu_core_count%22%3A+%221%22%2C+%22cpu_model%22%3A+%22Intel%28R%29+Core%28TM%29+i5-7300HQ+CPU+%40+2.50GHz%22%2C+%22ram%22%3A+%5B%7B%22capacity%22%3A+%221024%22%2C+%22slot%22%3A+%22RAM+slot+%230%22%2C+%22model%22%3A+%22DRAM%22%2C+%22manufacturer%22%3A+%22Not+Specified%22%2C+%22sn%22%3A+%22Not+Specified%22%2C+%22asset_tag%22%3A+%22Not+Specified%22%7D%5D%2C+%22ram_size%22%3A+0.9497604370117188%2C+%22nic%22%3A+%5B%5D%2C+%22physical_disk_driver%22%3A+%5B%7B%22model%22%3A+%22VMware+Virtual+Platform%5Cn%22%2C+%22size%22%3A+%2232.2%22%2C+%22sn%22%3A+%22VMware-56+4d+25+68+5c+ee+dc+fb-a1+5e+77+6a+5f+e7+66+60%5Cn%22%7D%5D%7D'
发送完毕!
返回结果:成功收到数据!
日志记录成功!

然后我们可以在pycharm 页面收到:

mark

规整如下:

mark