钉钉告警
global:
resolve_timeout: 10m
route:
group_by: ['alertname']
group_wait: 10s
group_interval: 10s
repeat_interval: 5m
receiver: 'webhook1'
receivers:
- name: 'webhook1' # 这里对应 prometheus-webhook-dingtalk 这个配置文件里
webhook_configs:
- url: 'http://x.x.x.x:8060/dingtalk/webhook1/send'
- url: 'http://x.x.x.x:8066/wechat/webhook1/send'
AI 代码解读
#!/usr/bin/env python
# -*- coding: utf-8 -*-
'''
实现调用钉钉机器人接口发送告警信息到IT群
'''
import time
import requests
import json
import sys
import logging
import re
import hmac
import hashlib
import base64
import urllib.parse
from flask import Flask, request, jsonify
app = Flask(__name__)
class CheckJSON():
def getkeys(self, data):
keys_all_list = []
def getkeys(data):
if (type(data) == type({})):
keys = data.keys()
for key in keys:
value = data.get(key)
if (type(value) != type({}) and type(value) != type([])):
keys_all_list.append(key)
elif (type(value) == type({})):
keys_all_list.append(key)
getkeys(value)
elif (type(value) == type([])):
keys_all_list.append(key)
for para in value:
if (type(para) == type({})
or type(para) == type([])):
getkeys(para)
else:
keys_all_list.append(para)
getkeys(data)
return keys_all_list
def is_exists(self, data, tagkey):
if (type(data) != type({})):
print('Please input a json!')
else:
key_list = self.getkeys(data)
for key in key_list:
if (key == tagkey):
return True
return False
def get_timestamp_sign(secret):
timestamp = str(round(time.time() * 1000))
#secret = "SEC767ff0761242961b5be36e26b7525f0ff7261be4b90b9639ce795b4cdd11c423"
secret_enc = secret.encode('utf-8')
string_to_sign = '{}\n{}'.format(timestamp, secret)
string_to_sign_enc = string_to_sign.encode('utf-8')
hmac_code = hmac.new(secret_enc, string_to_sign_enc,
digestmod=hashlib.sha256).digest()
sign = urllib.parse.quote_plus(base64.b64encode(hmac_code))
#print("timestamp: ", timestamp)
#print("sign:", sign)
return (timestamp, sign)
def get_signed_url(webhook, secret):
timestamp, sign = get_timestamp_sign(secret)
webhook = webhook + "×tamp=" + timestamp + "&sign=" + sign
return webhook
def get_webhook(webhook, secret, mode):
if mode == 0: # only 敏感字
webhook = webhook
elif mode == 1 or mode ==2 : # 敏感字和加签 或 # 敏感字+加签+ip
webhook = get_signed_url(webhook, secret)
else:
webhook = ""
print("error! mode: ", mode ," webhook : ", webhook)
return webhook
def verify_datetime(datetime_):
pattern = r'((?!0000)[0-9]{4}-((0[1-9]|1[0-2])-(0[1-9]|1[0-9]|2[0-8])|(0[13-9]|1[0-2])-(29|30)|(0[13578]|1[02])-31)|([0-9]{2}(0[48]|[2468][048]|[13579][26])|(0[48]|[2468][048]|[13579][26])00)-02-29) (20|21|22|23|[0-1]\d):[0-5]\d:[0-5]\d$'
if re.match(pattern, datetime_):
return True
return False
def format_date_tz(raw_date):
date = raw_date.replace('T', ' ').replace('Z', '').split('.')[0]
if not verify_datetime(date):
raise ValueError(raw_date, date)
return date
def trim(docstring):
if not docstring:
return ''
# Convert tabs to spaces (following the normal Python rules)
# and split into a list of lines:
lines = docstring.expandtabs().splitlines()
# Determine minimum indentation (first line doesn't count):
indent = sys.maxsize
for line in lines[1:]:
stripped = line.lstrip()
if stripped:
indent = min(indent, len(line) - len(stripped))
# Remove indentation (first line is special):
trimmed = [lines[0].strip()]
if indent < sys.maxsize:
for line in lines[1:]:
trimmed.append(line[indent:].rstrip())
# Strip off trailing and leading blank lines:
while trimmed and not trimmed[-1]:
trimmed.pop()
while trimmed and not trimmed[0]:
trimmed.pop(0)
# Return a single string:
return '\n'.join(trimmed)
def get_message():
check_json = CheckJSON()
try:
if not check_json.is_exists(json.loads(request.data), 'topic'):
data = json.loads(request.data)
# print(json.dumps(data, encoding="UTF-8", ensure_ascii=False))
alerts = data['alerts']
# element_lens = len(alerts)
# print(element_lens)
msg_list = []
str_msg = ''
num = 0
while num < len(alerts):
alert_status = alerts[num].get('status')
if alert_status == 'firing':
env = alerts[num].get('labels')['env'] # 告警环境
alert_object = alerts[num].get('labels')['app'] # 告警对象
alert_name = alerts[num].get('labels')['alertname'] # 告警主题
alert_desc = alerts[num].get('annotations')['description'] # 告警详情
alert_starts_at = format_date_tz(alerts[num].get('startsAt')) # 触发时间
alert_status = alerts[num].get('status') # 告警状态
str_msg = '''> **=============start============**\n\n> **告警环境:** {0}\n\n> **告警对象:** {1}\n\n> **告警主题:** {2}\n\n> **告警详情:** <font color='#FF4500'> {3} </font>\n\n> **触发时间:** {4}\n\n> **告警状态:** <font color='#FF4500'> {5} </font>\n\n> **=============end=============**'''.format(env, alert_object, alert_name, alert_desc, alert_starts_at, alert_status)
elif alert_status == 'resolved':
env = alerts[num].get('labels')['env'] # 告警环境
alert_object = alerts[num].get('labels')['app'] # 告警对象
alert_name = alerts[num].get('labels')['alertname'] # 告警主题
alert_desc = alerts[num].get('annotations')['description'] # 告警详情
alert_starts_at = format_date_tz(alerts[num].get('startsAt')) # 触发时间
alert_ends_at = format_date_tz(alerts[num].get('endsAt')) # 恢复时间
alert_status = alerts[num].get('status') # 告警状态
str_msg = '''> **============start=============**\n\n> **告警环境:** {0}\n\n> **告警对象:** {1}\n\n> **告警主题:** {2}\n\n> **告警详情:** <font color='#FF4500'> {3} </font>\n\n> **触发时间:** {4}\n\n> **恢复时间:** {5}\n\n> **告警状态:** <font color='#9ACD32'> {6} </font>\n\n> **===========end===============**'''.format(env, alert_object, alert_name, alert_desc, alert_starts_at, alert_ends_at, alert_status)
msg_list.append(str_msg)
# print(msg_list)
str_msg = ''.join(msg_list)
# print(str_msg)
num += 1
return trim(str_msg)
else:
data = json.loads(request.data)
# print(json.dumps(data, encoding="UTF-8", ensure_ascii=False))
alerts = data['alerts']
msg_list = []
str_msg = ''
num = 0
while num < len(alerts):
alert_status = alerts[num].get('status')
if alert_status == 'firing':
env = alerts[num].get('labels')['env'] # 告警环境
alert_object = alerts[num].get('labels')['app'] # 告警业务
alert_name = alerts[num].get('labels')['alertname'] # 告警主题
alert_group = alerts[num].get('annotations')['group'] # 告警消息组
alert_topic = alerts[num].get('annotations')['topic'] # 告警topic
alert_desc = alerts[num].get('annotations')['description'] # 告警详情
alert_starts_at = format_date_tz(alerts[num].get('startsAt')) # 触发时间
alert_status = alerts[num].get('status') # 告警状态
str_msg = '''
============start=============
告警环境: {0}
告警业务: {1}
告警主题: {2}
告警消息组: {3}
告警Topic: {4}
告警详情: {5}
触发时间: {6}
告警状态: {7}
===========end================
'''.format(env, alert_object, alert_name, alert_group,
alert_topic, alert_desc, alert_starts_at,
alert_status)
elif alert_status == 'resolved':
env = alerts[num].get('labels')['env'] # 告警环境
alert_object = alerts[num].get('labels')['app'] # 告警业务
alert_name = alerts[num].get('labels')['alertname'] # 告警主题
alert_group = alerts[num].get('annotations')['group'] # 告警消息组
alert_topic = alerts[num].get('annotations')['topic'] # 告警topic
alert_desc = alerts[num].get('annotations')['description'] # 告警详情
alert_starts_at = format_date_tz(alerts[num].get('startsAt')) # 触发时间
alert_ends_at = format_date_tz(alerts[num].get('endsAt')) # 恢复时间
alert_status = alerts[num].get('status') # 告警状态
str_msg = '''
============start=============
告警环境: {0}
告警业务: {1}
告警主题: {2}
告警消息组: {3}
告警Topic: {4}
告警详情: {5}
触发时间: {6}
恢复时间: {7}
告警状态: {8}
===========end================
'''.format(env, alert_object, alert_name, alert_group,
alert_topic, alert_desc, alert_starts_at,
alert_ends_at, alert_status)
msg_list.append(str_msg)
# print(msg_list)
str_msg = ''.join(msg_list)
# print(str_msg)
num += 1
return trim(str_msg)
except Exception as e:
logging.error(time.strftime("%Y-%m-%d %H:%M:%S", time.localtime()) + e)
print(e)
@app.route('/dingtalk/webhook1/send', methods=['POST'])
def send_ding_message():
robot_webhook = "https://oapi.dingtalk.com/robot/send?access_token=4704705dea51cb9d5bc78167sdsfafas1c9eaf1becf559d7d0311754adcf2fde31e2d46e"
robot_secret = "SEC9c87827412e3d9c265e3b92b7f4067ad39d73dda7416825ab681f6bfsdf3f30ee1"
webhook = get_webhook(robot_webhook, robot_secret, 1) # 主要模式有 0 : 敏感字 1:# 敏感字 +加签 3:敏感字+加签+IP
# 请求头部
header = {"Content-Type": "application/json", "Charset": "UTF-8"}
msg = get_message()
print(msg)
#payload = {
# "msgtype": 'text',
# "text": {'content': msg},
#}
payload = {
"msgtype": "markdown",
"markdown": {"title": "芯动科技服务器告警", "text":msg},
}
# 对请求的数据进行json封装
message_json = json.dumps(payload)
# print(message_json)
requests.DEFAULT_RETRIES = 3
# 发送请求
info = requests.post(url=webhook, data=message_json, headers=header)
# 打印返回结果
print(info.text)
if (json.loads(info.text)['errcode'] == 0):
return jsonify({
'retval': 0,
'msg': 'ok',
'description': '告警消息发送成功'
})
logging.error(
time.strftime("%Y-%m-%d %H:%M:%S", time.localtime()) +
' 钉钉接口调用失败 ' + json.loads(info.text)['sub_msg'])
return jsonify({
'retval': json.loads(info.text)['sub_code'],
'msg': json.loads(info.text)['sub_msg'],
'description': '告警消息发送失败'
})
if __name__ == "__main__":
# 运维群机器人消息通知
LOG_FORMAT = "%(asctime)s - %(levelname)s - %(message)s"
DATE_FORMAT = "%m/%d/%Y %H:%M:%S %p"
logging.basicConfig(filename='/opt/py/dingtalk-robot.log', level=logging.DEBUG, format=LOG_FORMAT,
datefmt=DATE_FORMAT)
print('---------------- 启动app前')
app.run(debug=True, host='0.0.0.0', port=8060)
AI 代码解读