698 lines
31 KiB
Python
698 lines
31 KiB
Python
# -*- coding: utf-8 -*-
|
||
import time
|
||
import asyncio
|
||
import threading
|
||
import tomllib
|
||
import toml
|
||
|
||
import wechat_ipad
|
||
from loguru import logger
|
||
|
||
from base.func_epic import is_friday, get_free
|
||
from base.func_news import News
|
||
from configuration import Config
|
||
from base.plugin_common.event_system import EventType, EventSystem
|
||
from base.plugin_common.message_plugin_interface import MessagePluginInterface
|
||
from base.plugin_common.plugin_interface import PluginStatus
|
||
from base.plugin_common.plugin_manager import PluginManager
|
||
from base.plugin_common.plugin_registry import PluginRegistry
|
||
from utils.sehuatang.shehuatang import pdf_file_path
|
||
from utils.email_util import EmailSender
|
||
from utils.revoke.message_auto_revoke import MessageAutoRevoke
|
||
from utils.robot_cmd.robot_command import GroupBotManager, Feature, PermissionStatus
|
||
|
||
from db.connection import DBConnectionManager
|
||
from db.contacts_db import ContactsDBOperator
|
||
from utils.wechat.contact_manager import ContactManager
|
||
from utils.wechat.message_to_db import MessageStorage
|
||
from wechat_ipad import WechatAPIClient
|
||
from wechat_ipad.models.message import WxMessage, MessageType
|
||
from plugins.xiuren_image.meitu_dl import meitu_dowload_pub_pic
|
||
|
||
|
||
class Robot:
|
||
"""个性化自己的机器人
|
||
"""
|
||
|
||
def __init__(self, config: Config) -> None:
|
||
super().__init__()
|
||
self.config = config
|
||
self.LOG = logger
|
||
|
||
# wechat_ipad 相关属性
|
||
self.ipad_bot: WechatAPIClient
|
||
self.ipad_config = None
|
||
self.ipad_running = False
|
||
self.ipad_thread = None
|
||
self.ipad_loop = None
|
||
self.wxid = None
|
||
self.nickname = None
|
||
self.alias = None
|
||
self.phone = None
|
||
self.message_auto_revoke: MessageAutoRevoke = None
|
||
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.contacts_db = ContactsDBOperator(self.db_manager)
|
||
# 初始化联系人管理器
|
||
self.contact_manager = ContactManager.get_instance()
|
||
self.allContacts = {} # 将在登录后填充
|
||
|
||
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,
|
||
"event_system": self.event_system,
|
||
"plugin_registry": self.plugin_registry,
|
||
"db_pool": self.db_pool,
|
||
"redis_pool": self.redis_pool
|
||
}
|
||
|
||
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.email_sender = EmailSender(
|
||
smtp_server=self.config.email.get("smtp_server", "smtp.163.com"),
|
||
smtp_port=self.config.email.get("smtp_port", 465),
|
||
sender_email=self.config.email.get("sender_email", "bovine_liu@163.com"),
|
||
sender_password=self.config.email.get("sender_password", "LTS9BhmX9XhS36QS")
|
||
)
|
||
|
||
def init_wechat_ipad(self):
|
||
"""初始化wechat_ipad客户端"""
|
||
try:
|
||
# 读取config.toml文件
|
||
with open("wechat_ipad/config.toml", "rb") as f:
|
||
self.ipad_config = tomllib.load(f)
|
||
|
||
self.LOG.info("正在初始化wechat_ipad客户端...")
|
||
|
||
# 检查必要的配置
|
||
server_url = self.ipad_config.get("server_url", "")
|
||
if server_url == "":
|
||
self.LOG.error("server_url不能为空,wechat_ipad初始化失败")
|
||
return False
|
||
|
||
server_ip = self.ipad_config.get("server_ip", "")
|
||
server_port = self.ipad_config.get("server_port", 8058)
|
||
|
||
# 创建事件循环
|
||
self.ipad_loop = asyncio.new_event_loop()
|
||
|
||
# 在新线程中启动wechat_ipad客户端
|
||
self.ipad_thread = threading.Thread(
|
||
target=self._run_wechat_ipad_client,
|
||
args=(server_ip, server_port),
|
||
daemon=True
|
||
)
|
||
self.ipad_thread.start()
|
||
|
||
self.LOG.info("wechat_ipad客户端初始化完成")
|
||
return True
|
||
except Exception as e:
|
||
self.LOG.error(f"初始化wechat_ipad客户端失败: {e}")
|
||
return False
|
||
|
||
def _run_wechat_ipad_client(self, server_ip, server_port):
|
||
"""在新线程中运行wechat_ipad客户端"""
|
||
asyncio.set_event_loop(self.ipad_loop)
|
||
self.ipad_loop.run_until_complete(self._wechat_ipad_core(server_ip, server_port))
|
||
|
||
async def _wechat_ipad_core(self, server_ip, server_port):
|
||
"""wechat_ipad核心逻辑,基于bot-core.py"""
|
||
try:
|
||
self.LOG.info("启动wechat_ipad bot")
|
||
# 调用登录接口
|
||
self.ipad_bot = wechat_ipad.WechatAPIClient(server_ip, server_port)
|
||
self.message_auto_revoke = MessageAutoRevoke(self.ipad_bot)
|
||
wxid = self.ipad_config.get("wxid", "")
|
||
device_name = self.ipad_config.get("device_name", "")
|
||
device_id = self.ipad_config.get("device_id", "")
|
||
|
||
if device_name == "":
|
||
device_name = self.ipad_bot.create_device_name()
|
||
if device_id == "":
|
||
device_id = self.ipad_bot.create_device_id()
|
||
|
||
# 登录逻辑
|
||
if not await self.ipad_bot.is_logged_in(wxid):
|
||
await self._handle_ipad_login(wxid, device_name, device_id)
|
||
else: # 已登录
|
||
self.ipad_bot.wxid = wxid
|
||
profile = await self.ipad_bot.get_profile()
|
||
self.ipad_bot.nickname = profile.get("NickName").get("string")
|
||
self.ipad_bot.alias = profile.get("Alias")
|
||
self.ipad_bot.phone = profile.get("BindMobile").get("string")
|
||
|
||
# 更新Robot类的属性
|
||
self.wxid = self.ipad_bot.wxid
|
||
self.nickname = self.ipad_bot.nickname
|
||
self.alias = self.ipad_bot.alias
|
||
self.phone = self.ipad_bot.phone
|
||
|
||
self.LOG.info(
|
||
f"wechat_ipad登录账号信息: wxid: {self.wxid} 昵称: {self.nickname} 微信号: {self.alias} 手机号: {self.phone}")
|
||
|
||
# 注入加载完成的bot
|
||
self.plugin_manager.inject_bot(self.ipad_bot)
|
||
self.LOG.info(f"wechat_ipad登录设备信息: device_name: {device_name} device_id: {device_id}")
|
||
self.LOG.info("wechat_ipad登录成功")
|
||
|
||
# 登录成功后加载联系人信息
|
||
self.allContacts = self.get_all_contacts()
|
||
friends = await self.ipad_bot.get_contract_list()
|
||
self.head_images = self.get_all_head_images()
|
||
self.all_chatroom_members = self.contacts_db.get_chatroom_member_list_name_all()
|
||
# self.LOG.debug(f"all_chatroom_members:{self.all_chatroom_members}")
|
||
self.contact_manager.set_contacts(self.allContacts, friends, self.head_images, self.all_chatroom_members)
|
||
|
||
self.message_storage = MessageStorage(self.ipad_bot)
|
||
|
||
# # 获取扩展信息,显示相关内容
|
||
ext_profile = await self.ipad_bot.get_profile_info_ext()
|
||
self.ipad_bot.profile_ext = ext_profile
|
||
self.head_image = ext_profile.get("SmallHeadImgUrl")
|
||
|
||
# 先接受堆积消息
|
||
self.LOG.info("处理堆积消息中")
|
||
|
||
await self.ipad_bot.send_text_message("filehelper", "ipad客户端启动成功")
|
||
count = 0
|
||
while True:
|
||
data = await self.ipad_bot.sync_message()
|
||
data = data.get("AddMsgs")
|
||
if not data:
|
||
if count > 2:
|
||
break
|
||
else:
|
||
count += 1
|
||
continue
|
||
|
||
self.LOG.debug(f"接受到 {len(data)} 条消息")
|
||
await asyncio.sleep(1)
|
||
self.LOG.info("处理堆积消息完毕")
|
||
|
||
# 标记为运行中
|
||
self.ipad_running = True
|
||
# 开启自动心跳(作为后台任务)
|
||
heartbeat_task = asyncio.create_task(self._heartbeat_task())
|
||
heartbeat_task_long = asyncio.create_task(self._heartbeat_task_long())
|
||
# 开始处理消息
|
||
self.LOG.info("开始处理wechat_ipad消息")
|
||
while self.ipad_running:
|
||
try:
|
||
data = await self.ipad_bot.sync_message()
|
||
except Exception as e:
|
||
self.LOG.error(f"获取新消息失败 {e}")
|
||
if "用户可能退出" in str(e):
|
||
self.LOG.error(f"用户可能退出: {e}")
|
||
self.email_sender.send_wechat_alert(self.config.email.get("alert_recipient"),
|
||
f"用户可能退出: {e}", self.wxid,
|
||
self.nickname)
|
||
await self.login_twice_auto_auth()
|
||
await asyncio.sleep(5)
|
||
continue
|
||
|
||
data = data.get("AddMsgs")
|
||
if data:
|
||
tasks = []
|
||
for message in data:
|
||
# self.LOG.debug(f"sync_message.处理消息消息内容: {message}")
|
||
# 处理消息
|
||
try:
|
||
wxmsg: WxMessage = WxMessage.from_json(message)
|
||
except Exception as e:
|
||
self.LOG.error(f"WxMessage.from_json 解析失败,消息内容: {message},错误: {e}")
|
||
continue # 跳过本条消息,继续处理下一条
|
||
tasks.append(self._process_ipad_message(wxmsg))
|
||
if tasks:
|
||
await asyncio.gather(*tasks)
|
||
# 使用异步睡眠替代忙等待循环
|
||
await asyncio.sleep(0.5)
|
||
|
||
except Exception as e:
|
||
self.LOG.exception(f"wechat_ipad客户端运行出错: {e}")
|
||
self.ipad_running = False
|
||
|
||
async def _handle_ipad_login(self, wxid, device_name, device_id):
|
||
"""处理wechat_ipad登录"""
|
||
while not await self.ipad_bot.is_logged_in(wxid):
|
||
# 需要登录
|
||
try:
|
||
if await self.ipad_bot.get_cached_info(wxid):
|
||
# 尝试唤醒登录
|
||
uuid = await self.ipad_bot.awaken_login(wxid)
|
||
self.LOG.info(f"获取到登录uuid: {uuid}")
|
||
else:
|
||
# 二维码登录
|
||
if not device_name:
|
||
device_name = self.ipad_bot.create_device_name()
|
||
if not device_id:
|
||
device_id = self.ipad_bot.create_device_id()
|
||
uuid, url = await self.ipad_bot.get_qr_code(device_id=device_id, device_name=device_name,
|
||
print_qr=True)
|
||
self.LOG.info(f"获取到登录uuid: {uuid}")
|
||
self.LOG.info(f"获取到登录二维码: {url}")
|
||
except Exception as e:
|
||
self.LOG.error(f"登录过程出错: {e}")
|
||
# 二维码登录
|
||
if not device_name:
|
||
device_name = self.ipad_bot.create_device_name()
|
||
if not device_id:
|
||
device_id = self.ipad_bot.create_device_id()
|
||
uuid, url = await self.ipad_bot.get_qr_code(device_id=device_id, device_name=device_name, print_qr=True)
|
||
self.LOG.info(f"获取到登录uuid: {uuid}")
|
||
self.LOG.info(f"获取到登录二维码: {url}")
|
||
|
||
while True:
|
||
self.LOG.info(f"uuid: {uuid}, url: {url}")
|
||
stat, data = await self.ipad_bot.check_login_uuid(uuid, device_id=device_id)
|
||
if stat:
|
||
break
|
||
self.LOG.info(f"等待登录中,过期倒计时:{data}")
|
||
await asyncio.sleep(5)
|
||
|
||
# 保存登录信息
|
||
self.ipad_config["wxid"] = self.ipad_bot.wxid
|
||
self.ipad_config["device_name"] = device_name
|
||
self.ipad_config["device_id"] = device_id
|
||
self.ipad_config["login_time"] = time.strftime("%Y-%m-%d %H:%M:%S", time.localtime())
|
||
with open("wechat_ipad/config.toml", "w", encoding="utf-8") as f:
|
||
toml.dump(self.ipad_config, f)
|
||
|
||
# 获取登录账号信息
|
||
self.ipad_bot.wxid = data.get("acctSectResp").get("userName")
|
||
self.ipad_bot.nickname = data.get("acctSectResp").get("nickName")
|
||
self.ipad_bot.alias = data.get("acctSectResp").get("alias")
|
||
self.ipad_bot.phone = data.get("acctSectResp").get("bindMobile")
|
||
|
||
# 更新Robot类的属性
|
||
self.wxid = self.ipad_bot.wxid
|
||
self.nickname = self.ipad_bot.nickname
|
||
self.alias = self.ipad_bot.alias
|
||
self.phone = self.ipad_bot.phone
|
||
self.LOG.info(
|
||
f"wechat_ipad登录账号信息: wxid: {self.wxid} 昵称: {self.nickname} 微信号: {self.alias} 手机号: {self.phone}")
|
||
break
|
||
|
||
async def _heartbeat_task(self):
|
||
"""wechat_ipad心跳任务"""
|
||
self.LOG.info("开启wechat_ipad心跳!")
|
||
while self.ipad_running:
|
||
try:
|
||
success = await self.ipad_bot.heartbeat()
|
||
if success:
|
||
self.LOG.debug("心跳进行中")
|
||
else:
|
||
self.LOG.warning("心跳失败")
|
||
except Exception as e:
|
||
self.LOG.error(f"wechat_ipad heartbeat: {e}")
|
||
if "用户可能退出" in str(e):
|
||
self.LOG.error(f"用户可能退出: {e}")
|
||
self.email_sender.send_wechat_alert(self.config.email.get("alert_recipient"), f"用户可能退出: {e}",
|
||
self.wxid,
|
||
self.nickname)
|
||
await self.login_twice_auto_auth()
|
||
await asyncio.sleep(60)
|
||
|
||
async def _heartbeat_task_long(self):
|
||
"""wechat_ipad心跳任务"""
|
||
self.LOG.info("开启wechat_ipad长连接心跳!")
|
||
while self.ipad_running:
|
||
try:
|
||
success = await self.ipad_bot.heartbeat_long()
|
||
if success:
|
||
self.LOG.debug("长连接心跳进行中")
|
||
else:
|
||
self.LOG.warning("长连接心跳失败")
|
||
except Exception as e:
|
||
self.LOG.error(f"wechat_ipad heartbeat long: {e}")
|
||
if "用户可能退出" in str(e):
|
||
self.LOG.error(f"用户可能退出: {e}")
|
||
self.email_sender.send_wechat_alert(self.config.email.get("alert_recipient"), f"用户可能退出: {e}",
|
||
self.wxid,
|
||
self.nickname)
|
||
await self.login_twice_auto_auth()
|
||
await asyncio.sleep(120)
|
||
|
||
async def _process_ipad_message(self, message: WxMessage):
|
||
"""处理wechat_ipad消息"""
|
||
try:
|
||
# self.LOG.debug(f"message: {message}")
|
||
# 消息已经是WxMessage对象,直接使用其属性和方法
|
||
|
||
# 判断是否为群消息
|
||
is_group = message.from_group()
|
||
group_id = message.roomid
|
||
|
||
# 检测群聊是否已加入机器人管理
|
||
if is_group and group_id not in GroupBotManager.local_cache["group_list"]:
|
||
self.LOG.info(f"检测到新群聊: {group_id},自动添加到机器人管理列表并启用机器人功能")
|
||
# 添加群组到列表
|
||
GroupBotManager.local_cache["group_list"].add(group_id)
|
||
# 保存到Redis
|
||
redis_conn = self.db_manager.get_redis_connection()
|
||
redis_conn.sadd("group:list", group_id)
|
||
# 设置ROBOT功能为启用状态
|
||
GroupBotManager.set_group_permission(group_id, Feature.ROBOT, PermissionStatus.ENABLED)
|
||
|
||
# 获取群成员信息并更新数据库
|
||
try:
|
||
chatroom_info = await self.ipad_bot.get_chatroom_info(group_id)
|
||
self.LOG.debug(f"获取到群信息: {chatroom_info}")
|
||
if chatroom_info:
|
||
# 保存群信息到数据库
|
||
self.contacts_db.save_chatroom_info(chatroom_info)
|
||
members = await self.ipad_bot.get_chatroom_member_list(group_id)
|
||
# 保存群成员信息
|
||
if members:
|
||
# 兼容逻辑已放到save_chatroom_member_simple内部
|
||
self.contacts_db.save_chatroom_member_simple(group_id, members)
|
||
self.LOG.info(f"member_list: {members}")
|
||
# 更新联系人缓存
|
||
for member in members:
|
||
wxid = member.get("UserName", "")
|
||
nick_name = member.get("NickName", "")
|
||
displayName = member.get("DisplayName", "")
|
||
small_head_img_url = member.get("SmallHeadImgUrl", "")
|
||
# 如果displayName不为空,使用displayName
|
||
if displayName:
|
||
nick_name = displayName
|
||
if wxid:
|
||
self.allContacts[wxid] = nick_name
|
||
|
||
self.head_images[wxid] = small_head_img_url
|
||
|
||
friends = await self.ipad_bot.get_contract_list()
|
||
self.all_chatroom_members = self.contacts_db.get_chatroom_member_list_name_all()
|
||
self.contact_manager.set_contacts(self.allContacts, friends, self.head_images,
|
||
self.all_chatroom_members)
|
||
self.LOG.info(f"已更新群 {group_id} 的成员信息")
|
||
except Exception as e:
|
||
self.LOG.error(f"获取群成员信息失败: {e}")
|
||
|
||
# 发布消息接收事件
|
||
self.event_system.publish(EventType.MESSAGE_RECEIVED, {"message": message})
|
||
|
||
# 尝试使用插件处理消息
|
||
plugin_processed = await self.process_plugin_message(message)
|
||
|
||
if is_group:
|
||
self.LOG.debug(f"入库和记录群消息: {message}")
|
||
# 调用统计逻辑进行聊天数据统计:
|
||
try:
|
||
self.message_storage.process_message(message)
|
||
except Exception as e:
|
||
self.LOG.error(f"process_message error: {e}")
|
||
|
||
# # 聊天记录入库动作:
|
||
try:
|
||
self.message_storage.archive_message(message)
|
||
# 单独处理图片消息 后续写定时任务自动完成下载。延时处理。
|
||
if message.msg_type == MessageType.IMAGE: # 图片消息类型
|
||
self.message_storage.process_image(message)
|
||
except Exception as e:
|
||
self.LOG.error(f"archive_message error: {e}")
|
||
|
||
except Exception as e:
|
||
self.LOG.error(f"处理wechat_ipad消息出错: {e}")
|
||
|
||
def stop_wechat_ipad(self):
|
||
"""停止wechat_ipad客户端"""
|
||
self.ipad_running = False
|
||
if self.ipad_loop:
|
||
self.ipad_loop.stop()
|
||
self.LOG.info("wechat_ipad客户端已停止")
|
||
|
||
def keep_running_and_block_process(self) -> None:
|
||
"""
|
||
保持机器人运行,不让进程退出
|
||
"""
|
||
while True:
|
||
time.sleep(1)
|
||
|
||
async 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:
|
||
await self.ipad_bot.send_text_message(r, msg)
|
||
except Exception as e:
|
||
self.LOG.error(f"send_group_txt_message:{feature.description} error:{e}")
|
||
|
||
async 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:
|
||
await self.ipad_bot.send_at_message(r, "98堂 PDF已就绪,请手动发送", ["Jyunere"])
|
||
# await self.ipad_bot.send_image_message(r, path)
|
||
except Exception as e:
|
||
self.LOG.error(f"send_group_file_message:{feature.description} error:{e}")
|
||
|
||
async def process_plugin_message(self, msg) -> bool:
|
||
"""使用插件处理消息"""
|
||
# 获取所有消息处理插件
|
||
|
||
# 关闭00:30-05:00的系统交互,降低被风控风险
|
||
current_hour = time.localtime().tm_hour
|
||
current_minute = time.localtime().tm_min
|
||
is_sleep_time = (current_hour == 0 and current_minute >= 30) or (1 <= current_hour < 5)
|
||
|
||
if is_sleep_time:
|
||
# 只处理特定消息,如管理员消息或紧急消息
|
||
self.LOG.info(f"夜间休眠时间(00:30-05:00),忽略消息: {msg}")
|
||
return False
|
||
message_plugins = self.plugin_registry.get_plugins_by_type(MessagePluginInterface)
|
||
if not message_plugins:
|
||
return False
|
||
|
||
# 依次尝试处理消息
|
||
for plugin in message_plugins:
|
||
if plugin.status != PluginStatus.RUNNING:
|
||
continue
|
||
|
||
try:
|
||
# 转换消息为插件可处理的格式
|
||
plugin_msg = {
|
||
"type": msg.msg_type,
|
||
"content": msg.content.clean_content,
|
||
"sender": msg.sender,
|
||
"roomid": msg.roomid if msg.from_group() else "",
|
||
"is_at": msg.is_at(self.wxid),
|
||
"timestamp": time.time(),
|
||
"all_contacts": self.allContacts,
|
||
"full_wx_msg": msg,
|
||
"gbm": self.gbm,
|
||
"bot": self.ipad_bot,
|
||
"revoke": self.message_auto_revoke
|
||
}
|
||
|
||
# 检查插件是否可以处理该消息
|
||
if plugin.can_process(plugin_msg):
|
||
processed, _ = await 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 get_all_contacts(self) -> dict:
|
||
"""获取所有联系人信息并返回字典格式 {wxid: nickname}"""
|
||
try:
|
||
# 从数据库获取联系人信息
|
||
contacts = self.contacts_db.get_all_contacts()
|
||
return contacts
|
||
except Exception as e:
|
||
self.LOG.error(f"获取联系人信息失败: {e}")
|
||
return {}
|
||
|
||
def get_all_head_images(self) -> dict:
|
||
"""获取所有的联系人头像信息"""
|
||
try:
|
||
# 从数据库获取所有联系人的头像信息
|
||
head_images = self.contacts_db.get_all_contacts_avatar()
|
||
return head_images
|
||
except Exception as e:
|
||
self.LOG.error(f"获取所有联系人头像信息失败: {e}")
|
||
return {}
|
||
|
||
async def refresh_contacts_db(self):
|
||
"""刷新联系人信息"""
|
||
# 获取用户所有的联系人,并保存到数据库
|
||
self.LOG.info("开始刷新联系人信息")
|
||
# 删除所有的联系人信息
|
||
self.contacts_db.delete_all_contacts()
|
||
self.LOG.debug("已删除所有的联系人信息")
|
||
contacts = await self.ipad_bot.get_contract_list()
|
||
self.LOG.debug(f"获取到的联系人:{contacts}")
|
||
# 获取联系人详细信息,get_contract_detail每次可以获取20个
|
||
# 每次获取20个,需要循环获取
|
||
# 将联系人列表分成每组20个
|
||
batch_size = 20
|
||
for i in range(0, len(contacts), batch_size):
|
||
# 获取当前批次的联系人
|
||
batch_contacts = contacts[i:i + batch_size]
|
||
# 获取这批联系人的详细信息
|
||
contact_info = await self.ipad_bot.get_contract_detail(batch_contacts)
|
||
self.LOG.debug(f"获取到的联系人详细信息数量:{len(contact_info)}")
|
||
self.contacts_db.save_contacts(contact_info, "friends")
|
||
|
||
# 获取群聊列表
|
||
groups = self.contacts_db.get_chatroom_list()
|
||
# 调用接口完成群成员信息获取与保存逻辑
|
||
for group in groups:
|
||
# 调用接口获取群成员信息
|
||
group_id = group["chatroom_id"]
|
||
chatroom_info = await self.ipad_bot.get_chatroom_info(group_id)
|
||
self.LOG.debug(f"获取到的群成员信息:{chatroom_info}")
|
||
if chatroom_info.get("UserName", ""):
|
||
# 保存群信息到数据库
|
||
self.contacts_db.save_chatroom_info(chatroom_info)
|
||
members = await self.ipad_bot.get_chatroom_member_list(group_id)
|
||
# 保存群成员信息
|
||
if members:
|
||
# 兼容逻辑已放到save_chatroom_member_simple内部
|
||
self.contacts_db.save_chatroom_member_simple(group_id, members)
|
||
self.LOG.info(f"member_list: {members}")
|
||
# 更新联系人缓存
|
||
for member in members:
|
||
wxid = member.get("UserName", "")
|
||
nick_name = member.get("NickName", "")
|
||
displayName = member.get("DisplayName", "")
|
||
small_head_img_url = member.get("SmallHeadImgUrl", "")
|
||
# 如果displayName不为空,使用displayName
|
||
if displayName:
|
||
nick_name = displayName
|
||
if wxid:
|
||
self.allContacts[wxid] = nick_name
|
||
|
||
self.head_images[wxid] = small_head_img_url
|
||
friends = await self.ipad_bot.get_contract_list()
|
||
|
||
self.all_chatroom_members = self.contacts_db.get_chatroom_member_list_name_all()
|
||
self.contact_manager.set_contacts(self.allContacts, friends, self.head_images,
|
||
self.all_chatroom_members)
|
||
self.LOG.info(f"已更新群 {group_id} 的成员信息")
|
||
else:
|
||
self.LOG.error(f"获取群 {group_id} 信息失败,证明用户无该群信息,删除群的相关资料。")
|
||
# 删除群数据库中的群信息
|
||
self.contacts_db.delete_chatroom_all_info(group_id)
|
||
self.LOG.info("联系人信息刷新完成")
|
||
|
||
async def login_twice_auto_auth(self) -> None:
|
||
try:
|
||
self.LOG.info(f"定时进行二次登录动作")
|
||
resp = await self.ipad_bot.twice_auto_auth()
|
||
if resp:
|
||
self.LOG.info(f"定时二次登录成功!")
|
||
if self.ipad_running:
|
||
self.LOG.info(f"ipad_wechat running:{self.ipad_running}")
|
||
else:
|
||
self.ipad_running = True
|
||
self.LOG.info(f"ipad_wechat stopped change running:{self.ipad_running}")
|
||
else:
|
||
self.LOG.error(f"定时二次登录失败!")
|
||
except Exception as e:
|
||
self.LOG.error(f"login_twice_auto_auth error: {e}")
|
||
|
||
# ============================================== 业务内容==========================================================
|
||
|
||
async def news_baidu_report_auto(self) -> None:
|
||
try:
|
||
news = News().get_baidu_news()
|
||
await self.send_group_txt_message(news, Feature.DAILY_NEWS)
|
||
except Exception as e:
|
||
self.LOG.error(f"newsBaiduReportAuto error:{e}")
|
||
|
||
async def news_en_report(self, website, sender: str = None) -> None:
|
||
try:
|
||
news = News().get_eng_news(website)
|
||
await self.ipad_bot.send_text_message(sender, news)
|
||
except Exception as e:
|
||
self.LOG.error(f"newsEnReport error:{e}")
|
||
|
||
# 使用装饰器标记定时任务 星期五 10:30 执行
|
||
|
||
async def send_epic_free_games(self):
|
||
try:
|
||
if is_friday():
|
||
games = get_free()
|
||
await self.send_group_txt_message(games, Feature.EPIC)
|
||
except Exception as e:
|
||
self.LOG.error(f"sendEpicFreeGames error:{e}")
|
||
|
||
# 使用装饰器标记定时任务
|
||
|
||
async 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}")
|
||
|
||
async def generate_sehuatang_pdf(self):
|
||
try:
|
||
self.LOG.info("开始生成PDF,generate_sehuatang_pdf")
|
||
path = pdf_file_path()
|
||
# 暂时只发4K群
|
||
await self.send_group_file_message(path, Feature.PDF_CAPABILITY)
|
||
except Exception as e:
|
||
self.LOG.error(f"generateSehuatangPdf error:{e}")
|
||
|
||
async 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}")
|
||
|
||
async 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 = await self.message_storage.generate_and_send_ranking(r, self.allContacts)
|
||
await self.ipad_bot.send_text_message(r, output)
|
||
except Exception as e:
|
||
self.LOG.error(f"SendRanking error:{e}")
|
||
|
||
async def send_ranking_task(self):
|
||
result = await self.ipad_bot.get_chatroom_nickname("Jyunere", "43687793133@chatroom")
|
||
self.LOG.info(f"send_ranking_task:{result}")
|