import asyncio from datetime import datetime from nonebot.log import logger from nonebot.rule import TrieRule from nonebot.utils import escape_tag from nonebot.matcher import matchers from nonebot.typing import Set, Type, Union, NoReturn from nonebot.typing import Bot, Event, Matcher, PreProcessor from nonebot.exception import IgnoredException, ExpiredException from nonebot.exception import StopPropagation, _ExceptionContainer _event_preprocessors: Set[PreProcessor] = set() def event_preprocessor(func: PreProcessor) -> PreProcessor: _event_preprocessors.add(func) return func async def _run_matcher(Matcher: Type[Matcher], bot: Bot, event: Event, state: dict) -> Union[None, NoReturn]: if Matcher.expire_time and datetime.now() > Matcher.expire_time: raise _ExceptionContainer([ExpiredException]) try: if not await Matcher.check_perm( bot, event) or not await Matcher.check_rule(bot, event, state): return except Exception as e: logger.opt(colors=True, exception=e).error( f"Rule check failed for {Matcher}.") return logger.info(f"Event will be handled by {Matcher}") matcher = Matcher() # TODO: BeforeMatcherRun try: logger.debug(f"Running matcher {matcher}") await matcher.run(bot, event, state) except Exception as e: logger.opt(colors=True, exception=e).error( f"Running matcher {matcher} failed." ) exceptions = [] if Matcher.temp: exceptions.append(ExpiredException) if Matcher.block: exceptions.append(StopPropagation) if exceptions: raise _ExceptionContainer(exceptions) async def handle_event(bot: Bot, event: Event): log_msg = f"{bot.type.upper()} | {event.self_id} [{event.name}]: " if event.type == "message": log_msg += f"Message {event.id} from " log_msg += str(event.user_id) if event.detail_type == "group": log_msg += f"@[群:{event.group_id}]:" log_msg += ' "' + "".join( map( lambda x: escape_tag(str(x)) if x.type == "text" else f"{escape_tag(str(x))}", event.message)) + '"' # type: ignore elif event.type == "notice": log_msg += f"Notice {event.raw_event}" elif event.type == "request": log_msg += f"Request {event.raw_event}" elif event.type == "meta_event": log_msg += f"MetaEvent {event.detail_type}" logger.opt(colors=True).info(log_msg) coros = [] state = {} for preprocessor in _event_preprocessors: coros.append(preprocessor(bot, event, state)) if coros: try: logger.debug("Running PreProcessors...") await asyncio.gather(*coros) except IgnoredException: logger.opt( colors=True).info(f"Event {event.name} is ignored") return # Trie Match _, _ = TrieRule.get_value(bot, event, state) break_flag = False for priority in sorted(matchers.keys()): if break_flag: break pending_tasks = [ _run_matcher(matcher, bot, event, state.copy()) for matcher in matchers[priority] ] logger.debug(f"Checking for matchers in priority {priority}...") results = await asyncio.gather(*pending_tasks, return_exceptions=True) i = 0 for index, result in enumerate(results): if isinstance(result, _ExceptionContainer): e_list = result.exceptions if StopPropagation in e_list: if not break_flag: break_flag = True logger.debug("Stop event propagation") if ExpiredException in e_list: logger.debug( f"Matcher {matchers[priority][index - i]} will be removed." ) del matchers[priority][index - i] i += 1