QChatGPT/pkg/qqbot/manager.py

305 lines
11 KiB
Python
Raw Normal View History

import asyncio
2022-12-11 17:17:33 +08:00
import json
import os
import threading
import mirai.models.bus
2023-01-17 15:43:28 +08:00
from mirai import At, GroupMessage, MessageEvent, Mirai, StrangerMessage, WebSocketAdapter, HTTPAdapter, \
FriendMessage, Image
2022-12-09 16:18:25 +08:00
2022-12-07 23:37:52 +08:00
import pkg.openai.session
2022-12-15 18:50:15 +08:00
import pkg.openai.manager
2022-12-26 19:37:25 +08:00
from func_timeout import FunctionTimedOut
2022-12-08 21:58:02 +08:00
import logging
2022-12-07 23:37:52 +08:00
2022-12-11 17:17:33 +08:00
import pkg.qqbot.filter
2022-12-26 19:37:25 +08:00
import pkg.qqbot.process as processor
import pkg.utils.context
2022-12-07 23:37:52 +08:00
2023-01-13 16:49:56 +08:00
import pkg.plugin.host as plugin_host
import pkg.plugin.models as plugin_models
2022-12-08 12:06:04 +08:00
# 并行运行
def go(func, args=()):
thread = threading.Thread(target=func, args=args, daemon=True)
thread.start()
# 检查消息是否符合泛响应匹配机制
def check_response_rule(text: str) -> (bool, str):
config = pkg.utils.context.get_config()
2022-12-19 17:15:17 +08:00
if not hasattr(config, 'response_rules'):
return False, ''
rules = config.response_rules
# 检查前缀匹配
2022-12-19 17:15:17 +08:00
if 'prefix' in rules:
for rule in rules['prefix']:
if text.startswith(rule):
return True, text.replace(rule, "", 1)
# 检查正则表达式匹配
2022-12-19 17:15:17 +08:00
if 'regexp' in rules:
for rule in rules['regexp']:
import re
match = re.match(rule, text)
if match:
2022-12-19 17:17:49 +08:00
return True, text
return False, ""
2022-12-11 16:10:12 +08:00
# 控制QQ消息输入输出的类
2022-12-07 23:37:52 +08:00
class QQBotManager:
retry = 3
bot: Mirai = None
2022-12-07 23:37:52 +08:00
2022-12-11 17:17:33 +08:00
reply_filter = None
2023-01-07 16:50:34 +08:00
enable_banlist = False
ban_person = []
ban_group = []
def __init__(self, mirai_http_api_config: dict, timeout: int = 60, retry: int = 3, first_time_init=True):
2022-12-07 23:37:52 +08:00
self.timeout = timeout
self.retry = retry
2023-01-07 16:50:34 +08:00
# 加载禁用列表
if os.path.exists("banlist.py"):
import banlist
self.enable_banlist = banlist.enable
self.ban_person = banlist.person
self.ban_group = banlist.group
logging.info("加载禁用列表: person: {}, group: {}".format(self.ban_person, self.ban_group))
config = pkg.utils.context.get_config()
2022-12-11 17:17:33 +08:00
if os.path.exists("sensitive.json") \
and config.sensitive_word_filter is not None \
and config.sensitive_word_filter:
with open("sensitive.json", "r", encoding="utf-8") as f:
self.reply_filter = pkg.qqbot.filter.ReplyFilter(json.load(f)['words'])
else:
self.reply_filter = pkg.qqbot.filter.ReplyFilter([])
2023-01-02 13:06:48 +08:00
# 由于YiriMirai的bot对象是单例的且shutdown方法暂时无法使用
# 故只在第一次初始化时创建bot对象重载之后使用原bot对象
# 因此bot的配置不支持热重载
if first_time_init:
self.first_time_init(mirai_http_api_config)
else:
self.bot = pkg.utils.context.get_qqbot_manager().bot
pkg.utils.context.set_qqbot_manager(self)
# Caution: 注册新的事件处理器之后请务必在unsubscribe_all中编写相应的取消订阅代码
@self.bot.on(FriendMessage)
async def on_friend_message(event: FriendMessage):
def friend_message_handler(event: FriendMessage):
# 触发事件
args = {
"launcher_type": "person",
"launcher_id": event.sender.id,
"sender_id": event.sender.id,
"message_chain": event.message_chain,
}
plugin_event = plugin_host.emit(plugin_models.PersonMessageReceived, **args)
if plugin_event.is_prevented_default():
return
self.on_person_message(event)
go(friend_message_handler, (event,))
@self.bot.on(StrangerMessage)
async def on_stranger_message(event: StrangerMessage):
def stranger_message_handler(event: StrangerMessage):
# 触发事件
args = {
"launcher_type": "person",
"launcher_id": event.sender.id,
"sender_id": event.sender.id,
"message_chain": event.message_chain,
}
plugin_event = plugin_host.emit(plugin_models.PersonMessageReceived, **args)
if plugin_event.is_prevented_default():
return
self.on_person_message(event)
go(stranger_message_handler, (event,))
@self.bot.on(GroupMessage)
async def on_group_message(event: GroupMessage):
def group_message_handler(event: GroupMessage):
# 触发事件
args = {
"launcher_type": "group",
"launcher_id": event.group.id,
"sender_id": event.sender.id,
"message_chain": event.message_chain,
}
plugin_event = plugin_host.emit(plugin_models.GroupMessageReceived, **args)
if plugin_event.is_prevented_default():
return
self.on_group_message(event)
go(group_message_handler, (event,))
def unsubscribe_all():
2023-01-02 13:06:48 +08:00
"""取消所有订阅
用于在热重载流程中卸载所有事件处理器
"""
assert isinstance(self.bot, Mirai)
bus = self.bot.bus
assert isinstance(bus, mirai.models.bus.ModelEventBus)
bus.unsubscribe(FriendMessage, on_friend_message)
bus.unsubscribe(StrangerMessage, on_stranger_message)
bus.unsubscribe(GroupMessage, on_group_message)
self.unsubscribe_all = unsubscribe_all
def first_time_init(self, mirai_http_api_config: dict):
"""热重载后不再运行此函数"""
if 'adapter' not in mirai_http_api_config or mirai_http_api_config['adapter'] == "WebSocketAdapter":
bot = Mirai(
qq=mirai_http_api_config['qq'],
adapter=WebSocketAdapter(
verify_key=mirai_http_api_config['verifyKey'],
host=mirai_http_api_config['host'],
port=mirai_http_api_config['port']
)
2022-12-07 23:37:52 +08:00
)
elif mirai_http_api_config['adapter'] == "HTTPAdapter":
bot = Mirai(
qq=mirai_http_api_config['qq'],
adapter=HTTPAdapter(
verify_key=mirai_http_api_config['verifyKey'],
host=mirai_http_api_config['host'],
port=mirai_http_api_config['port']
)
)
2022-12-07 23:37:52 +08:00
2022-12-26 19:37:25 +08:00
else:
raise Exception("未知的适配器类型")
2022-12-07 23:37:52 +08:00
self.bot = bot
def send(self, event, msg, check_quote=True):
config = pkg.utils.context.get_config()
asyncio.run(
2023-01-01 18:27:34 +08:00
self.bot.send(event, msg, quote=True if hasattr(config,
"quote_origin") and config.quote_origin and check_quote else False))
2022-12-11 16:10:12 +08:00
# 私聊消息处理
def on_person_message(self, event: MessageEvent):
2022-12-07 23:37:52 +08:00
reply = ''
if event.sender.id == self.bot.qq:
pass
else:
if Image in event.message_chain:
pass
else:
# 超时则重试,重试超过次数则放弃
failed = 0
for i in range(self.retry):
try:
reply = processor.process_message('person', event.sender.id, str(event.message_chain),
event.message_chain,
event.sender.id)
break
except FunctionTimedOut:
logging.warning("person_{}: 超时,重试中({})".format(event.sender.id, i))
2022-12-13 16:04:51 +08:00
pkg.openai.session.get_session('person_{}'.format(event.sender.id)).release_response_lock()
2023-01-08 14:49:23 +08:00
if "person_{}".format(event.sender.id) in pkg.qqbot.process.processing:
pkg.qqbot.process.processing.remove('person_{}'.format(event.sender.id))
failed += 1
continue
2022-12-07 23:37:52 +08:00
if failed == self.retry:
2022-12-13 16:04:51 +08:00
pkg.openai.session.get_session('person_{}'.format(event.sender.id)).release_response_lock()
2022-12-13 00:45:56 +08:00
self.notify_admin("{} 请求超时".format("person_{}".format(event.sender.id)))
reply = ["[bot]err:请求超时"]
2022-12-07 23:37:52 +08:00
if reply:
return self.send(event, reply, check_quote=False)
2022-12-07 23:37:52 +08:00
2022-12-11 16:10:12 +08:00
# 群消息处理
def on_group_message(self, event: GroupMessage):
2022-12-07 23:37:52 +08:00
reply = ''
def process(text=None) -> str:
replys = ""
if At(self.bot.qq) in event.message_chain:
event.message_chain.remove(At(self.bot.qq))
2022-12-08 12:06:04 +08:00
# 超时则重试,重试超过次数则放弃
failed = 0
for i in range(self.retry):
try:
2022-12-26 19:37:25 +08:00
replys = processor.process_message('group', event.group.id,
str(event.message_chain).strip() if text is None else text,
event.message_chain,
event.sender.id)
break
except FunctionTimedOut:
logging.warning("group_{}: 超时,重试中({})".format(event.group.id, i))
pkg.openai.session.get_session('group_{}'.format(event.group.id)).release_response_lock()
2023-01-25 22:51:09 +08:00
if "group_{}".format(event.group.id) in pkg.qqbot.process.processing:
pkg.qqbot.process.processing.remove('group_{}'.format(event.group.id))
failed += 1
continue
2022-12-07 23:37:52 +08:00
if failed == self.retry:
pkg.openai.session.get_session('group_{}'.format(event.group.id)).release_response_lock()
self.notify_admin("{} 请求超时".format("group_{}".format(event.group.id)))
replys = ["[bot]err:请求超时"]
return replys
if Image in event.message_chain:
pass
elif At(self.bot.qq) not in event.message_chain:
check, result = check_response_rule(str(event.message_chain).strip())
if check:
2022-12-19 17:25:09 +08:00
reply = process(result.strip())
else:
# 直接调用
reply = process()
2022-12-07 23:37:52 +08:00
if reply:
return self.send(event, reply)
2022-12-07 23:37:52 +08:00
2022-12-11 16:10:12 +08:00
# 通知系统管理员
def notify_admin(self, message: str):
config = pkg.utils.context.get_config()
2022-12-12 22:11:32 +08:00
if hasattr(config, "admin_qq") and config.admin_qq != 0:
logging.info("通知管理员:{}".format(message))
send_task = self.bot.send_friend_message(config.admin_qq, "[bot]{}".format(message))
threading.Thread(target=asyncio.run, args=(send_task,)).start()
2023-02-10 19:03:25 +08:00
def notify_admin_message_chain(self, message):
config = pkg.utils.context.get_config()
if hasattr(config, "admin_qq") and config.admin_qq != 0:
logging.info("通知管理员:{}".format(message))
send_task = self.bot.send_friend_message(config.admin_qq, message)
threading.Thread(target=asyncio.run, args=(send_task,)).start()