* fix: 修复 astrbot.core.star 等包下的 type checking error * refactor: improve type checking and annotations * chore: ruff format
102 lines
3.4 KiB
Python
102 lines
3.4 KiB
Python
import inspect
|
||
import traceback
|
||
import typing as T
|
||
from astrbot import logger
|
||
from astrbot.core.star.star_handler import star_handlers_registry, EventType
|
||
from astrbot.core.star.star import star_map
|
||
from astrbot.core.message.message_event_result import MessageEventResult, CommandResult
|
||
from astrbot.core.platform.astr_message_event import AstrMessageEvent
|
||
|
||
|
||
async def call_handler(
|
||
event: AstrMessageEvent,
|
||
handler: T.Callable[..., T.Awaitable[T.Any]],
|
||
*args,
|
||
**kwargs,
|
||
) -> T.AsyncGenerator[T.Any, None]:
|
||
"""执行事件处理函数并处理其返回结果
|
||
|
||
该方法负责调用处理函数并处理不同类型的返回值。它支持两种类型的处理函数:
|
||
1. 异步生成器: 实现洋葱模型,每次 yield 都会将控制权交回上层
|
||
2. 协程: 执行一次并处理返回值
|
||
|
||
Args:
|
||
event (AstrMessageEvent): 事件对象
|
||
handler (Awaitable): 事件处理函数
|
||
|
||
Returns:
|
||
AsyncGenerator[None, None]: 异步生成器,用于在管道中传递控制流
|
||
"""
|
||
ready_to_call = None # 一个协程或者异步生成器
|
||
|
||
trace_ = None
|
||
|
||
try:
|
||
ready_to_call = handler(event, *args, **kwargs)
|
||
except TypeError:
|
||
logger.error("处理函数参数不匹配,请检查 handler 的定义。", exc_info=True)
|
||
|
||
if not ready_to_call:
|
||
return
|
||
|
||
if inspect.isasyncgen(ready_to_call):
|
||
_has_yielded = False
|
||
try:
|
||
async for ret in ready_to_call:
|
||
# 这里逐步执行异步生成器, 对于每个 yield 返回的 ret, 执行下面的代码
|
||
# 返回值只能是 MessageEventResult 或者 None(无返回值)
|
||
_has_yielded = True
|
||
if isinstance(ret, (MessageEventResult, CommandResult)):
|
||
# 如果返回值是 MessageEventResult, 设置结果并继续
|
||
event.set_result(ret)
|
||
yield
|
||
else:
|
||
# 如果返回值是 None, 则不设置结果并继续
|
||
# 继续执行后续阶段
|
||
yield ret
|
||
if not _has_yielded:
|
||
# 如果这个异步生成器没有执行到 yield 分支
|
||
yield
|
||
except Exception as e:
|
||
logger.error(f"Previous Error: {trace_}")
|
||
raise e
|
||
elif inspect.iscoroutine(ready_to_call):
|
||
# 如果只是一个协程, 直接执行
|
||
ret = await ready_to_call
|
||
if isinstance(ret, (MessageEventResult, CommandResult)):
|
||
event.set_result(ret)
|
||
yield
|
||
else:
|
||
yield ret
|
||
|
||
|
||
async def call_event_hook(
|
||
event: AstrMessageEvent,
|
||
hook_type: EventType,
|
||
*args,
|
||
**kwargs,
|
||
) -> bool:
|
||
"""调用事件钩子函数
|
||
|
||
Returns:
|
||
bool: 如果事件被终止,返回 True
|
||
#"""
|
||
handlers = star_handlers_registry.get_handlers_by_event_type(
|
||
hook_type, plugins_name=event.plugins_name
|
||
)
|
||
for handler in handlers:
|
||
try:
|
||
logger.debug(
|
||
f"hook({hook_type.name}) -> {star_map[handler.handler_module_path].name} - {handler.handler_name}"
|
||
)
|
||
await handler.handler(event, *args, **kwargs)
|
||
except BaseException:
|
||
logger.error(traceback.format_exc())
|
||
|
||
if event.is_stopped():
|
||
logger.info(
|
||
f"{star_map[handler.handler_module_path].name} - {handler.handler_name} 终止了事件传播。"
|
||
)
|
||
|
||
return event.is_stopped()
|