mirror of
https://github.com/progval/irctest.git
synced 2025-04-05 23:09:48 +00:00
Move list_match to its own module, and prepare generalizing AnyStr
This commit is contained in:
@ -24,7 +24,7 @@ import unittest
|
|||||||
|
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from . import basecontrollers, client_mock, runner, tls
|
from . import basecontrollers, client_mock, patma, runner, tls
|
||||||
from .authentication import Authentication
|
from .authentication import Authentication
|
||||||
from .basecontrollers import TestCaseControllerConfig
|
from .basecontrollers import TestCaseControllerConfig
|
||||||
from .exceptions import ConnectionClosed
|
from .exceptions import ConnectionClosed
|
||||||
@ -65,13 +65,6 @@ TController = TypeVar("TController", bound=basecontrollers._BaseController)
|
|||||||
T = TypeVar("T")
|
T = TypeVar("T")
|
||||||
|
|
||||||
|
|
||||||
class AnyStr:
|
|
||||||
"""Used as a wildcard when matching message arguments
|
|
||||||
(see assertMessageMatch and listMatch)"""
|
|
||||||
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
class ChannelJoinException(Exception):
|
class ChannelJoinException(Exception):
|
||||||
def __init__(self, code: str, params: List[str]):
|
def __init__(self, code: str, params: List[str]):
|
||||||
super().__init__(f"Failed to join channel ({code}): {params}")
|
super().__init__(f"Failed to join channel ({code}): {params}")
|
||||||
@ -116,7 +109,7 @@ class _IrcTestCase(unittest.TestCase, Generic[TController]):
|
|||||||
Takes the message as first arguments, and comparisons to be made
|
Takes the message as first arguments, and comparisons to be made
|
||||||
as keyword arguments.
|
as keyword arguments.
|
||||||
|
|
||||||
Uses self.listMatch on the params argument.
|
Uses patma.list_match on the params argument.
|
||||||
"""
|
"""
|
||||||
error = self.messageDiffers(msg, **kwargs)
|
error = self.messageDiffers(msg, **kwargs)
|
||||||
if error:
|
if error:
|
||||||
@ -130,7 +123,7 @@ class _IrcTestCase(unittest.TestCase, Generic[TController]):
|
|||||||
def messageDiffers(
|
def messageDiffers(
|
||||||
self,
|
self,
|
||||||
msg: Message,
|
msg: Message,
|
||||||
params: Optional[List[Union[str, Type[AnyStr]]]] = None,
|
params: Optional[List[Union[str, patma.Operator]]] = None,
|
||||||
target: Optional[str] = None,
|
target: Optional[str] = None,
|
||||||
nick: Optional[str] = None,
|
nick: Optional[str] = None,
|
||||||
fail_msg: Optional[str] = None,
|
fail_msg: Optional[str] = None,
|
||||||
@ -152,7 +145,7 @@ class _IrcTestCase(unittest.TestCase, Generic[TController]):
|
|||||||
msg=msg,
|
msg=msg,
|
||||||
)
|
)
|
||||||
|
|
||||||
if params and not self.listMatch(msg.params, params):
|
if params and not patma.list_match(msg.params, params):
|
||||||
fail_msg = fail_msg or "params to be {expects}, got {got}: {msg}"
|
fail_msg = fail_msg or "params to be {expects}, got {got}: {msg}"
|
||||||
return fail_msg.format(
|
return fail_msg.format(
|
||||||
*extra_format, got=msg.params, expects=params, msg=msg
|
*extra_format, got=msg.params, expects=params, msg=msg
|
||||||
@ -170,22 +163,6 @@ class _IrcTestCase(unittest.TestCase, Generic[TController]):
|
|||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def listMatch(
|
|
||||||
self, got: List[str], expected: List[Union[str, Type[AnyStr]]]
|
|
||||||
) -> bool:
|
|
||||||
"""Returns True iff the list are equal.
|
|
||||||
The ellipsis (aka. "..." aka triple dots) can be used on the 'expected'
|
|
||||||
side as a wildcard, matching any *single* value."""
|
|
||||||
if len(got) != len(expected):
|
|
||||||
return False
|
|
||||||
for (got_value, expected_value) in zip(got, expected):
|
|
||||||
if expected_value is AnyStr:
|
|
||||||
# wildcard
|
|
||||||
continue
|
|
||||||
if got_value != expected_value:
|
|
||||||
return False
|
|
||||||
return True
|
|
||||||
|
|
||||||
def assertIn(
|
def assertIn(
|
||||||
self,
|
self,
|
||||||
member: Any,
|
member: Any,
|
||||||
|
51
irctest/patma.py
Normal file
51
irctest/patma.py
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
"""Pattern-matching utilities"""
|
||||||
|
|
||||||
|
import dataclasses
|
||||||
|
import re
|
||||||
|
from typing import List, Union
|
||||||
|
|
||||||
|
|
||||||
|
class Operator:
|
||||||
|
"""Used as a wildcards and operators when matching message arguments
|
||||||
|
(see assertMessageMatch and match_list)"""
|
||||||
|
|
||||||
|
def __init__(self) -> None:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class AnyStr(Operator):
|
||||||
|
"""Wildcard matching any string"""
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return "AnyStr"
|
||||||
|
|
||||||
|
|
||||||
|
@dataclasses.dataclass
|
||||||
|
class StrRe(Operator):
|
||||||
|
regexp: str
|
||||||
|
|
||||||
|
def __repr__(self) -> str:
|
||||||
|
return f"StrRe(r'{self.regexp}')"
|
||||||
|
|
||||||
|
|
||||||
|
ANYSTR = AnyStr()
|
||||||
|
"""Singleton, spares two characters"""
|
||||||
|
|
||||||
|
|
||||||
|
def match_list(got: List[str], expected: List[Union[str, Operator]]) -> bool:
|
||||||
|
"""Returns True iff the list are equal.
|
||||||
|
The ellipsis (aka. "..." aka triple dots) can be used on the 'expected'
|
||||||
|
side as a wildcard, matching any *single* value."""
|
||||||
|
if len(got) != len(expected):
|
||||||
|
return False
|
||||||
|
for (got_value, expected_value) in zip(got, expected):
|
||||||
|
if isinstance(expected_value, AnyStr):
|
||||||
|
# wildcard
|
||||||
|
continue
|
||||||
|
elif isinstance(expected_value, StrRe):
|
||||||
|
if not re.match(expected_value.regexp, got_value):
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
if got_value != expected_value:
|
||||||
|
return False
|
||||||
|
return True
|
@ -1,5 +1,5 @@
|
|||||||
from irctest import cases
|
from irctest import cases
|
||||||
from irctest.cases import AnyStr
|
from irctest.patma import ANYSTR
|
||||||
from irctest.runner import CapabilityNotSupported, ImplementationChoice
|
from irctest.runner import CapabilityNotSupported, ImplementationChoice
|
||||||
|
|
||||||
|
|
||||||
@ -40,7 +40,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
|||||||
self.assertMessageMatch(
|
self.assertMessageMatch(
|
||||||
m,
|
m,
|
||||||
command="CAP",
|
command="CAP",
|
||||||
params=[AnyStr, "NAK", "foo"],
|
params=[ANYSTR, "NAK", "foo"],
|
||||||
fail_msg="Expected CAP NAK after requesting non-existing "
|
fail_msg="Expected CAP NAK after requesting non-existing "
|
||||||
"capability, got {msg}.",
|
"capability, got {msg}.",
|
||||||
)
|
)
|
||||||
@ -67,7 +67,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
|||||||
self.assertMessageMatch(
|
self.assertMessageMatch(
|
||||||
m,
|
m,
|
||||||
command="CAP",
|
command="CAP",
|
||||||
params=[AnyStr, "NAK", "foo qux bar baz qux quux"],
|
params=[ANYSTR, "NAK", "foo qux bar baz qux quux"],
|
||||||
fail_msg="Expected “CAP NAK :foo qux bar baz qux quux” after "
|
fail_msg="Expected “CAP NAK :foo qux bar baz qux quux” after "
|
||||||
"sending “CAP REQ :foo qux bar baz qux quux”, but got {msg}.",
|
"sending “CAP REQ :foo qux bar baz qux quux”, but got {msg}.",
|
||||||
)
|
)
|
||||||
@ -86,7 +86,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
|||||||
self.assertMessageMatch(
|
self.assertMessageMatch(
|
||||||
m,
|
m,
|
||||||
command="CAP",
|
command="CAP",
|
||||||
params=[AnyStr, "NAK", "foo multi-prefix bar"],
|
params=[ANYSTR, "NAK", "foo multi-prefix bar"],
|
||||||
fail_msg="Expected “CAP NAK :foo multi-prefix bar” after "
|
fail_msg="Expected “CAP NAK :foo multi-prefix bar” after "
|
||||||
"sending “CAP REQ :foo multi-prefix bar”, but got {msg}.",
|
"sending “CAP REQ :foo multi-prefix bar”, but got {msg}.",
|
||||||
)
|
)
|
||||||
@ -95,7 +95,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
|||||||
self.assertMessageMatch(
|
self.assertMessageMatch(
|
||||||
m,
|
m,
|
||||||
command="CAP",
|
command="CAP",
|
||||||
params=[AnyStr, "NAK", "multi-prefix bar"],
|
params=[ANYSTR, "NAK", "multi-prefix bar"],
|
||||||
fail_msg="Expected “CAP NAK :multi-prefix bar” after "
|
fail_msg="Expected “CAP NAK :multi-prefix bar” after "
|
||||||
"sending “CAP REQ :multi-prefix bar”, but got {msg}.",
|
"sending “CAP REQ :multi-prefix bar”, but got {msg}.",
|
||||||
)
|
)
|
||||||
@ -104,7 +104,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
|||||||
self.assertMessageMatch(
|
self.assertMessageMatch(
|
||||||
m,
|
m,
|
||||||
command="CAP",
|
command="CAP",
|
||||||
params=[AnyStr, "NAK", "foo multi-prefix"],
|
params=[ANYSTR, "NAK", "foo multi-prefix"],
|
||||||
fail_msg="Expected “CAP NAK :foo multi-prefix” after "
|
fail_msg="Expected “CAP NAK :foo multi-prefix” after "
|
||||||
"sending “CAP REQ :foo multi-prefix”, but got {msg}.",
|
"sending “CAP REQ :foo multi-prefix”, but got {msg}.",
|
||||||
)
|
)
|
||||||
@ -114,7 +114,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
|||||||
self.assertMessageMatch(
|
self.assertMessageMatch(
|
||||||
m,
|
m,
|
||||||
command="CAP",
|
command="CAP",
|
||||||
params=[AnyStr, "ACK", "multi-prefix"],
|
params=[ANYSTR, "ACK", "multi-prefix"],
|
||||||
fail_msg="Expected “CAP ACK :multi-prefix” after "
|
fail_msg="Expected “CAP ACK :multi-prefix” after "
|
||||||
"sending “CAP REQ :multi-prefix”, but got {msg}.",
|
"sending “CAP REQ :multi-prefix”, but got {msg}.",
|
||||||
)
|
)
|
||||||
@ -134,7 +134,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
|||||||
self.sendLine(1, "user user 0 * realname")
|
self.sendLine(1, "user user 0 * realname")
|
||||||
self.sendLine(1, "CAP END")
|
self.sendLine(1, "CAP END")
|
||||||
m = self.getRegistrationMessage(1)
|
m = self.getRegistrationMessage(1)
|
||||||
self.assertMessageMatch(m, command="CAP", params=[AnyStr, "ACK", AnyStr])
|
self.assertMessageMatch(m, command="CAP", params=[ANYSTR, "ACK", ANYSTR])
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
set(m.params[2].split()), {cap1, cap2}, "Didn't ACK both REQed caps"
|
set(m.params[2].split()), {cap1, cap2}, "Didn't ACK both REQed caps"
|
||||||
)
|
)
|
||||||
@ -152,9 +152,9 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
|||||||
self.sendLine(1, f"CAP REQ :-{cap2}")
|
self.sendLine(1, f"CAP REQ :-{cap2}")
|
||||||
m = self.getMessage(1)
|
m = self.getMessage(1)
|
||||||
# Must be either ACK or NAK
|
# Must be either ACK or NAK
|
||||||
if self.messageDiffers(m, command="CAP", params=[AnyStr, "ACK", f"-{cap2}"]):
|
if self.messageDiffers(m, command="CAP", params=[ANYSTR, "ACK", f"-{cap2}"]):
|
||||||
self.assertMessageMatch(
|
self.assertMessageMatch(
|
||||||
m, command="CAP", params=[AnyStr, "NAK", f"-{cap2}"]
|
m, command="CAP", params=[ANYSTR, "NAK", f"-{cap2}"]
|
||||||
)
|
)
|
||||||
raise ImplementationChoice(f"Does not support CAP REQ -{cap2}")
|
raise ImplementationChoice(f"Does not support CAP REQ -{cap2}")
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user