melobot.handle.register 源代码

from asyncio import Lock
from functools import partial, wraps

from typing_extensions import Callable, Sequence, cast

from ..adapter.model import Event, TextEvent
from ..ctx import FlowCtx, ParseArgsCtx
from ..di import inject_deps
from ..session.base import enter_session
from ..session.option import DefaultRule, Rule
from ..typ._enum import LogicMode
from ..typ.base import AsyncCallable, SyncOrAsyncCallable
from ..utils.check import Checker, checker_join
from ..utils.common import get_obj_name
from ..utils.match import (
    ContainMatcher,
    EndMatcher,
    FullMatcher,
    Matcher,
    RegexMatcher,
    StartMatcher,
)
from ..utils.parse import AbstractParseArgs, CmdArgFormatter, CmdParser, Parser
from .base import Flow, no_deps_node


[文档] class FlowDecorator:
[文档] def __init__( self, checker: Checker | None | Callable[[Event], bool] = None, matcher: Matcher | None = None, parser: Parser | None = None, priority: int = 0, block: bool = False, temp: bool = False, decos: Sequence[Callable[[Callable], Callable]] | None = None, rule: Rule[Event] | None = None, ) -> None: """处理流装饰器 :param checker: 检查器 :param matcher: 匹配器 :param parser: 解析器 :param priority: 优先级 :param block: 是否阻断向低优先级传播 :param temp: 是否临时使用(处理一次事件后停用) :param decos: 装饰器组 :param rule: 会话规则 """ self.checker: Checker | None if callable(checker): self.checker = Checker.new(checker) else: self.checker = checker self.matcher = matcher self.parser = parser self.priority = priority self.block = block self.decos = decos self.rule = rule self._temp = temp self._invalid = False self._lock = Lock() self._flow: Flow
async def _pre_process(self, event: Event) -> tuple[bool, AbstractParseArgs | None]: if self.checker: status = await self.checker.check(event) if not status: return (False, None) args: AbstractParseArgs | None = None if isinstance(event, TextEvent): if self.matcher: status = await self.matcher.match(event.text) if not status: return (False, None) if self.parser: args = await self.parser.parse(event.text) if args: return (True, args) return (False, None) return (True, None) async def _process( self, func: AsyncCallable[..., bool | None], event: Event, args: AbstractParseArgs | None, ) -> bool | None: event.spread = not self.block parse_args_ctx = ParseArgsCtx() if args is not None: args_token = parse_args_ctx.add(args) else: args_token = None try: if self.rule is not None: async with enter_session(self.rule): return await func() return await func() finally: if args_token: parse_args_ctx.remove(args_token) async def auto_flow_wrapped(self, func: AsyncCallable[..., bool | None]) -> bool | None: if self._invalid: self._flow.dismiss() return None event = FlowCtx().get().completion.event if not self._temp: passed, args = await self._pre_process(event) if not passed: return None return await self._process(func, event, args) async with self._lock: if self._invalid: self._flow.dismiss() return None passed, args = await self._pre_process(event) if not passed: return None self._invalid = True return await self._process(func, event, args)
[文档] def __call__(self, func: SyncOrAsyncCallable[..., bool | None]) -> Flow: """实现了此方法,因此可被当作装饰器使用 :param func: 被装饰的函数 :return: 处理流对象 """ func = inject_deps(func) if self.decos is not None: for deco in reversed(self.decos): func = deco(func) n = no_deps_node(wraps(func)(partial(self.auto_flow_wrapped, func))) n.name = get_obj_name(func, otype="callable") self._flow = Flow(f"OneBotV11Flow[{n.name}]", (n,), priority=self.priority) return self._flow
[文档] def on_event( checker: Checker | None | Callable[[Event], bool] = None, matcher: Matcher | None = None, parser: Parser | None = None, priority: int = 0, block: bool = False, temp: bool = False, decos: Sequence[Callable[[Callable], Callable]] | None = None, rule: Rule[Event] | None = None, ) -> FlowDecorator: """绑定任意事件的处理流装饰方法 在前期的教程中,处理流装饰方法也称为绑定方法 :param checker: 检查器 :param matcher: 匹配器 :param parser: 解析器 :param priority: 优先级 :param block: 是否阻断向低优先级传播 :param temp: 是否临时使用(处理一次事件后停用) :param decos: 装饰器组 :param rule: 会话规则 :return: 处理流装饰器 """ return FlowDecorator(checker, matcher, parser, priority, block, temp, decos, rule)
[文档] def on_text( checker: Checker | None | Callable[[TextEvent], bool] = None, matcher: Matcher | None = None, parser: Parser | None = None, priority: int = 0, block: bool = False, temp: bool = False, decos: Sequence[Callable[[Callable], Callable]] | None = None, legacy_session: bool = False, ) -> FlowDecorator: """绑定文本事件的处理流装饰方法 在前期的教程中,处理流装饰方法也称为绑定方法 :param checker: 检查器 :param matcher: 匹配器 :param parser: 解析器 :param priority: 优先级 :param block: 是否阻断向低优先级传播 :param temp: 是否临时使用(处理一次事件后停用) :param decos: 装饰器组 :param legacy_session: 是否自动启用传统会话 :return: 处理流装饰器 """ if legacy_session: rule = DefaultRule() else: rule = None return on_event( checker_join(lambda e: isinstance(e, TextEvent), checker), # type: ignore[arg-type] matcher, parser, priority, block, temp, decos, cast(Rule[Event] | None, rule), )
[文档] def on_command( cmd_start: str | list[str], cmd_sep: str | list[str], targets: str | list[str], fmtters: list[CmdArgFormatter | None] | None = None, checker: Checker | None | Callable[[TextEvent], bool] = None, matcher: Matcher | None = None, priority: int = 0, block: bool = False, temp: bool = False, decos: Sequence[Callable[[Callable], Callable]] | None = None, legacy_session: bool = False, ) -> FlowDecorator: """绑定文本事件的“命令解析”处理流装饰方法 在前期的教程中,处理流装饰方法也称为绑定方法 :param cmd_start: 命令起始符 :param cmd_sep: 命令分隔符 :param targets: 解析匹配的命令名 :param fmtters: 命令参数格式化器 :param checker: 检查器 :param matcher: 匹配器 :param priority: 优先级 :param block: 是否阻断向低优先级传播 :param temp: 是否临时使用(处理一次事件后停用) :param decos: 装饰器组 :param legacy_session: 是否自动启用传统会话 :return: 处理流装饰器 """ return on_text( checker, matcher, CmdParser(cmd_start=cmd_start, cmd_sep=cmd_sep, targets=targets, fmtters=fmtters), priority, block, temp, decos, legacy_session, )
[文档] def on_start_match( target: str | list[str], logic_mode: LogicMode = LogicMode.OR, checker: Checker | Callable[[TextEvent], bool] | None = None, parser: Parser | None = None, priority: int = 0, block: bool = False, temp: bool = False, decos: Sequence[Callable[[Callable], Callable]] | None = None, legacy_session: bool = False, ) -> FlowDecorator: """绑定文本事件的“字符串起始匹配”处理流装饰方法 在前期的教程中,处理流装饰方法也称为绑定方法 :param target: 匹配的目标字符串 :param logic_mode: 匹配的逻辑模式 :param checker: 检查器 :param parser: 解析器 :param priority: 优先级 :param block: 是否阻断向低优先级传播 :param temp: 是否临时使用(处理一次事件后停用) :param decos: 装饰器组 :param legacy_session: 是否自动启用传统会话 :return: 处理流装饰器 """ return on_text( checker, StartMatcher(target, logic_mode), parser, priority, block, temp, decos, legacy_session, )
[文档] def on_contain_match( target: str | list[str], logic_mode: LogicMode = LogicMode.OR, checker: Checker | Callable[[TextEvent], bool] | None = None, parser: Parser | None = None, priority: int = 0, block: bool = False, temp: bool = False, decos: Sequence[Callable[[Callable], Callable]] | None = None, legacy_session: bool = False, ) -> FlowDecorator: """绑定文本事件的“字符串包含匹配”处理流装饰方法 在前期的教程中,处理流装饰方法也称为绑定方法 :param target: 匹配的目标字符串 :param logic_mode: 匹配的逻辑模式 :param checker: 检查器 :param parser: 解析器 :param priority: 优先级 :param block: 是否阻断向低优先级传播 :param temp: 是否临时使用(处理一次事件后停用) :param decos: 装饰器组 :param legacy_session: 是否自动启用传统会话 :return: 处理流装饰器 """ return on_text( checker, ContainMatcher(target, logic_mode), parser, priority, block, temp, decos, legacy_session, )
[文档] def on_full_match( target: str | list[str], logic_mode: LogicMode = LogicMode.OR, checker: Checker | Callable[[TextEvent], bool] | None = None, parser: Parser | None = None, priority: int = 0, block: bool = False, temp: bool = False, decos: Sequence[Callable[[Callable], Callable]] | None = None, legacy_session: bool = False, ) -> FlowDecorator: """绑定文本事件的“字符串完整匹配”处理流装饰方法 在前期的教程中,处理流装饰方法也称为绑定方法 :param target: 匹配的目标字符串 :param logic_mode: 匹配的逻辑模式 :param checker: 检查器 :param parser: 解析器 :param priority: 优先级 :param block: 是否阻断向低优先级传播 :param temp: 是否临时使用(处理一次事件后停用) :param decos: 装饰器组 :param legacy_session: 是否自动启用传统会话 :return: 处理流装饰器 """ return on_text( checker, FullMatcher(target, logic_mode), parser, priority, block, temp, decos, legacy_session, )
[文档] def on_end_match( target: str | list[str], logic_mode: LogicMode = LogicMode.OR, checker: Checker | Callable[[TextEvent], bool] | None = None, parser: Parser | None = None, priority: int = 0, block: bool = False, temp: bool = False, decos: Sequence[Callable[[Callable], Callable]] | None = None, legacy_session: bool = False, ) -> FlowDecorator: """绑定文本事件的“字符串末尾匹配”处理流装饰方法 在前期的教程中,处理流装饰方法也称为绑定方法 :param target: 匹配的目标字符串 :param logic_mode: 匹配的逻辑模式 :param checker: 检查器 :param parser: 解析器 :param priority: 优先级 :param block: 是否阻断向低优先级传播 :param temp: 是否临时使用(处理一次事件后停用) :param decos: 装饰器组 :param legacy_session: 是否自动启用传统会话 :return: 处理流装饰器 """ return on_text( checker, EndMatcher(target, logic_mode), parser, priority, block, temp, decos, legacy_session, )
[文档] def on_regex_match( target: str, logic_mode: LogicMode = LogicMode.OR, checker: Checker | Callable[[TextEvent], bool] | None = None, parser: Parser | None = None, priority: int = 0, block: bool = False, temp: bool = False, decos: Sequence[Callable[[Callable], Callable]] | None = None, legacy_session: bool = False, ) -> FlowDecorator: """绑定文本事件的“正则匹配”处理流装饰方法 在前期的教程中,处理流装饰方法也称为绑定方法 :param target: 匹配的目标字符串 :param logic_mode: 匹配的逻辑模式 :param checker: 检查器 :param parser: 解析器 :param priority: 优先级 :param block: 是否阻断向低优先级传播 :param temp: 是否临时使用(处理一次事件后停用) :param decos: 装饰器组 :param legacy_session: 是否自动启用传统会话 :return: 处理流装饰器 """ return on_text( checker, RegexMatcher(target, logic_mode), parser, priority, block, temp, decos, legacy_session, )