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:
2021-04-17 23:10:10 +02:00
committed by Valentin Lorentz
parent 498b67ae96
commit cfe0b0d3dd
3 changed files with 170 additions and 14 deletions

View File

@ -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

View File

@ -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,

View 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