python3 監控腳本

週末在家花了3個晚上重寫了以前4個監控腳本,記錄下python

用的是mysql數據庫 還在強化學習mysql

企業微信監控效果:
python3 監控腳本
郵件監控效果:
python3 監控腳本
增長了郵件告警,每臺機監控的閾值能夠自定義
使用paramiko登入每臺機收集CPU 內存 磁盤 網卡 併發鏈接數ios

# -*- coding: utf-8 -*-

import paramiko
import pymysql
import time
import requests
import json
import sys
#郵件告警
import smtplib
from email.mime.text import MIMEText
from email.header import Header

def msg(text):
    #企業微信告警,wx表格控制,能夠發送給不一樣的機器人,initiative控制是否@收件人,0關閉 1開啓 默認關閉
    if wx != '':
        cursor.execute(f"select * from wx WHERE wx.name = '{wx}'")
        wx_row = cursor.fetchone()
        headers = {'Content-Type': 'application/json;charset=utf-8'}
        api_url = wx_row[1]
        if wx_row[2] == 0:
            wx_phone = ''
        else:
            wx_phone = phone
        json_text = {
            "msgtype": "text",
            "text": {
                "content": text,
                "mentioned_mobile_list": [wx_phone]
            },
        }
        requests.post(api_url, json.dumps(json_text), headers=headers).content
    #郵件告警 須要配置好smtp發件代理,info表格填入郵件收件人,填入開啓
    if mail != '':
        send_mail(text)

def send_mail(text):
    # 調用QQ第三方 SMTP 服務
    mail_host = "smtp.qq.com"  # 設置服務器
    mail_user = "450433231@qq.com"  # 用戶名
    mail_pass = ""  # 受權碼 SMTP代理提供
    sender = '450433231@qq.com' #發件郵箱
    receivers = [f"{mail}"]  # 接收郵箱
    message = MIMEText(f'{text}', 'plain', 'utf-8')  # 正文
    message['From'] = Header("監控", 'utf-8')  # 發件人
    message['To'] = Header(f"{mail}", 'utf-8')  # 收件人
    subject = '【重要告警】'  # 主題
    message['Subject'] = Header(subject, 'utf-8')
    try:
        smtpObj = smtplib.SMTP()
        smtpObj.connect(mail_host, 25)  # 25 爲 SMTP 端口號
        smtpObj.login(mail_user, mail_pass)
        smtpObj.sendmail(sender, receivers, message.as_string())
    except smtplib.SMTPException as f:
        out_log(f"{mail} 沒法發送郵件", f)

def down_time(ip, table):
    #計算故障時間
    cursor.execute(f"select {table}.start_time,{table}.end_time from {table} WHERE ip = '{ip}'")
    cpu_time = cursor.fetchone()
    try:
        start_time = cpu_time[0]
        end_time = cpu_time[1]
        duration = end_time - start_time
        return f"\n開始時間 {start_time} 結束時間 {end_time} 持續時間 {duration}"
    except:
        return "\n未記錄時間"

def out_log(text):
    #寫日誌
    with open(f'{sys.argv[0].split(".")[0]}.log', mode='a') as f:
        print(text, file=f)

class Check():
    def __init__(self, switch, threshold, client):
        self.switch = switch
        self.threshold = threshold
        self.client = client

    def cpu(self):
        if self.switch[1] == 1:
            stdin, stdout, stderr = self.client.exec_command(
                r"""iostat -x|awk NR==4'{printf(100-$6)}'""")
            cpu_result = stdout.read().decode()
            cursor.execute(f"select cpu.send from cpu WHERE ip = '{ip}'")
            send = cursor.fetchone()
            if float(cpu_result) > self.threshold[1] and send[0] == 0:
                cursor.execute(f"UPDATE cpu SET send = 1,start_time = NOW() WHERE ip = '{ip}'")
                msg(f'{cur_time} {ip} {system_info} CPU使用率:{cpu_result}% 請檢查')
            elif float(cpu_result) < self.threshold[1] and send[0] == 1:
                cursor.execute(f"UPDATE cpu SET send = 0,end_time = NOW() WHERE ip = '{ip}'")
                msg(f'{cur_time} {ip} {system_info} CPU使用率:{cpu_result}% 已恢復{down_time(ip, "cpu")}')
            return cpu_result

    def mem(self):
        if self.switch[2] == 1:
            stdin, stdout, stderr = self.client.exec_command(
                r"""free|awk '/Mem:/{printf("%.2f",($2-$4)/$2*100)}'""")
            mem_result = stdout.read().decode()
            stdin, stdout, stderr = self.client.exec_command(
                r"""free|awk '/Swap:/{if($2==0){printf$2}else{printf("%.2f",$3/$2*100)}}'""")
            swap_result = stdout.read().decode()
            cursor.execute(f"select mem.send from mem WHERE ip = '{ip}'")
            send = cursor.fetchone()
            if float(mem_result) > self.threshold[2] and send[0] == 0:
                cursor.execute(f"UPDATE mem SET send = 1,start_time = NOW() WHERE ip = '{ip}'")
                msg(f'{cur_time} {ip} {system_info} 內存使用率:{mem_result}% swap使用率:{swap_result}% 請檢查')
            elif float(mem_result) < self.threshold[2] and send[0] == 1:
                cursor.execute(f"UPDATE mem SET send = 0,end_time = NOW() WHERE ip = '{ip}'")
                msg(
                    f'{cur_time} {ip} {system_info} 內存使用率:{mem_result}% swap使用率:{swap_result}% 已恢復{down_time(ip, "mem")}')
            return mem_result

    def disk(self):
        if self.switch[3] == 1:
            stdin, stdout, stderr = self.client.exec_command(
                r"""df -P|awk  -F "%" 'NR>1{print$1}'|awk '{print$1,$5}'|grep -v '/dev/sr0'""")
            disk_result = stdout.read().decode()
            cursor.execute(f"select disk.send from disk WHERE ip = '{ip}'")
            send = cursor.fetchone()
            disk_use_sum = []
            for disk in disk_result.split('\n'):
                disk = disk.split(' ')
                disk_name = disk[0]
                if disk_name == '':
                    break
                disk_use = disk[1]
                disk_use_sum.append(int(disk_use))
                if disk_use > str(self.threshold[3]) and send[0] == 0:
                    cursor.execute(f"UPDATE disk SET send = 1,start_time = NOW() WHERE ip = '{ip}'")
                    msg(f'{cur_time} {ip} {system_info} 磁盤:{disk_name} 使用率:{disk_use}% 請檢查')
            return sum(disk_use_sum)

    def speed(self):
        if self.switch[4] == 1:
            stdin, stdout, stderr = self.client.exec_command(
                r"""export LANG="en_US.UTF-8";sar -n DEV 1 1|grep "Average:"|grep "$(ifconfig |awk -F ":" '/flags/{print$1}'|egrep -v "lo|veth|docker")"|awk '{printf$5+$6}'""")
            speed_result = stdout.read().decode()
            cursor.execute(f"select speed.send from speed WHERE ip = '{ip}'")
            send = cursor.fetchone()
            if float(speed_result) > float(self.threshold[4]) and send[0] == 0:
                cursor.execute(f"UPDATE speed SET send = 1,start_time = NOW() WHERE ip = '{ip}'")
                msg(f'{cur_time} {ip} {system_info} 網卡速率:{speed_result}kB/s 請檢查')
            elif float(speed_result) < float(self.threshold[4]) and send[0] == 1:
                cursor.execute(f"UPDATE speed SET send = 0,end_time = NOW() WHERE ip = '{ip}'")
                #msg(f'{cur_time} {ip} {system_info} 網卡速率:{speed_result}kB/s 已恢復{down_time(ip, "speed")}')
            return speed_result

    def connect(self):
        if self.switch[5] == 1 and port != None:
            stdin, stdout, stderr = self.client.exec_command(
                r"""ss -ant|grep "ESTAB"|grep "%s"|wc -l""" % port)
            connect_result = stdout.read().decode().replace("\n", "")
            stdin, stdout, stderr = self.client.exec_command(
                r"""ss -ant|wc -l""")
            connect_all_result = stdout.read().decode().replace("\n", "")
            cursor.execute(f"select connect.send from connect WHERE ip = '{ip}'")
            send = cursor.fetchone()
            if int(connect_result) > self.threshold[5] and send[0] == 0:
                cursor.execute(f"UPDATE connect SET send = 1,start_time = NOW() WHERE ip = '{ip}'")
                msg(
                    f'{cur_time} {ip} {system_info} ESTABLISHED併發鏈接數:{connect_result} 全狀態鏈接數:{connect_all_result} 請檢查')
            elif int(connect_result) < self.threshold[5] and send[0] == 1:
                cursor.execute(f"UPDATE connect SET send = 0,end_time = NOW() WHERE ip = '{ip}'")
                #msg(f'{cur_time} {ip} {system_info} ESTABLISHED併發鏈接數:{connect_result} 全狀態鏈接數:{connect_all_result} 已恢復{down_time(ip, "connect")}')
            return connect_result, connect_all_result

db = pymysql.connect("127.0.0.1", user="root", passwd="root.123", db="monitor")
cursor = db.cursor()
# 數據表同步
cursor.execute("INSERT IGNORE INTO switch(ip) SELECT info.ip FROM info")
cursor.execute("INSERT IGNORE INTO threshold(ip) SELECT info.ip FROM info")
cursor.execute("INSERT IGNORE INTO cpu(ip) SELECT info.ip FROM info")
cursor.execute("INSERT IGNORE INTO mem(ip) SELECT info.ip FROM info")
cursor.execute("INSERT IGNORE INTO disk(ip) SELECT info.ip FROM info")
cursor.execute("INSERT IGNORE INTO speed(ip) SELECT info.ip FROM info")
cursor.execute("INSERT IGNORE INTO connect(ip) SELECT info.ip FROM info")
db.commit()
# 獲取須要監控的IP地址等信息
cursor.execute("select * from info")
info_table = cursor.fetchall()

for info_row in info_table:
    ip = info_row[0]
    system_info = info_row[1]
    phone = info_row[2]
    monitor = info_row[3]
    port = info_row[4]
    user = info_row[5]
    password = info_row[6]
    remote_port = info_row[7]
    mail = info_row[8]
    wx = info_row[9]
    # 是否開啓監控 表格info.open控制,0關閉 1開啓 默認開啓
    if monitor != 1:
        continue
    try:
        client = paramiko.client.SSHClient()
        client.set_missing_host_key_policy(paramiko.client.AutoAddPolicy())
        client.connect(ip, port=remote_port, username=user,
                       password=password, timeout=3)

    except:
        out_log(f"{ip} 鏈接失敗\n")
    #表格switch控制監控項是否開啓 0關閉 1開啓 默認開啓
    cursor.execute(f"select * from switch where ip = '{ip}'")
    switch_row = cursor.fetchone()
    #表格threshold設置不一樣的監控閾值 默認CPU 80 內存90 磁盤90 網卡60M 鏈接數800
    cursor.execute(f"select * from threshold where ip = '{ip}'")
    threshold_row = cursor.fetchone()
    cur_time = time.strftime("%Y-%m-%d %H:%M:%S", time.localtime())
    #開始登入巡檢
    check = Check(switch_row, threshold_row, client)
    cpu_use = check.cpu()
    mem_use = check.mem()
    disk_all_use = check.disk()
    speed_use = check.speed()
    coonect_use = check.connect()
    client.close()
    db.commit()
    out_log(
        f"{cur_time} {ip} CPU使用率:{cpu_use}% 內存使用率:{mem_use}% 磁盤總0使用率:{disk_all_use}% 網卡速率:{speed_use}kB/s 端口{port} ESTABLISHED併發鏈接數{coonect_use[0]} 全狀態鏈接數:{coonect_use[1]}")
db.commit()
db.close()

附上建庫sqlweb

SET FOREIGN_KEY_CHECKS=0;

-- ----------------------------
-- Table structure for `connect`
-- ----------------------------
DROP TABLE IF EXISTS `connect`;
CREATE TABLE `connect` (
  `ip` varchar(64) NOT NULL DEFAULT '',
  `send` int(10) DEFAULT '0',
  `start_time` datetime DEFAULT NULL,
  `end_time` datetime DEFAULT NULL,
  PRIMARY KEY (`ip`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

-- ----------------------------
-- Records of connect
-- ----------------------------
INSERT INTO `connect` VALUES ('127.0.0.1', '0', null, null);

-- ----------------------------
-- Table structure for `cpu`
-- ----------------------------
DROP TABLE IF EXISTS `cpu`;
CREATE TABLE `cpu` (
  `ip` varchar(64) NOT NULL DEFAULT '',
  `send` int(10) DEFAULT '0',
  `start_time` datetime DEFAULT NULL,
  `end_time` datetime DEFAULT NULL,
  PRIMARY KEY (`ip`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

-- ----------------------------
-- Records of cpu
-- ----------------------------
INSERT INTO `cpu` VALUES ('127.0.0.1', '0', null, null);

-- ----------------------------
-- Table structure for `disk`
-- ----------------------------
DROP TABLE IF EXISTS `disk`;
CREATE TABLE `disk` (
  `ip` varchar(64) NOT NULL DEFAULT '',
  `send` int(10) DEFAULT '0',
  `start_time` datetime DEFAULT NULL,
  `end_time` datetime DEFAULT NULL,
  PRIMARY KEY (`ip`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

-- ----------------------------
-- Records of disk
-- ----------------------------
INSERT INTO `disk` VALUES ('127.0.0.1', '0', null, null);

-- ----------------------------
-- Table structure for `info`
-- ----------------------------
DROP TABLE IF EXISTS `info`;
CREATE TABLE `info` (
  `ip` varchar(64) NOT NULL,
  `system_info` varchar(255) DEFAULT NULL,
  `phone` varchar(64) DEFAULT NULL,
  `open` int(1) NOT NULL DEFAULT '1',
  `port` int(10) DEFAULT NULL,
  `user` varchar(64) NOT NULL,
  `password` varchar(64) NOT NULL,
  `remote_port` int(10) NOT NULL,
  `mail` varchar(64) DEFAULT NULL,
  `wx` varchar(64) DEFAULT NULL,
  PRIMARY KEY (`ip`,`remote_port`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

-- ----------------------------
-- Records of info
-- ----------------------------
INSERT INTO `info` VALUES ('127.0.0.1', '監控系統名', null, '1', null, 'centos', 'password', '22', null, 'default');

-- ----------------------------
-- Table structure for `mem`
-- ----------------------------
DROP TABLE IF EXISTS `mem`;
CREATE TABLE `mem` (
  `ip` varchar(64) NOT NULL DEFAULT '',
  `send` int(10) DEFAULT '0',
  `start_time` datetime DEFAULT NULL,
  `end_time` datetime DEFAULT NULL,
  PRIMARY KEY (`ip`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

-- ----------------------------
-- Records of mem
-- ----------------------------
INSERT INTO `mem` VALUES ('127.0.0.1', '0', null, null);

-- ----------------------------
-- Table structure for `speed`
-- ----------------------------
DROP TABLE IF EXISTS `speed`;
CREATE TABLE `speed` (
  `ip` varchar(64) NOT NULL DEFAULT '',
  `send` int(10) DEFAULT '0',
  `start_time` datetime DEFAULT NULL,
  `end_time` datetime DEFAULT NULL,
  PRIMARY KEY (`ip`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

-- ----------------------------
-- Records of speed
-- ----------------------------
INSERT INTO `speed` VALUES ('127.0.0.1', '0', null, null);

-- ----------------------------
-- Table structure for `switch`
-- ----------------------------
DROP TABLE IF EXISTS `switch`;
CREATE TABLE `switch` (
  `ip` varchar(64) NOT NULL DEFAULT '',
  `cpu` int(10) DEFAULT '1',
  `mem` int(10) DEFAULT '1',
  `disk` int(10) DEFAULT '1',
  `speed` int(10) DEFAULT '1',
  `connect` int(10) DEFAULT '1',
  PRIMARY KEY (`ip`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

-- ----------------------------
-- Records of switch
-- ----------------------------
INSERT INTO `switch` VALUES ('127.0.0.1', '1', '1', '1', '1', '1');

-- ----------------------------
-- Table structure for `threshold`
-- ----------------------------
DROP TABLE IF EXISTS `threshold`;
CREATE TABLE `threshold` (
  `ip` varchar(64) NOT NULL,
  `cpu` int(10) NOT NULL DEFAULT '80',
  `mem` int(10) NOT NULL DEFAULT '90',
  `disk` int(10) NOT NULL DEFAULT '90',
  `speed` int(10) NOT NULL DEFAULT '1200000',
  `connect` int(10) NOT NULL DEFAULT '800',
  PRIMARY KEY (`ip`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

-- ----------------------------
-- Records of threshold
-- ----------------------------
INSERT INTO `threshold` VALUES ('127.0.0.1', '80', '90', '90', '1200000', '800');

-- ----------------------------
-- Table structure for `wx`
-- ----------------------------
DROP TABLE IF EXISTS `wx`;
CREATE TABLE `wx` (
  `name` varchar(64) NOT NULL,
  `webhook` varchar(255) NOT NULL,
  `initiative` int(10) NOT NULL DEFAULT '0',
  PRIMARY KEY (`name`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

-- ----------------------------
-- Records of wx
-- ----------------------------
INSERT INTO `wx` VALUES ('default', '', '0');
相關文章
相關標籤/搜索