2020-04-07 21:58:10 +08:00
|
|
|
|
import re
|
2018-07-30 23:41:19 +08:00
|
|
|
|
import asyncio
|
2020-04-07 21:58:10 +08:00
|
|
|
|
from typing import Callable, Iterable
|
2018-06-15 06:58:24 +08:00
|
|
|
|
|
2020-03-15 22:48:22 +08:00
|
|
|
|
from aiocqhttp import Event as CQEvent
|
2020-04-07 21:58:10 +08:00
|
|
|
|
from aiocqhttp.message import escape, unescape, Message, MessageSegment
|
2018-06-15 06:58:24 +08:00
|
|
|
|
|
2018-07-04 09:28:31 +08:00
|
|
|
|
from . import NoneBot
|
2018-06-27 22:50:01 +08:00
|
|
|
|
from .log import logger
|
2018-07-02 16:54:29 +08:00
|
|
|
|
from .natural_language import handle_natural_language
|
2020-04-07 21:58:10 +08:00
|
|
|
|
from .command import handle_command, SwitchException
|
|
|
|
|
from .plugin import PluginManager
|
2018-06-15 06:58:24 +08:00
|
|
|
|
|
2018-07-30 23:41:19 +08:00
|
|
|
|
_message_preprocessors = set()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def message_preprocessor(func: Callable) -> Callable:
|
|
|
|
|
_message_preprocessors.add(func)
|
|
|
|
|
return func
|
|
|
|
|
|
2018-06-27 16:36:40 +08:00
|
|
|
|
|
2020-04-11 14:56:39 +08:00
|
|
|
|
class CanceledException(Exception):
|
|
|
|
|
"""
|
|
|
|
|
Raised by message_preprocessor indicating that
|
|
|
|
|
the bot should ignore the message
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
def __init__(self, reason):
|
|
|
|
|
"""
|
|
|
|
|
:param reason: reason to ignore the message
|
|
|
|
|
"""
|
|
|
|
|
self.reason = reason
|
|
|
|
|
|
|
|
|
|
|
2020-03-15 22:48:22 +08:00
|
|
|
|
async def handle_message(bot: NoneBot, event: CQEvent) -> None:
|
|
|
|
|
_log_message(event)
|
2018-07-21 00:46:34 +08:00
|
|
|
|
|
2020-03-15 22:48:22 +08:00
|
|
|
|
assert isinstance(event.message, Message)
|
|
|
|
|
if not event.message:
|
2020-04-07 21:58:10 +08:00
|
|
|
|
event.message.append(MessageSegment.text('')) # type: ignore
|
2019-01-06 21:38:48 +08:00
|
|
|
|
|
2020-04-11 14:56:39 +08:00
|
|
|
|
raw_to_me = event.get('to_me', False)
|
|
|
|
|
_check_at_me(bot, event)
|
|
|
|
|
_check_calling_me_nickname(bot, event)
|
|
|
|
|
event['to_me'] = raw_to_me or event['to_me']
|
|
|
|
|
|
2018-07-30 23:41:19 +08:00
|
|
|
|
coros = []
|
2020-04-07 21:58:10 +08:00
|
|
|
|
plugin_manager = PluginManager()
|
2020-03-15 22:48:22 +08:00
|
|
|
|
for preprocessor in _message_preprocessors:
|
2020-04-07 21:58:10 +08:00
|
|
|
|
coros.append(preprocessor(bot, event, plugin_manager))
|
2018-07-30 23:41:19 +08:00
|
|
|
|
if coros:
|
2020-04-11 14:56:39 +08:00
|
|
|
|
try:
|
|
|
|
|
await asyncio.gather(*coros)
|
|
|
|
|
except CanceledException:
|
|
|
|
|
logger.info(f'Message {event["message_id"]} is ignored')
|
|
|
|
|
return
|
2018-06-15 06:58:24 +08:00
|
|
|
|
|
2018-07-24 23:59:45 +08:00
|
|
|
|
while True:
|
|
|
|
|
try:
|
2020-04-20 13:50:38 +08:00
|
|
|
|
handled = await handle_command(bot, event,
|
|
|
|
|
plugin_manager.cmd_manager)
|
2018-07-24 23:59:45 +08:00
|
|
|
|
break
|
|
|
|
|
except SwitchException as e:
|
|
|
|
|
# we are sure that there is no session existing now
|
2020-03-15 22:48:22 +08:00
|
|
|
|
event['message'] = e.new_message
|
|
|
|
|
event['to_me'] = True
|
2018-06-15 06:58:24 +08:00
|
|
|
|
if handled:
|
2020-03-15 22:48:22 +08:00
|
|
|
|
logger.info(f'Message {event.message_id} is handled as a command')
|
2018-06-27 16:36:40 +08:00
|
|
|
|
return
|
2018-06-30 10:58:56 +08:00
|
|
|
|
|
2020-04-20 13:50:38 +08:00
|
|
|
|
handled = await handle_natural_language(bot, event,
|
|
|
|
|
plugin_manager.nlp_manager)
|
2018-07-01 11:01:24 +08:00
|
|
|
|
if handled:
|
2020-03-15 22:48:22 +08:00
|
|
|
|
logger.info(f'Message {event.message_id} is handled '
|
2018-07-21 00:46:34 +08:00
|
|
|
|
f'as natural language')
|
2018-07-01 11:01:24 +08:00
|
|
|
|
return
|
2018-07-21 00:46:34 +08:00
|
|
|
|
|
|
|
|
|
|
2020-03-15 22:48:22 +08:00
|
|
|
|
def _check_at_me(bot: NoneBot, event: CQEvent) -> None:
|
|
|
|
|
if event.detail_type == 'private':
|
|
|
|
|
event['to_me'] = True
|
2019-02-21 21:41:17 +08:00
|
|
|
|
else:
|
|
|
|
|
# group or discuss
|
2020-03-15 22:48:22 +08:00
|
|
|
|
event['to_me'] = False
|
|
|
|
|
at_me_seg = MessageSegment.at(event.self_id)
|
2019-02-21 21:41:17 +08:00
|
|
|
|
|
|
|
|
|
# check the first segment
|
2020-03-15 22:48:22 +08:00
|
|
|
|
first_msg_seg = event.message[0]
|
2019-02-21 21:41:17 +08:00
|
|
|
|
if first_msg_seg == at_me_seg:
|
2020-03-15 22:48:22 +08:00
|
|
|
|
event['to_me'] = True
|
|
|
|
|
del event.message[0]
|
2019-02-21 21:41:17 +08:00
|
|
|
|
|
2020-03-15 22:48:22 +08:00
|
|
|
|
if not event['to_me']:
|
2019-02-21 21:41:17 +08:00
|
|
|
|
# check the last segment
|
|
|
|
|
i = -1
|
2020-03-15 22:48:22 +08:00
|
|
|
|
last_msg_seg = event.message[i]
|
2019-02-21 21:41:17 +08:00
|
|
|
|
if last_msg_seg.type == 'text' and \
|
|
|
|
|
not last_msg_seg.data['text'].strip() and \
|
2020-03-15 22:48:22 +08:00
|
|
|
|
len(event.message) >= 2:
|
2019-02-21 21:41:17 +08:00
|
|
|
|
i -= 1
|
2020-03-15 22:48:22 +08:00
|
|
|
|
last_msg_seg = event.message[i]
|
2019-02-21 21:41:17 +08:00
|
|
|
|
|
|
|
|
|
if last_msg_seg == at_me_seg:
|
2020-03-15 22:48:22 +08:00
|
|
|
|
event['to_me'] = True
|
|
|
|
|
del event.message[i:]
|
2019-02-21 21:41:17 +08:00
|
|
|
|
|
2020-03-15 22:48:22 +08:00
|
|
|
|
if not event.message:
|
|
|
|
|
event.message.append(MessageSegment.text(''))
|
2019-02-21 21:41:17 +08:00
|
|
|
|
|
|
|
|
|
|
2020-03-15 22:48:22 +08:00
|
|
|
|
def _check_calling_me_nickname(bot: NoneBot, event: CQEvent) -> None:
|
|
|
|
|
first_msg_seg = event.message[0]
|
2019-02-21 21:41:17 +08:00
|
|
|
|
if first_msg_seg.type != 'text':
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
first_text = first_msg_seg.data['text']
|
|
|
|
|
|
|
|
|
|
if bot.config.NICKNAME:
|
|
|
|
|
# check if the user is calling me with my nickname
|
|
|
|
|
if isinstance(bot.config.NICKNAME, str) or \
|
|
|
|
|
not isinstance(bot.config.NICKNAME, Iterable):
|
|
|
|
|
nicknames = (bot.config.NICKNAME,)
|
|
|
|
|
else:
|
|
|
|
|
nicknames = filter(lambda n: n, bot.config.NICKNAME)
|
|
|
|
|
nickname_regex = '|'.join(nicknames)
|
2020-04-20 13:50:38 +08:00
|
|
|
|
m = re.search(rf'^({nickname_regex})([\s,,]*|$)', first_text,
|
|
|
|
|
re.IGNORECASE)
|
2019-02-21 21:41:17 +08:00
|
|
|
|
if m:
|
|
|
|
|
nickname = m.group(1)
|
|
|
|
|
logger.debug(f'User is calling me {nickname}')
|
2020-03-15 22:48:22 +08:00
|
|
|
|
event['to_me'] = True
|
2019-02-21 21:41:17 +08:00
|
|
|
|
first_msg_seg.data['text'] = first_text[m.end():]
|
|
|
|
|
|
|
|
|
|
|
2020-03-15 22:48:22 +08:00
|
|
|
|
def _log_message(event: CQEvent) -> None:
|
|
|
|
|
msg_from = str(event.user_id)
|
|
|
|
|
if event.detail_type == 'group':
|
|
|
|
|
msg_from += f'@[群:{event.group_id}]'
|
|
|
|
|
elif event.detail_type == 'discuss':
|
|
|
|
|
msg_from += f'@[讨论组:{event.discuss_id}]'
|
|
|
|
|
logger.info(f'Self: {event.self_id}, '
|
|
|
|
|
f'Message {event.message_id} from {msg_from}: '
|
2020-03-16 16:55:56 +08:00
|
|
|
|
f'{repr(str(event.message))}')
|