mirror of
https://github.com/nonebot/nonebot2.git
synced 2024-12-18 17:35:46 +08:00
222 lines
6.6 KiB
Python
222 lines
6.6 KiB
Python
"""本模块模块实现了依赖注入的定义与处理。
|
|
|
|
FrontMatter:
|
|
sidebar_position: 0
|
|
description: nonebot.dependencies 模块
|
|
"""
|
|
|
|
import abc
|
|
import inspect
|
|
from typing import (
|
|
Any,
|
|
Dict,
|
|
List,
|
|
Type,
|
|
Generic,
|
|
TypeVar,
|
|
Callable,
|
|
Optional,
|
|
Awaitable,
|
|
cast,
|
|
)
|
|
|
|
from pydantic import BaseConfig
|
|
from pydantic.schema import get_annotation_from_field_info
|
|
from pydantic.fields import Required, FieldInfo, Undefined, ModelField
|
|
|
|
from nonebot.log import logger
|
|
from nonebot.exception import TypeMisMatch
|
|
from nonebot.typing import _DependentCallable
|
|
from nonebot.utils import run_sync, is_coroutine_callable
|
|
|
|
from .utils import check_field_type, get_typed_signature
|
|
|
|
R = TypeVar("R")
|
|
T = TypeVar("T", bound="Dependent")
|
|
|
|
|
|
class Param(abc.ABC, FieldInfo):
|
|
"""依赖注入的基本单元 —— 参数。
|
|
|
|
继承自 `pydantic.fields.FieldInfo`,用于描述参数信息(不包括参数名)。
|
|
"""
|
|
|
|
@classmethod
|
|
def _check_param(
|
|
cls, dependent: "Dependent", name: str, param: inspect.Parameter
|
|
) -> Optional["Param"]:
|
|
return None
|
|
|
|
@classmethod
|
|
def _check_parameterless(
|
|
cls, dependent: "Dependent", value: Any
|
|
) -> Optional["Param"]:
|
|
return None
|
|
|
|
@abc.abstractmethod
|
|
async def _solve(self, **kwargs: Any) -> Any:
|
|
raise NotImplementedError
|
|
|
|
|
|
class CustomConfig(BaseConfig):
|
|
arbitrary_types_allowed = True
|
|
|
|
|
|
class Dependent(Generic[R]):
|
|
"""依赖注入容器
|
|
|
|
参数:
|
|
call: 依赖注入的可调用对象,可以是任何 Callable 对象
|
|
pre_checkers: 依赖注入解析前的参数检查
|
|
params: 具名参数列表
|
|
parameterless: 匿名参数列表
|
|
allow_types: 允许的参数类型
|
|
"""
|
|
|
|
def __init__(
|
|
self,
|
|
*,
|
|
call: _DependentCallable[R],
|
|
pre_checkers: Optional[List[Param]] = None,
|
|
params: Optional[List[ModelField]] = None,
|
|
parameterless: Optional[List[Param]] = None,
|
|
allow_types: Optional[List[Type[Param]]] = None,
|
|
) -> None:
|
|
self.call = call
|
|
self.pre_checkers = pre_checkers or []
|
|
self.params = params or []
|
|
self.parameterless = parameterless or []
|
|
self.allow_types = allow_types or []
|
|
|
|
def __repr__(self) -> str:
|
|
return (
|
|
f"<Dependent call={self.call}, params={self.params},"
|
|
f" parameterless={self.parameterless}>"
|
|
)
|
|
|
|
def __str__(self) -> str:
|
|
return self.__repr__()
|
|
|
|
async def __call__(self, **kwargs: Any) -> R:
|
|
values = await self.solve(**kwargs)
|
|
|
|
if is_coroutine_callable(self.call):
|
|
return await cast(Callable[..., Awaitable[R]], self.call)(**values)
|
|
else:
|
|
return await run_sync(cast(Callable[..., R], self.call))(**values)
|
|
|
|
def parse_param(self, name: str, param: inspect.Parameter) -> Param:
|
|
for allow_type in self.allow_types:
|
|
if field_info := allow_type._check_param(self, name, param):
|
|
return field_info
|
|
raise ValueError(
|
|
f"Unknown parameter {name} for function {self.call} with type {param.annotation}"
|
|
)
|
|
|
|
def parse_parameterless(self, value: Any) -> Param:
|
|
for allow_type in self.allow_types:
|
|
if field_info := allow_type._check_parameterless(self, value):
|
|
return field_info
|
|
raise ValueError(
|
|
f"Unknown parameterless {value} for function {self.call} with type {type(value)}"
|
|
)
|
|
|
|
def prepend_parameterless(self, value: Any) -> None:
|
|
self.parameterless.insert(0, self.parse_parameterless(value))
|
|
|
|
def append_parameterless(self, value: Any) -> None:
|
|
self.parameterless.append(self.parse_parameterless(value))
|
|
|
|
@classmethod
|
|
def parse(
|
|
cls,
|
|
*,
|
|
call: _DependentCallable[R],
|
|
parameterless: Optional[List[Any]] = None,
|
|
allow_types: Optional[List[Type[Param]]] = None,
|
|
) -> "Dependent[R]":
|
|
signature = get_typed_signature(call)
|
|
params = signature.parameters
|
|
dependent = cls(
|
|
call=call,
|
|
allow_types=allow_types,
|
|
)
|
|
|
|
for param_name, param in params.items():
|
|
default_value = Required
|
|
if param.default != param.empty:
|
|
default_value = param.default
|
|
|
|
if isinstance(default_value, Param):
|
|
field_info = default_value
|
|
else:
|
|
field_info = dependent.parse_param(param_name, param)
|
|
|
|
default_value = field_info.default
|
|
|
|
annotation: Any = Any
|
|
required = default_value == Required
|
|
if param.annotation != param.empty:
|
|
annotation = param.annotation
|
|
annotation = get_annotation_from_field_info(
|
|
annotation, field_info, param_name
|
|
)
|
|
dependent.params.append(
|
|
ModelField(
|
|
name=param_name,
|
|
type_=annotation,
|
|
class_validators=None,
|
|
model_config=CustomConfig,
|
|
default=None if required else default_value,
|
|
required=required,
|
|
field_info=field_info,
|
|
)
|
|
)
|
|
|
|
parameterless_params = [
|
|
dependent.parse_parameterless(param) for param in (parameterless or [])
|
|
]
|
|
dependent.parameterless.extend(parameterless_params)
|
|
|
|
logger.trace(
|
|
f"Parsed dependent with call={call}, "
|
|
f"params={[param.field_info for param in dependent.params]}, "
|
|
f"parameterless={dependent.parameterless}"
|
|
)
|
|
|
|
return dependent
|
|
|
|
async def solve(
|
|
self,
|
|
**params: Any,
|
|
) -> Dict[str, Any]:
|
|
values: Dict[str, Any] = {}
|
|
|
|
for checker in self.pre_checkers:
|
|
await checker._solve(**params)
|
|
|
|
for param in self.parameterless:
|
|
await param._solve(**params)
|
|
|
|
for field in self.params:
|
|
field_info = field.field_info
|
|
assert isinstance(field_info, Param), "Params must be subclasses of Param"
|
|
value = await field_info._solve(**params)
|
|
if value is Undefined:
|
|
value = field.get_default()
|
|
|
|
try:
|
|
values[field.name] = check_field_type(field, value)
|
|
except TypeMisMatch:
|
|
logger.debug(
|
|
f"{field_info} "
|
|
f"type {type(value)} not match depends {self.call} "
|
|
f"annotation {field._type_display()}, ignored"
|
|
)
|
|
raise
|
|
|
|
return values
|
|
|
|
|
|
__autodoc__ = {"CustomConfig": False}
|