269 lines
9.1 KiB
Python
269 lines
9.1 KiB
Python
#coding: utf-8
|
||
# +-------------------------------------------------------------------
|
||
# | YakPanel
|
||
# +-------------------------------------------------------------------
|
||
# | Copyright (c) 2015-2020 YakPanel(www.yakpanel.com) All rights reserved.
|
||
# +-------------------------------------------------------------------
|
||
# | Author: 沐落 <cjx@yakpanel.com>
|
||
# | Author: lx
|
||
# | 消息通道邮箱模块
|
||
# | 常用功能
|
||
# 字体加粗 **bold** ,[这是一个链接](http://www.yakpanel.com),代码段:`code`
|
||
# 支持3种字体颜色 <font color="info">绿色</font> <font color="comment">灰色</font> <font color="warning">橙红色</font>
|
||
# +-------------------------------------------------------------------
|
||
|
||
import os, sys, public, json, requests,re
|
||
import sys, os,time
|
||
panelPath = "/www/server/panel"
|
||
os.chdir(panelPath)
|
||
sys.path.insert(0,panelPath + "/class/")
|
||
import public, json, requests
|
||
from requests.packages import urllib3
|
||
# 关闭警告
|
||
|
||
urllib3.disable_warnings()
|
||
import socket
|
||
import requests.packages.urllib3.util.connection as urllib3_cn
|
||
class weixin_msg:
|
||
|
||
__module_name = None
|
||
__default_pl = "{}/data/default_msg_channel.pl".format(panelPath)
|
||
conf_path = 'data/weixin.json'
|
||
__weixin_info = None
|
||
def __init__(self):
|
||
try:
|
||
self.__weixin_info = json.loads(public.readFile(self.conf_path))
|
||
if not 'weixin_url' in self.__weixin_info:
|
||
self.__weixin_info = None
|
||
except :
|
||
self.__weixin_info = None
|
||
self.__module_name = self.__class__.__name__.replace('_msg','')
|
||
|
||
def get_version_info(self,get):
|
||
"""
|
||
获取版本信息
|
||
"""
|
||
data = {}
|
||
data['ps'] = 'Wecom used to receive panel message push'
|
||
data['version'] = '1.2'
|
||
data['date'] = '2022-08-10'
|
||
data['author'] = 'YakPanel'
|
||
data['title'] = 'Wecom'
|
||
data['help'] = 'http://www.yakpanel.com'
|
||
return data
|
||
|
||
def __get_default_channel(self):
|
||
"""
|
||
@获取默认消息通道
|
||
"""
|
||
try:
|
||
if public.readFile(self.__default_pl) == self.__module_name:
|
||
return True
|
||
except:pass
|
||
return False
|
||
|
||
def get_config(self,get):
|
||
"""
|
||
获取微信配置
|
||
"""
|
||
data = {}
|
||
if self.__weixin_info :
|
||
|
||
#全局配置开关,1开启,0关闭
|
||
if not 'state' in self.__weixin_info:
|
||
self.__weixin_info['state'] = 1
|
||
|
||
data = self.__weixin_info
|
||
|
||
if not 'list' in data: data['list'] = {}
|
||
|
||
title = 'Default'
|
||
if 'title' in data: title = data['title']
|
||
|
||
data['list']['default'] = {'title':title,'data':data['weixin_url'],'state':self.__weixin_info['state']}
|
||
data['default'] = self.__get_default_channel()
|
||
return data
|
||
|
||
def set_config(self,get):
|
||
"""
|
||
设置微信配置
|
||
@url 微信URL
|
||
@atall 默认@全体成员
|
||
@key 唯一标识,default=兼容之前配置
|
||
@title string 备注
|
||
@user
|
||
"""
|
||
|
||
if not hasattr(get, 'url'):
|
||
return public.returnMsg(False, public.lang("Please fill in the complete information"))
|
||
|
||
title = 'Default'
|
||
if hasattr(get, 'title'):
|
||
title = get.title
|
||
if len(title) > 7:
|
||
return public.returnMsg(False, public.lang("Note name cannot exceed 7 characters"))
|
||
|
||
key,status,state ='default', 1, 1
|
||
if 'key' in get: key = get.key
|
||
if 'status' in get: status = int(get.status)
|
||
if 'state' in get: state = int(get.state)
|
||
|
||
if not self.__weixin_info: self.__weixin_info = {}
|
||
if not 'list' in self.__weixin_info: self.__weixin_info['list'] = {}
|
||
|
||
#全局配置开关,1开启,0关闭
|
||
self.__weixin_info['state'] = state
|
||
|
||
#增加多个机器人
|
||
self.__weixin_info['list'][key] = {
|
||
"data": get.url.strip(),
|
||
"title":title,
|
||
"status":status,
|
||
"addtime":int(time.time())
|
||
}
|
||
|
||
#兼容旧配置只有一条url的情况
|
||
if key == 'default':
|
||
self.__weixin_info['weixin_url'] = get.url.strip()
|
||
self.__weixin_info['title'] = title
|
||
|
||
#统一格式化输出,包含主机名,ip,推送时间
|
||
try:
|
||
info = public.get_push_info('Message channel configuration reminder',['>configuration status: <font color=#20a53a>Success</font>\n\n'])
|
||
ret = self.send_msg(info['msg'])
|
||
except:
|
||
ret = self.send_msg('YakPanel alarm test')
|
||
|
||
if ret['status']:
|
||
|
||
#默认消息通道
|
||
if 'default' in get and get['default']:
|
||
public.writeFile(self.__default_pl, self.__module_name)
|
||
|
||
if ret['success'] <= 0:
|
||
return public.returnMsg(False, public.lang("Failed to add, please check whether the URL is correct"))
|
||
|
||
public.writeFile(self.conf_path, json.dumps(self.__weixin_info))
|
||
return public.returnMsg(True, public.lang("successfully set"))
|
||
else:
|
||
return public.returnMsg(False, public.lang("Failed to add, please check whether the URL is correct"))
|
||
|
||
def get_send_msg(self,msg):
|
||
"""
|
||
@name 处理md格式
|
||
"""
|
||
try:
|
||
title = 'YakPanel warning notification'
|
||
if msg.find("####") >= 0:
|
||
msg = msg.replace("\n\n","""
|
||
""").strip()
|
||
try:
|
||
title = re.search(r"####(.+)", msg).groups()[0]
|
||
except:pass
|
||
|
||
except:pass
|
||
return msg,title
|
||
|
||
def send_msg(self,msg,to_user = 'default'):
|
||
"""
|
||
@name 微信发送信息
|
||
@msg string 消息正文(正文内容,必须包含
|
||
1、服务器名称
|
||
2、IP地址
|
||
3、发送时间
|
||
)
|
||
@to_user string 指定发送人
|
||
"""
|
||
if not self.__weixin_info :
|
||
return public.returnMsg(False, public.lang("Information is not configured correctly."))
|
||
|
||
if 'state' in self.__weixin_info and self.__weixin_info['state'] == 0:
|
||
return public.returnMsg(False, public.lang("Notifications have been turned off, please turn them on and try again."))
|
||
|
||
if msg.find('####') == -1:
|
||
try:
|
||
msg = public.get_push_info('Notification Configuration Reminder',['>Send Content:{}\n\n'.format(msg)])['msg']
|
||
except:pass
|
||
|
||
msg,title = self.get_send_msg(msg)
|
||
data = {
|
||
"msgtype": "markdown",
|
||
"markdown": {
|
||
"content": msg
|
||
}
|
||
}
|
||
headers = {'Content-Type': 'application/json'}
|
||
|
||
error,success = 0,0
|
||
conf = self.get_config(None)['list']
|
||
|
||
res = {}
|
||
for to_key in to_user.split(','):
|
||
if not to_key in conf: continue
|
||
try:
|
||
#x = requests.post(url = conf[to_key]['data'], data=json.dumps(data), headers=headers,verify=False,timeout=10)
|
||
|
||
allowed_gai_family_lib=urllib3_cn.allowed_gai_family
|
||
def allowed_gai_family():
|
||
family = socket.AF_INET
|
||
return family
|
||
urllib3_cn.allowed_gai_family = allowed_gai_family
|
||
x = requests.post(url = conf[to_key]['data'], data = json.dumps(data),verify=False, headers=headers,timeout=10)
|
||
urllib3_cn.allowed_gai_family=allowed_gai_family_lib
|
||
|
||
if x.json()["errcode"] == 0:
|
||
success += 1
|
||
res[conf[to_key]['title']] = 1
|
||
else:
|
||
error += 1
|
||
res[conf[to_key]['title']] = 0
|
||
except:
|
||
error += 1
|
||
res[conf[to_key]['title']] = 0
|
||
try:
|
||
public.write_push_log(self.__module_name,title,res)
|
||
except:pass
|
||
|
||
ret = public.returnMsg(True,'Send completed, send successfully {}, send failed {}.'.format(success,error))
|
||
ret['success'] = success
|
||
ret['error'] = error
|
||
|
||
return ret
|
||
|
||
|
||
def push_data(self,data):
|
||
"""
|
||
@name 统一发送接口
|
||
@data 消息内容
|
||
{"module":"mail","title":"提醒","msg":"提醒","to_email":"xx@qq.com","sm_type":"","sm_args":{}}
|
||
"""
|
||
if not 'to_user' in data:
|
||
data['to_user'] = 'default'
|
||
|
||
return self.send_msg(data['msg'],data['to_user'])
|
||
|
||
|
||
def _write_log(self,module,msg,res):
|
||
"""
|
||
@name 写日志
|
||
"""
|
||
user = '[ 默认 ] '
|
||
# for key in res:
|
||
# status = '<span style="color:#20a53a;">成功</span>'
|
||
# if res[key] == 0: status = '<span style="color:red;">成功</span>'
|
||
# user += '[ {}:{} ] '.format(key,status)
|
||
|
||
try:
|
||
msg_obj = public.init_msg(module)
|
||
if msg_obj: module = msg_obj.get_version_info(None)['title']
|
||
except:pass
|
||
|
||
log = '[{}] sent to {}, sending content: [{}]'.format(module,user,public.xsssec(msg))
|
||
public.WriteLog('message push',log)
|
||
|
||
def uninstall(self):
|
||
if os.path.exists(self.conf_path):
|
||
os.remove(self.conf_path)
|
||
|
||
|