# coding=utf-8
from email.mime.text import MIMEText
from email.header import Header
from functools import wraps
import smtplib
import psutil
import time
import json
# 邮件配置
SMTP_SERVER = ''
SMTP_USER = ''
SMTP_PASSWD = ''
RECEIVER_LIST = ['superhin@126.com','hanzhichao@spicespirit.com']
EMAIL_SUBJECT = '服务器检查邮件报警!!!'
EMAIL_TPL = ''
# CPU/MEMORY 报警百分比
CPU_WARN_PERCENT = 70
MEM_WARN_PERCENT = 70
# php-fpm 单线程 报价值(M)
RES_WARN = 128
# 日志文件,%s会替换成日期
TODAY = time.strftime('%Y-%m-%d',time.localtime(time.time()))
LOG_FILE = '/var/log/health_check_%s.log' % TODAY
# 用于显示执行时间的装饰器
def _show_time(func):
@wraps(func)
def wrapper(*args, **kwargs):
print("start %s ... ..." % func.__name__)
start_time = time.time()
func_result = func(*args, **kwargs)
print("%s 执行时间:%.3fs" % (func.__name__, time.time()-start_time))
return func_result
return wrapper
# 用于将返回列表整理成字典的装饰器
def _collect_result(func):
@wraps(func)
def wrapper(*args, **kwargs):
func_result = func(*args, **kwargs)
return {func.__name__: {'status': 'OK', 'process_list': []}} if not func_result else {func.__name__: {'status': 'WARN', 'process_list': func_result}}
return wrapper
@_show_time
@_collect_result
def check_cpu():
if psutil.cpu_percent(0.01) >= int(CPU_WARN_PERCENT):
return get_top_process('cpu', 10, True)
else:
return []
@_show_time
@_collect_result
def check_mem():
if psutil.virtual_memory().percent >= int(MEM_WARN_PERCENT):
return get_top_process('mem', 10, True)
else:
return []
# 获取mem/cpu占用最高的n个进程
def get_top_process(option='mem',n=10,added=True):
process_list = []
for proc in psutil.process_iter():
process_list.append((proc.name(), proc.memory_percent(), proc.cpu_percent(0.01)))
process_list.sort(key=lambda process_list:process_list[0]) # 按进程名排序
if added:
# 累加相同进程名进程资源占用百分比
process_list = reduce(
lambda x,y: x+[y] if x==[] or x[-1][0]!=y[0] else x[0:-1]+[(x[-1][0],x[-1][1]+y[1],x[-1][2]+y[2])],
[[]] + process_list)
sort_index = 1 if option.lower() == 'mem' else 2 # 排序列,option=mem,按prcess_list第2列(index=1)排序
# 将process_list按相应列从大到小排序,无累加-------# todo 先累加再排序
process_list.sort(key=lambda process_list:process_list[2 if sort_index==1 else 1], reverse=True)
process_list.sort(key=lambda process_list:process_list[sort_index], reverse=True)
return process_list[0:n]
@_show_time
@_collect_result
def check_zombie_process():
zombie_process_list = []
for proc in psutil.process_iter():
if proc.status() == 'zombie':
zombie_process_list.append(
(proc.name(), proc.status(), proc.create_time(), proc.memory_percent(), proc.cpu_percent(0.01))
)
return zombie_process_list
@_show_time
@_collect_result
def check_single_process(process_name='php-fpm'):
warn_process_list = []
for proc in psutil.process_iter():
if proc.name() == process_name:
if proc.memory_info().res()/1024/1024 >= int(RES_WARN):
warn_process_list.append(proc.name(), proc.create_time(), proc.memory_info().res(), proc.connections())
return warn_process_list
@_show_time
def write_log(check_result):
now= time.strftime('%Y-%m-%d %H:%M:%S',time.localtime(time.time()))
try:
with open(LOG_FILE,'ab') as f:
f.writelines(now + ' ' + check_result)
except IOError:
print("写日志文件失败!")
@_show_time
def send_mail(subject=EMAIL_SUBJECT, content=EMAIL_TPL):
mail_body = content
msg = MIMEText(mail_body, 'html', 'utf-8')
msg['Subject'] = Header(subject, 'utf-8')
msg['From'] = SMTP_USER
msg['To'] = ','.join(RECEIVER_LIST)
smtp = smtplib.SMTP_SSL()
smtp.connect(SMTP_SERVER,994)
smtp.login(SMTP_USER, SMTP_PASSWD)
for reserver in RECEIVER_LIST:
smtp.sendmail(SMTP_USER, reserver, msg.as_string())
smtp.quit()
print('Email has send out!')
if __name__ == '__main__':
print("主程序开始")
start_time = time.time()
check_result = reduce(
lambda x,y: dict(x, **y),
[check_cpu(), check_mem(), check_zombie_process(), check_single_process()]
)
for func, func_result in check_result.items():
if func_result['status'] != 'OK':
send_mail(func + '报警',json.dumps(func_result))
break
write_log(json.dumps(check_result))
print("总程序执行时间:%.3fs" % (time.time()-start_time))