507 lines
21 KiB
Python
507 lines
21 KiB
Python
# -*- coding: utf-8 -*-
|
||
import logging
|
||
import re
|
||
import time
|
||
import xml.etree.ElementTree as ET
|
||
from queue import Empty
|
||
from threading import Thread
|
||
import random
|
||
|
||
from gewechat_client import GewechatClient
|
||
|
||
from base.func_epic import is_friday, get_free
|
||
|
||
from base.func_news import News
|
||
from configuration import Config
|
||
from gewechat.call_back_message.message import WxMessage, MessageType
|
||
from utils.json_converter import json_to_object
|
||
from utils.wechat.message_to_db import MessageStorage
|
||
from plugin_common.event_system import EventType, EventSystem
|
||
from plugin_common.message_plugin_interface import MessagePluginInterface
|
||
from plugin_common.plugin_interface import PluginStatus
|
||
from plugin_common.plugin_manager import PluginManager
|
||
from plugin_common.plugin_registry import PluginRegistry
|
||
from utils.robot_cmd.robot_command import GroupBotManager
|
||
from job_mgmt import Job
|
||
from utils.robot_cmd.robot_command import Feature
|
||
from utils.robot_cmd.robot_command import PermissionStatus
|
||
|
||
from sehuatang.shehuatang import pdf_file_path
|
||
from utils.wechat.contact_manager import ContactManager
|
||
from xiuren.meitu_dl import meitu_dowload_pub_pic
|
||
from xiuren.xiuren_pdf import generate_pdf_from_images
|
||
|
||
from db.connection import DBConnectionManager
|
||
from message_util import MessageUtil
|
||
|
||
|
||
class Robot(Job):
|
||
"""个性化自己的机器人
|
||
"""
|
||
|
||
def __init__(self, config: Config, app_id: str, client: GewechatClient, chat_type: int) -> None:
|
||
self.app_id = app_id
|
||
self.client = client
|
||
self.config = config
|
||
self.LOG = logging.getLogger("Robot")
|
||
|
||
# 初始化联系人管理器并设置联系人
|
||
self.contact_manager = ContactManager.get_instance()
|
||
self.allContacts = self.get_all_contacts()
|
||
self.contact_manager.set_contacts(self.allContacts)
|
||
|
||
# 获取个人信息
|
||
obj = json_to_object(self.client.get_profile(self.app_id))
|
||
if obj.data.wxid is None:
|
||
self.LOG.info(f"获取个人信息失败,退出程序!")
|
||
return
|
||
self.wxid = obj.data.wxid
|
||
|
||
self.LOG.info(f"DB+REDIS 连接池开始初始化")
|
||
# 使用单例模式获取实例
|
||
self.db_manager = DBConnectionManager.get_instance(
|
||
mysql_config=self.config.mariadb,
|
||
redis_config=self.config.redis
|
||
)
|
||
self.LOG.info(f"数据库连接管理器初始化完成")
|
||
|
||
# 为了兼容现有代码,保留原有的连接池
|
||
self.db_pool = self.db_manager.mysql_pool
|
||
self.redis_pool = self.db_manager.redis_pool
|
||
|
||
# 初始化消息工具类 - 使用联系人管理器
|
||
self.message_util = MessageUtil(app_id, client)
|
||
self.groups = {} # 存储按group_id分组的消息列表,每个group_id最多保留10条消息
|
||
GroupBotManager.load_local_cache()
|
||
|
||
# 权限模块加载
|
||
self.gbm = GroupBotManager()
|
||
# 初始化插件系统
|
||
self.LOG.info("开始初始化插件系统...")
|
||
self.plugin_registry = PluginRegistry()
|
||
self.event_system = EventSystem()
|
||
self.plugin_modules = {} # 存储已加载的插件模块
|
||
self.plugins = {} # 存储已加载的插件实例
|
||
|
||
# 设置插件系统上下文
|
||
self.system_context = {
|
||
"config": config,
|
||
"client": client,
|
||
"event_system": self.event_system,
|
||
"plugin_registry": self.plugin_registry,
|
||
"db_pool": self.db_pool,
|
||
"redis_pool": self.redis_pool,
|
||
"message_util": self.message_util
|
||
}
|
||
|
||
self.plugin_manager = PluginManager(plugin_dir=getattr(self.config, "plugin_dir", "plugins"))
|
||
self.plugin_manager.set_system_context(self.system_context)
|
||
self.plugins = self.plugin_manager.load_all_plugins()
|
||
|
||
# 加载插件
|
||
self.LOG.info("插件系统初始化完成")
|
||
|
||
# 消息存档模块初始化,自动完成入库动作
|
||
self.message_storage = MessageStorage(self.client)
|
||
|
||
@staticmethod
|
||
def value_check(args: dict) -> bool:
|
||
if args:
|
||
return all(value is not None for key, value in args.items() if key != 'proxy')
|
||
return False
|
||
|
||
def toChitchat(self, msg: WxMessage) -> bool:
|
||
"""闲聊,接入 ChatGPT
|
||
"""
|
||
# 去除@的人和空格等字符
|
||
q = re.sub(r"@.*?[\u2005|\s]", "", msg.content.raw_content).replace(" ", "")
|
||
|
||
if q == "#今日百度新闻":
|
||
self.news_baidu_report((msg.roomid if msg.from_group() else msg.sender))
|
||
return True
|
||
elif q in ["nbc", "cnn", "abc", "fox", "bbc"]:
|
||
self.news_en_report(q, (msg.roomid if msg.from_group() else msg.sender))
|
||
return True
|
||
else:
|
||
# 如果是群消息,并且群没开启AI,则不处理该动作
|
||
if msg.from_group() and self.gbm.get_group_permission(msg.roomid,
|
||
Feature.AI_CAPABILITY) == PermissionStatus.ENABLED:
|
||
resp = self.gbm.get_enabled_features(msg.roomid)
|
||
self.message_util.send_text(resp, (msg.roomid if msg.from_group() else msg.sender), msg.sender)
|
||
return True
|
||
else:
|
||
return True
|
||
|
||
def processMsg(self, msg: WxMessage) -> None:
|
||
"""当接收到消息的时候,会调用本方法。如果不实现本方法,则打印原始消息。
|
||
此处可进行自定义发送的内容,如通过 msg.content 关键字自动获取当前天气信息,并发送到对应的群组@发送者
|
||
群号:msg.roomid 微信ID:msg.sender 消息内容:msg.content
|
||
content = "xx天气信息为:"
|
||
receivers = msg.roomid
|
||
self.sendTextMsg(content, receivers, msg.sender)
|
||
"""
|
||
try:
|
||
# 检测群聊是否已加入机器人管理,如果没有则自动添加并开启机器人功能
|
||
if msg.from_group() and msg.roomid not in GroupBotManager.local_cache["group_list"]:
|
||
self.LOG.info(f"检测到新群聊: {msg.roomid},自动添加到机器人管理列表并启用机器人功能")
|
||
# 添加群组到列表
|
||
GroupBotManager.local_cache["group_list"].add(msg.roomid)
|
||
# 保存到Redis
|
||
redis_conn = self.db_manager.get_redis_connection()
|
||
redis_conn.sadd("group:list", msg.roomid)
|
||
# 设置ROBOT功能为启用状态
|
||
GroupBotManager.set_group_permission(msg.roomid, Feature.ROBOT, PermissionStatus.ENABLED)
|
||
# 更新联系人信息
|
||
self.refresh_contacts()
|
||
except Exception as e:
|
||
self.LOG.error(f"加入新群,自动添加并开启机器人功能 error: {e}")
|
||
|
||
# 发布消息接收事件
|
||
self.event_system.publish(EventType.MESSAGE_RECEIVED, {"message": msg})
|
||
|
||
# 标记插件是否处理了消息
|
||
plugin_processed = False
|
||
|
||
# 尝试使用插件处理消息
|
||
if self.process_plugin_message(msg):
|
||
plugin_processed = True
|
||
|
||
# 群聊消息处理 - 无论插件是否处理过,都执行数据存储
|
||
if msg.from_group():
|
||
# 调用统计逻辑进行聊天数据统计:
|
||
try:
|
||
self.message_storage.process_message(msg)
|
||
except Exception as e:
|
||
self.LOG.error(f"process_message error: {e}")
|
||
|
||
# 聊天记录入库动作:
|
||
try:
|
||
self.message_storage.archive_message(msg)
|
||
# 单独处理图片消息
|
||
if msg.msg_type == 3: # 图片消息类型
|
||
self.message_storage.process_image(msg)
|
||
except Exception as e:
|
||
self.LOG.error(f"archive_message error: {e}")
|
||
|
||
# 如果插件已处理消息,则不再执行后续的业务逻辑
|
||
if plugin_processed:
|
||
return
|
||
|
||
# 记录在群里发的最新消息,可以通过撤回指令撤回
|
||
try:
|
||
if msg.from_self():
|
||
rsp = self.gbm.handle_command(msg.roomid, msg.content)
|
||
# 不在群里发送,防止被骚扰
|
||
if rsp is not None:
|
||
self.message_util.send_text(rsp, msg.roomid, msg.sender)
|
||
return
|
||
except Exception as e:
|
||
self.LOG.error(f"revoke_receive_message error: {e}")
|
||
|
||
return # 处理完群聊信息,后面就不需要处理了
|
||
|
||
# 如果插件已处理消息,则不再执行后续的业务逻辑
|
||
if plugin_processed:
|
||
return
|
||
|
||
elif msg.msg_type == MessageType.TEXT: # 文本消息
|
||
# 让配置加载更灵活,自己可以更新配置。也可以利用定时任务更新。
|
||
if msg.from_self():
|
||
if msg.content == "^更新$":
|
||
self.config.reload()
|
||
self.gbm.load_local_cache()
|
||
self.LOG.info("已更新")
|
||
if msg.content == "今日百度新闻":
|
||
self.news_baidu_report()
|
||
if msg.content == "TO_DB":
|
||
self.message_count_to_db()
|
||
if msg.content == "PDF":
|
||
self.generate_sehuatang_pdf()
|
||
if msg.content.raw_content.startswith("清除群-"):
|
||
self.gbm.handle_command(msg.roomid, msg.content)
|
||
else:
|
||
self.toChitchat(msg) # 闲聊
|
||
|
||
def onMsg(self, msg: WxMessage) -> int:
|
||
try:
|
||
self.LOG.info(msg) # 打印信息
|
||
# self.processMsg(msg)
|
||
except Exception as e:
|
||
self.LOG.error(e)
|
||
|
||
return 0
|
||
|
||
def keep_running_and_block_process(self) -> None:
|
||
"""
|
||
保持机器人运行,不让进程退出
|
||
"""
|
||
while True:
|
||
self.runPendingJobs()
|
||
time.sleep(1)
|
||
|
||
# 添加一个方法用于刷新联系人信息
|
||
def refresh_contacts(self):
|
||
"""刷新联系人信息"""
|
||
self.allContacts = self.get_all_contacts()
|
||
self.contact_manager.refresh_contacts(self.allContacts)
|
||
self.LOG.info("联系人信息已刷新")
|
||
|
||
def send_group_txt_message(self, msg: str, feature: Feature):
|
||
try:
|
||
receivers = self.gbm.get_group_list()
|
||
if not receivers:
|
||
return
|
||
for r in receivers:
|
||
if self.gbm.get_group_permission(r, feature) == PermissionStatus.ENABLED:
|
||
self.message_util.send_text(msg, r)
|
||
except Exception as e:
|
||
self.LOG.error(f"send_group_txt_message:{feature.description} error:{e}")
|
||
|
||
def send_group_file_message(self, path: str, feature: Feature):
|
||
try:
|
||
receivers = self.gbm.get_group_list()
|
||
if not receivers:
|
||
return
|
||
for r in receivers:
|
||
if self.gbm.get_group_permission(r, feature) == PermissionStatus.ENABLED:
|
||
self.message_util.send_file(path, r)
|
||
except Exception as e:
|
||
self.LOG.error(f"send_group_file_message:{feature.description} error:{e}")
|
||
|
||
def process_plugin_message(self, msg: WxMessage) -> bool:
|
||
"""使用插件处理消息"""
|
||
# 获取所有消息处理插件
|
||
message_plugins = self.plugin_registry.get_plugins_by_type(MessagePluginInterface)
|
||
|
||
# 依次尝试处理消息
|
||
for plugin in message_plugins:
|
||
if plugin.status != PluginStatus.RUNNING:
|
||
continue
|
||
|
||
try:
|
||
# 转换WxMessage为插件可处理的格式
|
||
plugin_msg = {
|
||
"type": msg.msg_type,
|
||
"content": msg.content,
|
||
"sender": msg.sender,
|
||
"roomid": msg.roomid if msg.from_group() else "",
|
||
"xml": msg.content.xml_content,
|
||
"is_at": msg.is_at(self.wxid), # 确保正确设置is_at标志
|
||
"timestamp": time.time(),
|
||
"message_util": self.message_util, # 提供消息工具类
|
||
"gbm": self.gbm, # 每次从程序变量中取,保证最新
|
||
"all_contacts": self.allContacts,
|
||
"full_wx_msg": msg
|
||
}
|
||
|
||
# 检查插件是否可以处理该消息
|
||
if plugin.can_process(plugin_msg):
|
||
processed, _ = plugin.process_message(plugin_msg)
|
||
if processed:
|
||
# 发布消息处理事件
|
||
self.event_system.publish(EventType.MESSAGE_PROCESSED, {
|
||
"message": msg,
|
||
"plugin": plugin.name
|
||
})
|
||
|
||
return True
|
||
except Exception as e:
|
||
self.LOG.error(f"插件 {plugin.name} 处理消息失败: {e}")
|
||
|
||
return False
|
||
|
||
# ============================================== 业务内容==========================================================
|
||
def news_baidu_report_auto(self) -> None:
|
||
try:
|
||
news = News().get_baidu_news()
|
||
self.send_group_txt_message(news, Feature.DAILY_NEWS)
|
||
except Exception as e:
|
||
self.LOG.error(f"newsBaiduReportAuto error:{e}")
|
||
|
||
def news_baidu_report(self, sender: str = None) -> None:
|
||
try:
|
||
news = News().get_baidu_news()
|
||
if news and isinstance(news, str):
|
||
self.message_util.send_text(news, sender)
|
||
else:
|
||
self.LOG.error("获取百度新闻返回值异常")
|
||
except Exception as e:
|
||
self.LOG.error(f"newsBaiduReport error:{e}")
|
||
# 发送错误信息给用户,让用户知道发生了什么
|
||
|
||
def news_en_report(self, website, sender: str = None) -> None:
|
||
try:
|
||
news = News().get_eng_news(website)
|
||
self.message_util.send_text(news, sender)
|
||
except Exception as e:
|
||
self.LOG.error(f"newsEnReport error:{e}")
|
||
|
||
# 使用装饰器标记定时任务 星期五 10:30 执行
|
||
def send_epic_free_games(self):
|
||
try:
|
||
if is_friday():
|
||
games = get_free()
|
||
self.send_group_txt_message(games, Feature.EPIC)
|
||
except Exception as e:
|
||
self.LOG.error(f"sendEpicFreeGames error:{e}")
|
||
|
||
# 使用装饰器标记定时任务
|
||
def message_count_to_db(self):
|
||
try:
|
||
self.message_storage.write_to_db()
|
||
except Exception as e:
|
||
self.LOG.error(f"write_to_db error:{e}")
|
||
|
||
def generate_sehuatang_pdf(self):
|
||
try:
|
||
self.LOG.info("开始生成PDF,generate_sehuatang_pdf")
|
||
path = pdf_file_path()
|
||
# 暂时只发4K群
|
||
self.send_group_file_message(path, Feature.PDF_CAPABILITY)
|
||
except Exception as e:
|
||
self.LOG.error(f"generateSehuatangPdf error:{e}")
|
||
|
||
def generate_and_send_ranking(self):
|
||
try:
|
||
receivers = self.gbm.get_group_list()
|
||
if not receivers:
|
||
return
|
||
for r in receivers:
|
||
if self.gbm.get_group_permission(r, Feature.DAILY_SUMMARY) == PermissionStatus.ENABLED:
|
||
output = self.message_storage.generate_and_send_ranking(r, self.allContacts)
|
||
self.message_util.send_text(output, r)
|
||
except Exception as e:
|
||
self.LOG.error(f"SendRanking error:{e}")
|
||
|
||
#
|
||
# # 设置定时任务
|
||
# def game_auto_tasks(self):
|
||
# try:
|
||
# group_ids = get_group_ids()
|
||
# for gid in group_ids:
|
||
# if self.gbm.get_group_permission(gid, Feature.TASK_GAME) == PermissionStatus.ENABLED:
|
||
# rep = run_random_task_assignment(group_id=gid)
|
||
# message = rep["message"]
|
||
# player_id = rep["player_id"]
|
||
# print(f"消息: {message}")
|
||
# print(f"玩家ID: {player_id}")
|
||
# self.send_text_msg(message, gid, player_id)
|
||
# except Exception as e:
|
||
# self.LOG.error(f"message_summary_robot error:{e}")
|
||
|
||
def xiu_ren_download_task(self):
|
||
try:
|
||
# 每天下载10组图,然后发一个帖子PDF
|
||
meitu_dowload_pub_pic()
|
||
except Exception as e:
|
||
self.LOG.error(f"xiu_ren_download_task error:{e}")
|
||
|
||
def xiu_ren_pdf_send(self):
|
||
try:
|
||
pub_path = generate_pdf_from_images("xiuren")
|
||
self.message_util.send_file(pub_path, "45317011307@chatroom")
|
||
except Exception as e:
|
||
self.LOG.error(f"xiu_ren_pdf_send error:{e}")
|
||
|
||
def get_all_contacts(self) -> dict:
|
||
"""获取所有联系人信息并保存到数据库"""
|
||
from db.contacts_db import ContactsDBOperator
|
||
|
||
try:
|
||
contacts_dict = {}
|
||
# 获取所有联系人的wxid列表
|
||
contacts_wxids = self.client.fetch_contacts_list(self.app_id)
|
||
self.LOG.info(f"contacts_wxids: {contacts_wxids}")
|
||
if not contacts_wxids:
|
||
self.LOG.warning("获取联系人列表为空")
|
||
return contacts_dict
|
||
|
||
# 初始化联系人数据库操作类
|
||
contacts_db = ContactsDBOperator()
|
||
|
||
# 将wxid列表分批处理,每批50个
|
||
batch_size = 50
|
||
for i in range(0, len(contacts_wxids), batch_size):
|
||
batch_wxids = contacts_wxids[i:i + batch_size]
|
||
|
||
# 批量获取联系人详细信息
|
||
contact_info = self.client.get_detail_info(self.app_id, batch_wxids)
|
||
|
||
# 处理返回的数据
|
||
if contact_info and contact_info.get("ret") == 200 and "data" in contact_info:
|
||
contact_data = contact_info.get("data", [])
|
||
|
||
if contact_data:
|
||
for contact in contact_data:
|
||
user_name = contact.get("userName")
|
||
if not user_name:
|
||
continue
|
||
|
||
try:
|
||
# 判断联系人类型
|
||
contact_type = "friends" # 默认为好友类型
|
||
if user_name.endswith("@chatroom"):
|
||
contact_type = "chatrooms"
|
||
# 如果是群聊,则获取群成员信息
|
||
self.update_chatroom_member_details(user_name)
|
||
elif user_name.startswith("gh_"):
|
||
contact_type = "ghs"
|
||
|
||
# 保存到数据库
|
||
contacts_db.save_contacts([contact], contact_type)
|
||
|
||
# 添加到返回字典
|
||
contacts_dict[user_name] = contact.get("nickName") or user_name
|
||
|
||
except Exception as e:
|
||
self.LOG.error(f"处理联系人 {user_name} 失败: {e}")
|
||
continue
|
||
else:
|
||
self.LOG.error(f"获取联系人详情失败: {contact_info}")
|
||
|
||
self.LOG.info(f"成功获取并保存{len(contacts_dict)}个联系人信息")
|
||
return contacts_dict
|
||
|
||
except Exception as e:
|
||
self.LOG.error(f"获取联系人信息失败: {e}")
|
||
return {}
|
||
|
||
def update_chatroom_member_details(self, chatroom_id):
|
||
"""更新群成员详细信息"""
|
||
try:
|
||
# 首先获取群成员列表
|
||
members_response = self.client.get_chatroom_member_list(self.app_id, chatroom_id)
|
||
if members_response and members_response.get('ret') == 200:
|
||
member_list = members_response.get('data', {}).get('memberList', [])
|
||
|
||
# 提取成员wxid列表
|
||
member_wxids = [member.get('wxid') for member in member_list if member.get('wxid')]
|
||
|
||
if member_wxids:
|
||
# 按照官方接口格式传递参数
|
||
details_response = self.client.get_chatroom_member_detail(
|
||
self.app_id,
|
||
chatroom_id,
|
||
member_wxids # 直接传递列表,不需要转换为集合
|
||
)
|
||
|
||
# 使用ContactsDBOperator处理响应
|
||
from db.contacts_db import ContactsDBOperator
|
||
contacts_db = ContactsDBOperator()
|
||
success = contacts_db.process_chatroom_member_detail_response(chatroom_id, details_response)
|
||
|
||
if success:
|
||
self.LOG.info(f"成功更新群聊{chatroom_id}的成员详细信息")
|
||
else:
|
||
self.LOG.error(f"更新群聊{chatroom_id}的成员详细信息失败")
|
||
|
||
return success
|
||
else:
|
||
self.LOG.warning(f"群聊{chatroom_id}没有成员")
|
||
return False
|
||
else:
|
||
self.LOG.error(f"获取群聊{chatroom_id}成员列表失败")
|
||
return False
|
||
except Exception as e:
|
||
self.LOG.error(f"更新群聊成员详细信息出错: {e}")
|