add got receive tests

This commit is contained in:
yanyongyu 2021-12-21 00:39:12 +08:00
parent d549cdf26a
commit e9b8515cf1
7 changed files with 111 additions and 52 deletions

View File

@ -58,6 +58,15 @@ class Dependent(Generic[R]):
self.parameterless = parameterless or [] self.parameterless = parameterless or []
self.allow_types = allow_types 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: async def __call__(self, **kwargs: Any) -> R:
values = await self.solve(**kwargs) values = await self.solve(**kwargs)

View File

@ -172,11 +172,6 @@ class Matcher(metaclass=MatcherMeta):
:说明: 事件响应器默认状态 :说明: 事件响应器默认状态
""" """
_default_parser: Optional[Dependent[None]] = None
"""
:类型: ``Optional[Dependent]``
:说明: 事件响应器默认参数解析函数
"""
_default_type_updater: Optional[Dependent[str]] = None _default_type_updater: Optional[Dependent[str]] = None
""" """
:类型: ``Optional[Dependent]`` :类型: ``Optional[Dependent]``
@ -441,28 +436,27 @@ class Matcher(metaclass=MatcherMeta):
* ``parameterless: Optional[List[Any]]``: 非参数类型依赖列表 * ``parameterless: Optional[List[Any]]``: 非参数类型依赖列表
""" """
_id = id or ""
async def _receive(event: Event, matcher: "Matcher") -> Union[None, NoReturn]: async def _receive(event: Event, matcher: "Matcher") -> Union[None, NoReturn]:
if matcher.get_receive(id): if matcher.get_receive(_id):
return return
if matcher.get_target() == RECEIVE_KEY.format(id=id or ""): if matcher.get_target() == RECEIVE_KEY.format(id=_id):
matcher.set_receive(id, event) matcher.set_receive(_id, event)
return return
matcher.set_target(RECEIVE_KEY.format(id=id or "")) matcher.set_target(RECEIVE_KEY.format(id=_id))
raise RejectedException raise RejectedException
parameterless = [params.Depends(_receive), *(parameterless or [])] _parameterless = [params.Depends(_receive), *(parameterless or [])]
def _decorator(func: T_Handler) -> T_Handler: def _decorator(func: T_Handler) -> T_Handler:
if cls.handlers and cls.handlers[-1].call is func: if cls.handlers and cls.handlers[-1].call is func:
func_handler = cls.handlers[-1] func_handler = cls.handlers[-1]
for depend in reversed(parameterless): for depend in reversed(_parameterless):
func_handler.prepend_parameterless(depend) func_handler.prepend_parameterless(depend)
else: else:
cls.append_handler( cls.append_handler(func, parameterless=_parameterless)
func,
parameterless=parameterless if cls.handlers else parameterless,
)
return func return func
@ -558,15 +552,8 @@ class Matcher(metaclass=MatcherMeta):
* ``message: Union[str, Message, MessageSegment]``: 消息内容 * ``message: Union[str, Message, MessageSegment]``: 消息内容
* ``**kwargs``: 其他传递给 ``bot.send`` 的参数请参考对应 adapter bot 对象 api * ``**kwargs``: 其他传递给 ``bot.send`` 的参数请参考对应 adapter bot 对象 api
""" """
bot = current_bot.get() if message is not None:
event = current_event.get() await cls.send(message, **kwargs)
state = current_state.get()
if isinstance(message, MessageTemplate):
_message = message.format(**state)
else:
_message = message
if _message is not None:
await bot.send(event=event, message=_message, **kwargs)
raise FinishedException raise FinishedException
@classmethod @classmethod
@ -585,15 +572,8 @@ class Matcher(metaclass=MatcherMeta):
* ``prompt: Union[str, Message, MessageSegment]``: 消息内容 * ``prompt: Union[str, Message, MessageSegment]``: 消息内容
* ``**kwargs``: 其他传递给 ``bot.send`` 的参数请参考对应 adapter bot 对象 api * ``**kwargs``: 其他传递给 ``bot.send`` 的参数请参考对应 adapter bot 对象 api
""" """
bot = current_bot.get() if prompt is not None:
event = current_event.get() await cls.send(prompt, **kwargs)
state = current_state.get()
if isinstance(prompt, MessageTemplate):
_prompt = prompt.format(**state)
else:
_prompt = prompt
if _prompt is not None:
await bot.send(event=event, message=_prompt, **kwargs)
raise PausedException raise PausedException
@classmethod @classmethod
@ -610,27 +590,20 @@ class Matcher(metaclass=MatcherMeta):
* ``prompt: Union[str, Message, MessageSegment]``: 消息内容 * ``prompt: Union[str, Message, MessageSegment]``: 消息内容
* ``**kwargs``: 其他传递给 ``bot.send`` 的参数请参考对应 adapter bot 对象 api * ``**kwargs``: 其他传递给 ``bot.send`` 的参数请参考对应 adapter bot 对象 api
""" """
bot = current_bot.get() if prompt is not None:
event = current_event.get() await cls.send(prompt, **kwargs)
state = current_state.get()
if isinstance(prompt, MessageTemplate):
_prompt = prompt.format(**state)
else:
_prompt = prompt
if _prompt is not None:
await bot.send(event=event, message=_prompt, **kwargs)
raise RejectedException raise RejectedException
def get_receive(self, id: Optional[str], default: T = None) -> Union[Event, T]: def get_receive(self, id: str, default: T = None) -> Union[Event, T]:
if id is None:
return self.state.get(LAST_RECEIVE_KEY, default)
return self.state.get(RECEIVE_KEY.format(id=id), default) return self.state.get(RECEIVE_KEY.format(id=id), default)
def set_receive(self, id: Optional[str], event: Event) -> None: def set_receive(self, id: str, event: Event) -> None:
if id is not None: self.state[RECEIVE_KEY.format(id=id)] = event
self.state[RECEIVE_KEY.format(id=id)] = event
self.state[LAST_RECEIVE_KEY] = event self.state[LAST_RECEIVE_KEY] = event
def get_last_receive(self, default: T = None) -> Union[Event, T]:
return self.state.get(LAST_RECEIVE_KEY, default)
def get_arg(self, key: str, default: T = None) -> Union[Event, T]: def get_arg(self, key: str, default: T = None) -> Union[Event, T]:
return self.state.get(ARG_KEY.format(key=key), default) return self.state.get(ARG_KEY.format(key=key), default)

View File

@ -318,16 +318,16 @@ class MatcherParam(Param):
return matcher return matcher
def Received(id: str, default: Any = None) -> Any: def Received(id: Optional[str] = None, default: Any = None) -> Any:
def _received(matcher: "Matcher"): def _received(matcher: "Matcher"):
return matcher.get_receive(id, default) return matcher.get_receive(id or "", default)
return Depends(_received, use_cache=False) return Depends(_received, use_cache=False)
def LastReceived(default: Any = None) -> Any: def LastReceived(default: Any = None) -> Any:
def _last_received(matcher: "Matcher") -> Any: def _last_received(matcher: "Matcher") -> Any:
return matcher.get_receive(None, default) return matcher.get_last_receive(default)
return Depends(_last_received, use_cache=False) return Depends(_last_received, use_cache=False)

View File

@ -1,2 +1,3 @@
DEBUG=true
NICKNAME=["test"] NICKNAME=["test"]
CONFIG_FROM_ENV= CONFIG_FROM_ENV=

35
tests/plugins/matcher.py Normal file
View File

@ -0,0 +1,35 @@
from nonebot import on_message
from nonebot.adapters import Event
from nonebot.params import ArgStr, Received, LastReceived
test_handle = on_message()
@test_handle.handle()
async def handle():
await test_handle.finish("send", at_sender=True)
test_got = on_message()
@test_got.got("key1", "prompt key1")
@test_got.got("key2", "prompt key2")
async def got(key1: str = ArgStr(), key2: str = ArgStr()):
assert key1 == "text"
assert key2 == "text"
await test_got.reject("reject", at_sender=True)
test_receive = on_message()
@test_receive.receive()
@test_receive.receive("receive")
async def receive(
x: Event = Received("receive"), y: Event = LastReceived(), z: Event = Received()
):
assert str(x.get_message()) == "text"
assert str(z.get_message()) == "text"
assert x is y
await test_receive.pause("pause", at_sender=True)

41
tests/test_matcher.py Normal file
View File

@ -0,0 +1,41 @@
from typing import TYPE_CHECKING, Set
import pytest
from nonebug import App
from utils import load_plugin, make_fake_event, make_fake_message
@pytest.mark.asyncio
async def test_matcher(app: App, load_plugin):
from plugins.matcher import test_got, test_handle, test_receive
message = make_fake_message()("text")
event = make_fake_event(_message=message)()
assert len(test_handle.handlers) == 1
async with app.test_matcher(test_handle) as ctx:
bot = ctx.create_bot()
ctx.receive_event(bot, event)
ctx.should_call_send(event, "send", "result", at_sender=True)
ctx.should_finished()
assert len(test_got.handlers) == 1
async with app.test_matcher(test_got) as ctx:
bot = ctx.create_bot()
ctx.receive_event(bot, event)
ctx.should_call_send(event, "prompt key1", "result1")
ctx.receive_event(bot, event)
ctx.should_call_send(event, "prompt key2", "result2")
ctx.receive_event(bot, event)
ctx.should_call_send(event, "reject", "result3", at_sender=True)
ctx.should_rejected()
assert len(test_receive.handlers) == 1
async with app.test_matcher(test_receive) as ctx:
bot = ctx.create_bot()
ctx.receive_event(bot, event)
ctx.receive_event(bot, event)
ctx.receive_event(bot, event)
ctx.should_call_send(event, "pause", "result", at_sender=True)
ctx.should_paused()

View File

@ -196,7 +196,7 @@ async def test_matcher(app: App, load_plugin):
event = make_fake_event()() event = make_fake_event()()
fake_matcher.set_receive("test", event) fake_matcher.set_receive("test", event)
event_next = make_fake_event()() event_next = make_fake_event()()
fake_matcher.set_receive(None, event_next) fake_matcher.set_receive("", event_next)
async with app.test_dependent( async with app.test_dependent(
receive, allow_types=[MatcherParam, DependParam] receive, allow_types=[MatcherParam, DependParam]