QChatGPT/pkg/qqbot/manager.py

432 lines
16 KiB
Python
Raw Normal View History

2024-01-23 20:55:20 +08:00
from __future__ import annotations
2022-12-11 17:17:33 +08:00
import json
import os
2023-11-13 21:59:23 +08:00
import logging
2023-11-26 23:04:14 +08:00
from mirai import At, GroupMessage, MessageEvent, StrangerMessage, \
FriendMessage, Image, MessageChain, Plain
2023-11-13 21:59:23 +08:00
import func_timeout
2022-12-09 16:18:25 +08:00
2023-11-13 21:59:23 +08:00
from ..openai import session as openai_session
2022-12-07 23:37:52 +08:00
2023-11-13 21:59:23 +08:00
from ..qqbot import filter as qqbot_filter
from ..qqbot import process as processor
from ..utils import context
from ..plugin import host as plugin_host
from ..plugin import models as plugin_models
2023-04-07 13:20:57 +08:00
import tips as tips_custom
2023-11-13 21:59:23 +08:00
from ..qqbot import adapter as msadapter
2024-01-23 20:55:20 +08:00
from ..boot import app
# 检查消息是否符合泛响应匹配机制
def check_response_rule(group_id:int, text: str):
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
2022-12-19 17:15:17 +08:00
2023-11-26 23:04:14 +08:00
rules = config['response_rules']
# 检查是否有特定规则
2023-11-26 23:04:14 +08:00
if 'prefix' not in config['response_rules']:
if str(group_id) in config['response_rules']:
rules = config['response_rules'][str(group_id)]
else:
2023-11-26 23:04:14 +08:00
rules = config['response_rules']['default']
# 检查前缀匹配
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, ""
def response_at(group_id: int):
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
2023-11-26 23:04:14 +08:00
use_response_rule = config['response_rules']
# 检查是否有特定规则
2023-11-26 23:04:14 +08:00
if 'prefix' not in config['response_rules']:
if str(group_id) in config['response_rules']:
use_response_rule = config['response_rules'][str(group_id)]
else:
2023-11-26 23:04:14 +08:00
use_response_rule = config['response_rules']['default']
if 'at' not in use_response_rule:
return True
return use_response_rule['at']
def random_responding(group_id):
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
2023-11-26 23:04:14 +08:00
use_response_rule = config['response_rules']
# 检查是否有特定规则
2023-11-26 23:04:14 +08:00
if 'prefix' not in config['response_rules']:
if str(group_id) in config['response_rules']:
use_response_rule = config['response_rules'][str(group_id)]
else:
2023-11-26 23:04:14 +08:00
use_response_rule = config['response_rules']['default']
if 'random_rate' in use_response_rule:
import random
return random.random() < use_response_rule['random_rate']
return False
2022-12-11 16:10:12 +08:00
# 控制QQ消息输入输出的类
2022-12-07 23:37:52 +08:00
class QQBotManager:
retry = 3
adapter: msadapter.MessageSourceAdapter = None
bot_account_id: int = 0
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
enable_private = True
enable_group = True
2023-01-07 16:50:34 +08:00
ban_person = []
ban_group = []
2024-01-23 20:55:20 +08:00
def __init__(self, first_time_init=True, ap: app.Application = None):
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
2023-11-26 23:04:14 +08:00
self.timeout = config['process_message_timeout']
self.retry = config['retry_times']
2023-01-02 13:06:48 +08:00
# 由于YiriMirai的bot对象是单例的且shutdown方法暂时无法使用
# 故只在第一次初始化时创建bot对象重载之后使用原bot对象
# 因此bot的配置不支持热重载
if first_time_init:
2023-11-26 23:04:14 +08:00
logging.debug("Use adapter:" + config['msg_source_adapter'])
if config['msg_source_adapter'] == 'yirimirai':
from pkg.qqbot.sources.yirimirai import YiriMiraiAdapter
2023-04-23 23:40:08 +08:00
2023-11-26 23:04:14 +08:00
mirai_http_api_config = config['mirai_http_api_config']
self.bot_account_id = config['mirai_http_api_config']['qq']
self.adapter = YiriMiraiAdapter(mirai_http_api_config)
2023-11-26 23:04:14 +08:00
elif config['msg_source_adapter'] == 'nakuru':
2023-04-23 23:40:08 +08:00
from pkg.qqbot.sources.nakuru import NakuruProjectAdapter
2023-11-26 23:04:14 +08:00
self.adapter = NakuruProjectAdapter(config['nakuru_config'])
2023-04-24 10:34:51 +08:00
self.bot_account_id = self.adapter.bot_account_id
else:
2023-11-13 21:59:23 +08:00
self.adapter = context.get_qqbot_manager().adapter
self.bot_account_id = context.get_qqbot_manager().bot_account_id
2023-12-21 18:51:10 +08:00
# 保存 account_id 到审计模块
from ..utils.center import apigroup
2023-12-22 12:38:27 +08:00
apigroup.APIGroup._runtime_info['account_id'] = "{}".format(self.bot_account_id)
2023-11-13 21:59:23 +08:00
context.set_qqbot_manager(self)
# 注册诸事件
# Caution: 注册新的事件处理器之后请务必在unsubscribe_all中编写相应的取消订阅代码
def on_friend_message(event: FriendMessage):
def friend_message_handler():
# 触发事件
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)
2023-11-13 21:59:23 +08:00
context.get_thread_ctl().submit_user_task(
friend_message_handler,
)
self.adapter.register_listener(
FriendMessage,
on_friend_message
)
def on_stranger_message(event: StrangerMessage):
def stranger_message_handler():
# 触发事件
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)
2023-11-13 21:59:23 +08:00
context.get_thread_ctl().submit_user_task(
stranger_message_handler,
)
2023-04-23 23:40:08 +08:00
# nakuru不区分好友和陌生人故仅为yirimirai注册陌生人事件
2023-11-26 23:04:14 +08:00
if config['msg_source_adapter'] == 'yirimirai':
2023-04-23 23:40:08 +08:00
self.adapter.register_listener(
StrangerMessage,
on_stranger_message
)
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)
2023-11-13 21:59:23 +08:00
context.get_thread_ctl().submit_user_task(
group_message_handler,
event
)
self.adapter.register_listener(
GroupMessage,
on_group_message
)
def unsubscribe_all():
2023-01-02 13:06:48 +08:00
"""取消所有订阅
用于在热重载流程中卸载所有事件处理器
"""
self.adapter.unregister_listener(
FriendMessage,
on_friend_message
)
2023-11-26 23:04:14 +08:00
if config['msg_source_adapter'] == 'yirimirai':
2023-04-24 11:21:51 +08:00
self.adapter.unregister_listener(
StrangerMessage,
on_stranger_message
)
self.adapter.unregister_listener(
GroupMessage,
on_group_message
)
self.unsubscribe_all = unsubscribe_all
# 加载禁用列表
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))
2022-12-07 23:37:52 +08:00
if hasattr(banlist, "enable_private"):
self.enable_private = banlist.enable_private
if hasattr(banlist, "enable_group"):
self.enable_group = banlist.enable_group
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
if os.path.exists("sensitive.json") \
2023-11-26 23:04:14 +08:00
and config['sensitive_word_filter'] is not None \
and config['sensitive_word_filter']:
with open("sensitive.json", "r", encoding="utf-8") as f:
sensitive_json = json.load(f)
2023-11-13 21:59:23 +08:00
self.reply_filter = qqbot_filter.ReplyFilter(
sensitive_words=sensitive_json['words'],
mask=sensitive_json['mask'] if 'mask' in sensitive_json else '*',
mask_word=sensitive_json['mask_word'] if 'mask_word' in sensitive_json else ''
)
2022-12-26 19:37:25 +08:00
else:
2023-11-13 21:59:23 +08:00
self.reply_filter = qqbot_filter.ReplyFilter([])
2022-12-07 23:37:52 +08:00
def send(self, event, msg, check_quote=True, check_at_sender=True):
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
2023-11-26 23:04:14 +08:00
if check_at_sender and config['at_sender']:
msg.insert(
0,
Plain(" \n")
)
# 当回复的正文中包含换行时quote可能会自带at此时就不再单独添加at只添加换行
2023-11-26 23:04:14 +08:00
if "\n" not in str(msg[1]) or config['msg_source_adapter'] == 'nakuru':
msg.insert(
0,
At(
event.sender.id
)
)
self.adapter.reply_message(
event,
msg,
2023-11-26 23:04:14 +08:00
quote_origin=True if 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 = ''
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
if not self.enable_private:
logging.debug("已在banlist.py中禁用所有私聊")
elif event.sender.id == self.bot_account_id:
2022-12-07 23:37:52 +08:00
pass
else:
if Image in event.message_chain:
pass
else:
# 超时则重试,重试超过次数则放弃
failed = 0
for i in range(self.retry):
try:
2023-11-26 23:04:14 +08:00
@func_timeout.func_set_timeout(config['process_message_timeout'])
def time_ctrl_wrapper():
reply = processor.process_message('person', event.sender.id, str(event.message_chain),
event.message_chain,
event.sender.id)
return reply
reply = time_ctrl_wrapper()
break
2023-11-13 21:59:23 +08:00
except func_timeout.FunctionTimedOut:
logging.warning("person_{}: 超时,重试中({})".format(event.sender.id, i))
2023-11-13 21:59:23 +08:00
openai_session.get_session('person_{}'.format(event.sender.id)).release_response_lock()
if "person_{}".format(event.sender.id) in processor.processing:
processor.processing.remove('person_{}'.format(event.sender.id))
failed += 1
continue
2022-12-07 23:37:52 +08:00
if failed == self.retry:
2023-11-13 21:59:23 +08:00
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)))
2023-04-07 13:23:58 +08:00
reply = [tips_custom.reply_message]
2022-12-07 23:37:52 +08:00
if reply:
return self.send(event, reply, check_quote=False, check_at_sender=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 = ''
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
def process(text=None) -> str:
replys = ""
if At(self.bot_account_id) in event.message_chain:
event.message_chain.remove(At(self.bot_account_id))
2022-12-08 12:06:04 +08:00
# 超时则重试,重试超过次数则放弃
failed = 0
for i in range(self.retry):
try:
2023-11-26 23:04:14 +08:00
@func_timeout.func_set_timeout(config['process_message_timeout'])
def time_ctrl_wrapper():
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)
return replys
replys = time_ctrl_wrapper()
break
2023-11-13 21:59:23 +08:00
except func_timeout.FunctionTimedOut:
logging.warning("group_{}: 超时,重试中({})".format(event.group.id, i))
2023-11-13 21:59:23 +08:00
openai_session.get_session('group_{}'.format(event.group.id)).release_response_lock()
if "group_{}".format(event.group.id) in processor.processing:
processor.processing.remove('group_{}'.format(event.group.id))
failed += 1
continue
2022-12-07 23:37:52 +08:00
if failed == self.retry:
2023-11-13 21:59:23 +08:00
openai_session.get_session('group_{}'.format(event.group.id)).release_response_lock()
self.notify_admin("{} 请求超时".format("group_{}".format(event.group.id)))
2023-04-07 13:23:58 +08:00
replys = [tips_custom.replys_message]
return replys
if not self.enable_group:
logging.debug("已在banlist.py中禁用所有群聊")
elif Image in event.message_chain:
pass
else:
if At(self.bot_account_id) in event.message_chain and response_at(event.group.id):
# 直接调用
reply = process()
else:
check, result = check_response_rule(event.group.id, str(event.message_chain).strip())
if check:
reply = process(result.strip())
# 检查是否随机响应
elif random_responding(event.group.id):
logging.info("随机响应group_{}消息".format(event.group.id))
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):
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
if config['admin_qq'] != 0 and config['admin_qq'] != []:
logging.info("通知管理员:{}".format(message))
2023-11-26 23:04:14 +08:00
if type(config['admin_qq']) == int:
self.adapter.send_message(
"person",
2023-11-26 23:04:14 +08:00
config['admin_qq'],
MessageChain([Plain("[bot]{}".format(message))])
)
2023-02-25 15:39:31 +08:00
else:
2023-11-26 23:04:14 +08:00
for adm in config['admin_qq']:
self.adapter.send_message(
"person",
adm,
MessageChain([Plain("[bot]{}".format(message))])
)
2023-02-10 19:03:25 +08:00
def notify_admin_message_chain(self, message):
2023-11-26 23:04:14 +08:00
config = context.get_config_manager().data
if config['admin_qq'] != 0 and config['admin_qq'] != []:
2023-02-10 19:03:25 +08:00
logging.info("通知管理员:{}".format(message))
2023-11-26 23:04:14 +08:00
if type(config['admin_qq']) == int:
self.adapter.send_message(
"person",
2023-11-26 23:04:14 +08:00
config['admin_qq'],
message
)
2023-02-25 15:39:31 +08:00
else:
2023-11-26 23:04:14 +08:00
for adm in config['admin_qq']:
self.adapter.send_message(
"person",
adm,
message
)