2025-07-25 08:19:17 +00:00
|
|
|
|
"""
|
|
|
|
|
|
微信群发功能
|
|
|
|
|
|
建议使用企业微信
|
|
|
|
|
|
但需要管理员提供企业id以及secret信息
|
|
|
|
|
|
通过wechatpy库实现
|
|
|
|
|
|
"""
|
2025-08-15 03:37:06 +00:00
|
|
|
|
import core.logger as logging
|
2025-08-04 13:07:44 +00:00
|
|
|
|
import requests
|
2025-10-21 09:49:04 +00:00
|
|
|
|
import base64
|
|
|
|
|
|
import hashlib
|
|
|
|
|
|
import json
|
2025-10-27 07:07:12 +00:00
|
|
|
|
import os
|
|
|
|
|
|
import time
|
2025-08-15 03:37:06 +00:00
|
|
|
|
logger = logging.logger
|
2025-08-04 13:07:44 +00:00
|
|
|
|
|
|
|
|
|
|
class Wechat:
|
2025-09-15 07:02:49 +00:00
|
|
|
|
def __init__(self, key: str):
|
2025-08-04 13:07:44 +00:00
|
|
|
|
# 虽然config在根目录,但是取决于调用代码在哪
|
|
|
|
|
|
# 只要启动代码文件在根目录,config就能找到
|
2025-09-15 07:02:49 +00:00
|
|
|
|
self.key = key
|
2025-08-04 13:07:44 +00:00
|
|
|
|
self.url = f"https://qyapi.weixin.qq.com/cgi-bin/webhook/send?key={self.key}"
|
2025-10-27 07:07:12 +00:00
|
|
|
|
self.image_path = r"./output/wechat/image"
|
|
|
|
|
|
os.makedirs(self.image_path, exist_ok=True)
|
2025-08-04 13:07:44 +00:00
|
|
|
|
|
|
|
|
|
|
def send_text(self, text: str):
|
|
|
|
|
|
"""
|
|
|
|
|
|
发送文本消息
|
|
|
|
|
|
"""
|
|
|
|
|
|
data = {
|
|
|
|
|
|
"msgtype": "text",
|
|
|
|
|
|
"text": {"content": text}
|
|
|
|
|
|
}
|
|
|
|
|
|
response = requests.post(self.url, json=data)
|
2025-10-21 09:49:04 +00:00
|
|
|
|
response.raise_for_status()
|
2025-08-04 13:07:44 +00:00
|
|
|
|
return response.json()
|
|
|
|
|
|
|
|
|
|
|
|
def send_markdown(self, text: str):
|
|
|
|
|
|
"""
|
|
|
|
|
|
发送markdown消息
|
|
|
|
|
|
"""
|
|
|
|
|
|
data = {
|
|
|
|
|
|
"msgtype": "markdown_v2",
|
|
|
|
|
|
"markdown_v2": {"content": text}
|
|
|
|
|
|
}
|
|
|
|
|
|
response = requests.post(self.url, json=data)
|
2025-10-21 09:49:04 +00:00
|
|
|
|
response.raise_for_status()
|
2025-08-04 13:07:44 +00:00
|
|
|
|
return response.json()
|
|
|
|
|
|
|
|
|
|
|
|
def send_image(self, image_url: str):
|
|
|
|
|
|
"""
|
|
|
|
|
|
发送图片消息
|
|
|
|
|
|
"""
|
2025-10-21 09:49:04 +00:00
|
|
|
|
image_bytes = self.download_image(image_url)
|
|
|
|
|
|
base64_str, md5_str = self.get_base64_and_md5(image_bytes)
|
2025-08-04 13:07:44 +00:00
|
|
|
|
data = {
|
|
|
|
|
|
"msgtype": "image",
|
2025-10-21 09:49:04 +00:00
|
|
|
|
"image": {
|
|
|
|
|
|
"base64": base64_str,
|
|
|
|
|
|
"md5": md5_str,
|
|
|
|
|
|
}
|
2025-08-04 13:07:44 +00:00
|
|
|
|
}
|
2025-10-27 07:07:12 +00:00
|
|
|
|
# 获取url中图片的名称
|
|
|
|
|
|
image_name = image_url.split("/")[-1].split(".")[0]
|
|
|
|
|
|
# 获取当前时间,格式为YYYYMMDDHHMMSS
|
|
|
|
|
|
now_time = time.strftime("%Y%m%d%H%M%S", time.localtime())
|
|
|
|
|
|
image_name = f"{image_name}_{now_time}.jpg"
|
|
|
|
|
|
image_path = os.path.join(self.image_path, image_name)
|
|
|
|
|
|
with open(image_path, "wb") as f:
|
|
|
|
|
|
f.write(image_bytes)
|
2025-10-28 08:56:47 +00:00
|
|
|
|
|
2025-08-04 13:07:44 +00:00
|
|
|
|
response = requests.post(self.url, json=data)
|
2025-10-21 09:49:04 +00:00
|
|
|
|
response.raise_for_status()
|
2025-10-27 07:07:12 +00:00
|
|
|
|
return response.json(), image_path, base64_str, md5_str
|
2025-10-21 09:49:04 +00:00
|
|
|
|
|
|
|
|
|
|
def download_image(self, image_url):
|
|
|
|
|
|
"""下载图片并返回 bytes"""
|
|
|
|
|
|
response = requests.get(image_url, timeout=10)
|
|
|
|
|
|
response.raise_for_status() # 抛出 HTTP 错误
|
|
|
|
|
|
return response.content
|
|
|
|
|
|
|
|
|
|
|
|
def get_base64_and_md5(self,image_bytes):
|
|
|
|
|
|
"""计算 Base64(不带 data: 前缀)和 MD5"""
|
|
|
|
|
|
b64 = base64.b64encode(image_bytes).decode('utf-8')
|
|
|
|
|
|
md5 = hashlib.md5(image_bytes).hexdigest()
|
|
|
|
|
|
return b64, md5
|
2025-08-04 13:07:44 +00:00
|
|
|
|
|
|
|
|
|
|
def send_file(self, file_url: str):
|
|
|
|
|
|
"""
|
|
|
|
|
|
发送文件消息
|
|
|
|
|
|
"""
|
|
|
|
|
|
data = {
|
|
|
|
|
|
"msgtype": "file",
|
|
|
|
|
|
"file": {"url": file_url}
|
|
|
|
|
|
}
|
|
|
|
|
|
response = requests.post(self.url, json=data)
|
2025-10-21 09:49:04 +00:00
|
|
|
|
response.raise_for_status()
|
2025-08-04 13:07:44 +00:00
|
|
|
|
return response.json()
|
|
|
|
|
|
|
|
|
|
|
|
def send_news(self, news: list):
|
|
|
|
|
|
"""
|
|
|
|
|
|
发送图文消息
|
|
|
|
|
|
"""
|
|
|
|
|
|
data = {
|
|
|
|
|
|
"msgtype": "news",
|
|
|
|
|
|
"news": {"articles": news}
|
|
|
|
|
|
}
|
|
|
|
|
|
response = requests.post(self.url, json=data)
|
2025-10-21 09:49:04 +00:00
|
|
|
|
response.raise_for_status()
|
2025-08-04 13:07:44 +00:00
|
|
|
|
return response.json()
|
|
|
|
|
|
|