mirror of
https://github.com/progval/irctest.git
synced 2025-04-05 06:49:47 +00:00
Add test for message matching commands
+ fix a bug in tested code + change conftest.py to allow missing --controller arg (which is an UI improvement, as it allows using 'pytest --help' now)
This commit is contained in:
19
conftest.py
19
conftest.py
@ -23,7 +23,10 @@ def pytest_configure(config):
|
|||||||
module_name = config.getoption("controller")
|
module_name = config.getoption("controller")
|
||||||
|
|
||||||
if module_name is None:
|
if module_name is None:
|
||||||
pytest.exit("--controller is required.", 1)
|
print("Missing --controller option, errors may occur.")
|
||||||
|
_IrcTestCase.controllerClass = None
|
||||||
|
_IrcTestCase.show_io = True # TODO
|
||||||
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
module = importlib.import_module(module_name)
|
module = importlib.import_module(module_name)
|
||||||
@ -57,10 +60,13 @@ def pytest_collection_modifyitems(session, config, items):
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
# First, check if we should run server tests or client tests
|
# First, check if we should run server tests or client tests
|
||||||
if issubclass(_IrcTestCase.controllerClass, BaseServerController):
|
server_tests = client_tests = False
|
||||||
|
if _IrcTestCase.controllerClass is None:
|
||||||
|
return
|
||||||
|
elif issubclass(_IrcTestCase.controllerClass, BaseServerController):
|
||||||
server_tests = True
|
server_tests = True
|
||||||
elif issubclass(_IrcTestCase.controllerClass, BaseClientController):
|
elif issubclass(_IrcTestCase.controllerClass, BaseClientController):
|
||||||
server_tests = False
|
client_tests = True
|
||||||
else:
|
else:
|
||||||
assert False, (
|
assert False, (
|
||||||
f"{_IrcTestCase.controllerClass} inherits neither "
|
f"{_IrcTestCase.controllerClass} inherits neither "
|
||||||
@ -86,13 +92,10 @@ def pytest_collection_modifyitems(session, config, items):
|
|||||||
if server_tests:
|
if server_tests:
|
||||||
filtered_items.append(item)
|
filtered_items.append(item)
|
||||||
elif issubclass(item.parent.cls, BaseClientTestCase):
|
elif issubclass(item.parent.cls, BaseClientTestCase):
|
||||||
if not server_tests:
|
if client_tests:
|
||||||
filtered_items.append(item)
|
filtered_items.append(item)
|
||||||
else:
|
else:
|
||||||
assert False, (
|
filtered_items.append(item)
|
||||||
f"{item}'s class inherits neither BaseServerTestCase "
|
|
||||||
"or BaseClientTestCase"
|
|
||||||
)
|
|
||||||
|
|
||||||
# Finally, rewrite in-place the list of tests pytest will run
|
# Finally, rewrite in-place the list of tests pytest will run
|
||||||
items[:] = filtered_items
|
items[:] = filtered_items
|
||||||
|
@ -99,7 +99,8 @@ class _IrcTestCase(unittest.TestCase, Generic[TController]):
|
|||||||
|
|
||||||
def setUp(self) -> None:
|
def setUp(self) -> None:
|
||||||
super().setUp()
|
super().setUp()
|
||||||
self.controller = self.controllerClass(self.config())
|
if self.controllerClass is not None:
|
||||||
|
self.controller = self.controllerClass(self.config())
|
||||||
if self.show_io:
|
if self.show_io:
|
||||||
print("---- new test ----")
|
print("---- new test ----")
|
||||||
|
|
||||||
@ -162,9 +163,10 @@ class _IrcTestCase(unittest.TestCase, Generic[TController]):
|
|||||||
|
|
||||||
if nick:
|
if nick:
|
||||||
got_nick = msg.prefix.split("!")[0] if msg.prefix else None
|
got_nick = msg.prefix.split("!")[0] if msg.prefix else None
|
||||||
if msg.prefix is None:
|
if nick != got_nick:
|
||||||
fail_msg = (
|
fail_msg = (
|
||||||
fail_msg or "expected nick to be {expects}, got {got} prefix: {msg}"
|
fail_msg
|
||||||
|
or "expected nick to be {expects}, got {got} instead: {msg}"
|
||||||
)
|
)
|
||||||
return fail_msg.format(
|
return fail_msg.format(
|
||||||
*extra_format, got=got_nick, expects=nick, param=key, msg=msg
|
*extra_format, got=got_nick, expects=nick, param=key, msg=msg
|
||||||
@ -354,9 +356,7 @@ class BaseClientTestCase(_IrcTestCase[basecontrollers.BaseClientController]):
|
|||||||
return line
|
return line
|
||||||
|
|
||||||
def getMessage(
|
def getMessage(
|
||||||
self,
|
self, *args: Any, filter_pred: Optional[Callable[[Message], bool]] = None
|
||||||
*args: Any,
|
|
||||||
filter_pred: Optional[Callable[[Message], bool]] = None,
|
|
||||||
) -> Message:
|
) -> Message:
|
||||||
"""Gets a message and returns it. If a filter predicate is given,
|
"""Gets a message and returns it. If a filter predicate is given,
|
||||||
fetches messages until the predicate returns a False on a message,
|
fetches messages until the predicate returns a False on a message,
|
||||||
|
153
irctest/self_tests/test_cases.py
Normal file
153
irctest/self_tests/test_cases.py
Normal file
@ -0,0 +1,153 @@
|
|||||||
|
from typing import Dict, List, Tuple
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from irctest import cases
|
||||||
|
from irctest.irc_utils.message_parser import parse_message
|
||||||
|
from irctest.patma import ANYDICT, ANYSTR, StrRe
|
||||||
|
|
||||||
|
# fmt: off
|
||||||
|
MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str]]] = [
|
||||||
|
(
|
||||||
|
# the specification:
|
||||||
|
dict(
|
||||||
|
command="PRIVMSG",
|
||||||
|
params=["#chan", "hello"],
|
||||||
|
),
|
||||||
|
# matches:
|
||||||
|
[
|
||||||
|
"PRIVMSG #chan hello",
|
||||||
|
"PRIVMSG #chan :hello",
|
||||||
|
"@tag1=bar PRIVMSG #chan :hello",
|
||||||
|
"@tag1=bar;tag2= PRIVMSG #chan :hello",
|
||||||
|
":foo!baz@qux PRIVMSG #chan hello",
|
||||||
|
"@tag1=bar :foo!baz@qux PRIVMSG #chan :hello",
|
||||||
|
],
|
||||||
|
# and does not match:
|
||||||
|
[
|
||||||
|
"PRIVMSG #chan hello2",
|
||||||
|
"PRIVMSG #chan2 hello",
|
||||||
|
]
|
||||||
|
),
|
||||||
|
(
|
||||||
|
# the specification:
|
||||||
|
dict(
|
||||||
|
command="PRIVMSG",
|
||||||
|
params=["#chan", StrRe("hello.*")],
|
||||||
|
),
|
||||||
|
# matches:
|
||||||
|
[
|
||||||
|
"PRIVMSG #chan hello",
|
||||||
|
"PRIVMSG #chan :hello",
|
||||||
|
"PRIVMSG #chan hello2",
|
||||||
|
"@tag1=bar PRIVMSG #chan :hello",
|
||||||
|
"@tag1=bar;tag2= PRIVMSG #chan :hello",
|
||||||
|
":foo!baz@qux PRIVMSG #chan hello",
|
||||||
|
"@tag1=bar :foo!baz@qux PRIVMSG #chan :hello",
|
||||||
|
],
|
||||||
|
# and does not match:
|
||||||
|
[
|
||||||
|
"PRIVMSG #chan :hi",
|
||||||
|
"PRIVMSG #chan2 hello",
|
||||||
|
]
|
||||||
|
),
|
||||||
|
(
|
||||||
|
# the specification:
|
||||||
|
dict(
|
||||||
|
nick="foo",
|
||||||
|
command="PRIVMSG",
|
||||||
|
),
|
||||||
|
# matches:
|
||||||
|
[
|
||||||
|
":foo!baz@qux PRIVMSG #chan hello",
|
||||||
|
"@tag1=bar :foo!baz@qux PRIVMSG #chan :hello",
|
||||||
|
],
|
||||||
|
# and does not match:
|
||||||
|
[
|
||||||
|
"PRIVMSG #chan :hi",
|
||||||
|
":foo2!baz@qux PRIVMSG #chan hello",
|
||||||
|
"@tag1=bar :foo2!baz@qux PRIVMSG #chan :hello",
|
||||||
|
]
|
||||||
|
),
|
||||||
|
(
|
||||||
|
# the specification:
|
||||||
|
dict(
|
||||||
|
tags={"tag1": "bar"},
|
||||||
|
command="PRIVMSG",
|
||||||
|
params=["#chan", "hello"],
|
||||||
|
),
|
||||||
|
# matches:
|
||||||
|
[
|
||||||
|
"@tag1=bar PRIVMSG #chan :hello",
|
||||||
|
"@tag1=bar :foo!baz@qux PRIVMSG #chan :hello",
|
||||||
|
],
|
||||||
|
# and does not match:
|
||||||
|
[
|
||||||
|
"@tag1=bar;tag2= PRIVMSG #chan :hello",
|
||||||
|
"@tag1=value1 PRIVMSG #chan :hello",
|
||||||
|
"PRIVMSG #chan hello",
|
||||||
|
":foo!baz@qux PRIVMSG #chan hello",
|
||||||
|
]
|
||||||
|
),
|
||||||
|
(
|
||||||
|
# the specification:
|
||||||
|
dict(
|
||||||
|
tags={"tag1": ANYSTR},
|
||||||
|
command="PRIVMSG",
|
||||||
|
params=["#chan", ANYSTR],
|
||||||
|
),
|
||||||
|
# matches:
|
||||||
|
[
|
||||||
|
"@tag1=bar PRIVMSG #chan :hello",
|
||||||
|
"@tag1=value1 PRIVMSG #chan :hello",
|
||||||
|
"@tag1=bar :foo!baz@qux PRIVMSG #chan :hello",
|
||||||
|
],
|
||||||
|
# and does not match:
|
||||||
|
[
|
||||||
|
"@tag1=bar;tag2= PRIVMSG #chan :hello",
|
||||||
|
"PRIVMSG #chan hello",
|
||||||
|
":foo!baz@qux PRIVMSG #chan hello",
|
||||||
|
]
|
||||||
|
),
|
||||||
|
(
|
||||||
|
# the specification:
|
||||||
|
dict(
|
||||||
|
tags={"tag1": "bar", **ANYDICT},
|
||||||
|
command="PRIVMSG",
|
||||||
|
params=["#chan", "hello"],
|
||||||
|
),
|
||||||
|
# matches:
|
||||||
|
[
|
||||||
|
"@tag1=bar PRIVMSG #chan :hello",
|
||||||
|
"@tag1=bar;tag2= PRIVMSG #chan :hello",
|
||||||
|
"@tag1=bar :foo!baz@qux PRIVMSG #chan :hello",
|
||||||
|
],
|
||||||
|
# and does not match:
|
||||||
|
[
|
||||||
|
"PRIVMG #chan :hello",
|
||||||
|
"@tag1=value1 PRIVMSG #chan :hello",
|
||||||
|
"PRIVMSG #chan hello2",
|
||||||
|
"PRIVMSG #chan2 hello",
|
||||||
|
":foo!baz@qux PRIVMSG #chan hello",
|
||||||
|
]
|
||||||
|
),
|
||||||
|
]
|
||||||
|
# fmt: on
|
||||||
|
|
||||||
|
|
||||||
|
class IrcTestCaseTestCase(cases._IrcTestCase):
|
||||||
|
def test_message_matching(self):
|
||||||
|
for (spec, positive_matches, negative_matches) in MESSAGE_SPECS:
|
||||||
|
with self.subTest(spec):
|
||||||
|
for msg in positive_matches:
|
||||||
|
with self.subTest(msg):
|
||||||
|
assert not self.messageDiffers(parse_message(msg), **spec), msg
|
||||||
|
assert self.messageEqual(parse_message(msg), **spec), msg
|
||||||
|
self.assertMessageMatch(parse_message(msg), **spec), msg
|
||||||
|
|
||||||
|
for msg in negative_matches:
|
||||||
|
with self.subTest(msg):
|
||||||
|
assert self.messageDiffers(parse_message(msg), **spec), msg
|
||||||
|
assert not self.messageEqual(parse_message(msg), **spec), msg
|
||||||
|
with pytest.raises(AssertionError):
|
||||||
|
self.assertMessageMatch(parse_message(msg), **spec), msg
|
Reference in New Issue
Block a user