Make all tests pass with Unreal (minus service tests)

This commit is contained in:
2021-07-02 21:41:35 +02:00
parent 2d2e788275
commit f83f2a4edf
12 changed files with 139 additions and 44 deletions

View File

@ -63,16 +63,22 @@ SOPEL_SELECTORS := \
# testNoticeNonexistentChannel fails: https://bugs.unrealircd.org/view.php?id=5949 # testNoticeNonexistentChannel fails: https://bugs.unrealircd.org/view.php?id=5949
# test_regressions::testTagCap fails: https://bugs.unrealircd.org/view.php?id=5948 # test_regressions::testTagCap fails: https://bugs.unrealircd.org/view.php?id=5948
# test_messages::testLineTooLong fails: https://bugs.unrealircd.org/view.php?id=5947 # test_messages::testLineTooLong fails: https://bugs.unrealircd.org/view.php?id=5947
UNREAL_SELECTORS := \ # testCapRemovalByClient and testNakWhole fail pending https://github.com/unrealircd/unrealircd/pull/148
# Tests marked with arbitrary_client_tags can't pass because Unreal whitelists tags it relays
UNREALIRCD_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \
and not strict \
and not testNoticeNonexistentChannel \ and not testNoticeNonexistentChannel \
and not (test_regressions and testTagCap) \ and not (test_regressions and testTagCap) \
and not (test_messages and testLineTooLong) \ and not (test_messages and testLineTooLong) \
and not (test_cap and (testCapRemovalByClient or testNakWhole)) \
and not arbitrary_client_tags \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
.PHONY: all flakes ergo charybdis .PHONY: all flakes charybdis ergo inspircd mammon limnoria sopel solanum unrealircd
all: flakes ergo inspircd limnoria sopel solanum all: flakes charybdis ergo inspircd mammon limnoria sopel solanum unrealircd
flakes: flakes:
pyflakes3 irctest pyflakes3 irctest
@ -97,3 +103,6 @@ solanum:
sopel: sopel:
$(PYTEST) $(PYTEST_ARGS) --controller=irctest.controllers.sopel -k '$(SOPEL_SELECTORS)' $(PYTEST) $(PYTEST_ARGS) --controller=irctest.controllers.sopel -k '$(SOPEL_SELECTORS)'
unrealircd:
$(PYTEST) $(PYTEST_ARGS) --controller=irctest.controllers.unrealircd -k '$(UNREALIRCD_SELECTORS)'

View File

@ -183,6 +183,8 @@ class BaseServerController(_BaseController):
port: int port: int
hostname: str hostname: str
services_controller: BaseServicesController services_controller: BaseServicesController
extban_mute_char: Optional[str] = None
"""Character used for the 'mute' extban"""
def run( def run(
self, self,

View File

@ -73,6 +73,7 @@ class CharybdisController(BaseServerController, DirectoryBasedController):
binary_name = "charybdis" binary_name = "charybdis"
supported_sasl_mechanisms = {"PLAIN"} supported_sasl_mechanisms = {"PLAIN"}
supports_sts = False supports_sts = False
extban_mute_char = None
def create_config(self) -> None: def create_config(self) -> None:
super().create_config() super().create_config()

View File

@ -68,6 +68,7 @@ class InspircdController(BaseServerController, DirectoryBasedController):
software_name = "InspIRCd" software_name = "InspIRCd"
supported_sasl_mechanisms = {"PLAIN"} supported_sasl_mechanisms = {"PLAIN"}
supports_sts = False supports_sts = False
extban_mute_char = "m"
def create_config(self) -> None: def create_config(self) -> None:
super().create_config() super().create_config()

View File

@ -78,14 +78,17 @@ set {{
default-server "irc.example.org"; default-server "irc.example.org";
help-channel "#Help"; help-channel "#Help";
cloak-keys {{ "aaaA1"; "bbbB2"; "cccC3"; }} cloak-keys {{ "aaaA1"; "bbbB2"; "cccC3"; }}
anti-flood {{ options {{
// Prevent throttling, especially test_buffering.py which identd-check; // Disable it, so it doesn't prefix idents with a tilde
// triggers anti-flood with its very long lines }}
unknown-users {{ anti-flood {{
lag-penalty 1; // Prevent throttling, especially test_buffering.py which
lag-penalty-bytes 10000; // triggers anti-flood with its very long lines
}} unknown-users {{
lag-penalty 1;
lag-penalty-bytes 10000;
}} }}
}}
}} }}
tld {{ tld {{
@ -102,6 +105,8 @@ class UnrealircdController(BaseServerController, DirectoryBasedController):
supported_sasl_mechanisms = {"PLAIN"} supported_sasl_mechanisms = {"PLAIN"}
supports_sts = False supports_sts = False
extban_mute_char = "q"
def create_config(self) -> None: def create_config(self) -> None:
super().create_config() super().create_config()
with self.open_file("server.conf"): with self.open_file("server.conf"):
@ -170,7 +175,7 @@ class UnrealircdController(BaseServerController, DirectoryBasedController):
) )
if run_services: if run_services:
assert False raise NotImplementedByController("Registration services")
def get_irctest_controller_class() -> Type[UnrealircdController]: def get_irctest_controller_class() -> Type[UnrealircdController]:

View File

@ -54,6 +54,15 @@ ANYDICT = {RemainingKeys(ANYSTR): AnyOptStr()}
`match_dict(got_tags, {"label": "foo", **ANYDICT})`""" `match_dict(got_tags, {"label": "foo", **ANYDICT})`"""
class _AnyListRemainder:
def __repr__(self) -> str:
return "*ANYLIST"
ANYLIST = [_AnyListRemainder()]
"""Matches any list remainder"""
def match_string(got: Optional[str], expected: Union[str, Operator, None]) -> bool: def match_string(got: Optional[str], expected: Union[str, Operator, None]) -> bool:
if isinstance(expected, AnyOptStr): if isinstance(expected, AnyOptStr):
return True return True
@ -78,6 +87,9 @@ def match_list(
The ANYSTR operator can be used on the 'expected' side as a wildcard, The ANYSTR operator can be used on the 'expected' side as a wildcard,
matching any *single* value; and StrRe("<regexp>") can be used to match regular matching any *single* value; and StrRe("<regexp>") can be used to match regular
expressions""" expressions"""
if expected[-1] is ANYLIST[0]:
expected = expected[0:-1]
got = got[0 : len(expected)] # Ignore remaining
if len(got) != len(expected): if len(got) != len(expected):
return False return False
return all( return all(

View File

@ -26,7 +26,7 @@ from irctest.numerics import (
RPL_TOPIC, RPL_TOPIC,
RPL_TOPICTIME, RPL_TOPICTIME,
) )
from irctest.patma import ANYSTR, StrRe from irctest.patma import ANYLIST, ANYSTR, StrRe
MODERN_CAPS = [ MODERN_CAPS = [
"server-time", "server-time",
@ -1296,16 +1296,13 @@ class OpModerated(cases.BaseServerTestCase):
class MuteExtban(cases.BaseServerTestCase): class MuteExtban(cases.BaseServerTestCase):
"""https://defs.ircdocs.horse/defs/isupport.html#extban """https://defs.ircdocs.horse/defs/isupport.html#extban
These tests assume that if the server advertizes the 'm' extban, It magically guesses what char the IRCd uses for mutes."""
then it supports mute.
This is not true of Charybdis, which introduced a conflicting 'm' def char(self):
exban for matching hostmasks in 2015 if self.controller.extban_mute_char is None:
(e2a9fa9cab3720215d8081e940109416e8214a29). raise runner.ExtbanNotSupported("", "mute")
else:
But Unreal was already using 'm' for muting since 2008 return self.controller.extban_mute_char
(f474e7e6dc2d36f96150ebe33b23b4ea76814415) and it is the most popular
definition so we're going with that one."""
@cases.mark_specifications("Ergo") @cases.mark_specifications("Ergo")
def testISupport(self): def testISupport(self):
@ -1313,7 +1310,7 @@ class MuteExtban(cases.BaseServerTestCase):
isupport = self.server_support isupport = self.server_support
token = isupport["EXTBAN"] token = isupport["EXTBAN"]
prefix, comma, types = token.partition(",") prefix, comma, types = token.partition(",")
self.assertIn("m", types, "Missing 'm' in ISUPPORT EXTBAN") self.assertIn(self.char, types, f"Missing '{self.char()}' in ISUPPORT EXTBAN")
self.assertEqual(prefix, "") self.assertEqual(prefix, "")
self.assertEqual(comma, ",") self.assertEqual(comma, ",")
@ -1325,15 +1322,15 @@ class MuteExtban(cases.BaseServerTestCase):
isupport = self.server_support isupport = self.server_support
token = isupport.get("EXTBAN", "") token = isupport.get("EXTBAN", "")
prefix, comma, types = token.partition(",") prefix, comma, types = token.partition(",")
if "m" not in types: if self.char() not in types:
raise runner.ExtbanNotSupported("m", "mute") raise runner.ExtbanNotSupported(self.char(), "mute")
clients = ("chanop", "bar") clients = ("chanop", "bar")
# Mute "bar" # Mute "bar"
self.joinChannel("chanop", "#chan") self.joinChannel("chanop", "#chan")
self.getMessages("chanop") self.getMessages("chanop")
self.sendLine("chanop", "MODE #chan +b m:bar!*@*") self.sendLine("chanop", f"MODE #chan +b {prefix}{self.char()}:bar!*@*")
replies = {msg.command for msg in self.getMessages("chanop")} replies = {msg.command for msg in self.getMessages("chanop")}
self.assertIn("MODE", replies) self.assertIn("MODE", replies)
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies) self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
@ -1344,6 +1341,21 @@ class MuteExtban(cases.BaseServerTestCase):
for client in clients: for client in clients:
self.getMessages(client) self.getMessages(client)
# "bar" sees the MODE too
self.sendLine("bar", "MODE #chan +b")
self.assertMessageMatch(
self.getMessage("bar"),
command="367",
params=[
"bar",
"#chan",
f"{prefix}{self.char()}:bar!*@*",
"chanop",
*ANYLIST,
],
)
self.getMessages("bar")
# "bar" talks: rejected # "bar" talks: rejected
self.sendLine("bar", "PRIVMSG #chan :hi from bar") self.sendLine("bar", "PRIVMSG #chan :hi from bar")
replies = self.getMessages("bar") replies = self.getMessages("bar")
@ -1354,7 +1366,7 @@ class MuteExtban(cases.BaseServerTestCase):
# remove mute on "bar" with -b # remove mute on "bar" with -b
self.getMessages("chanop") self.getMessages("chanop")
self.sendLine("chanop", "MODE #chan -b m:bar!*@*") self.sendLine("chanop", f"MODE #chan -b {prefix}{self.char()}:bar!*@*")
replies = {msg.command for msg in self.getMessages("chanop")} replies = {msg.command for msg in self.getMessages("chanop")}
self.assertIn("MODE", replies) self.assertIn("MODE", replies)
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies) self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
@ -1378,15 +1390,15 @@ class MuteExtban(cases.BaseServerTestCase):
isupport = self.server_support isupport = self.server_support
token = isupport.get("EXTBAN", "") token = isupport.get("EXTBAN", "")
prefix, comma, types = token.partition(",") prefix, comma, types = token.partition(",")
if "m" not in types: if self.char() not in types:
raise runner.ExtbanNotSupported("m", "mute") raise runner.ExtbanNotSupported(self.char(), "mute")
clients = ("chanop", "qux") clients = ("chanop", "qux")
# Mute "qux" # Mute "qux"
self.joinChannel("chanop", "#chan") self.joinChannel("chanop", "#chan")
self.getMessages("chanop") self.getMessages("chanop")
self.sendLine("chanop", "MODE #chan +b m:qux!*@*") self.sendLine("chanop", f"MODE #chan +b {prefix}{self.char()}:qux!*@*")
replies = {msg.command for msg in self.getMessages("chanop")} replies = {msg.command for msg in self.getMessages("chanop")}
self.assertIn("MODE", replies) self.assertIn("MODE", replies)
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies) self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
@ -1437,17 +1449,17 @@ class MuteExtban(cases.BaseServerTestCase):
isupport = self.server_support isupport = self.server_support
token = isupport.get("EXTBAN", "") token = isupport.get("EXTBAN", "")
prefix, comma, types = token.partition(",") prefix, comma, types = token.partition(",")
if "m" not in types: if self.char() not in types:
raise runner.ExtbanNotSupported("m", "mute") raise runner.ExtbanNotSupported(self.char(), "mute")
if "e" not in self.server_support["CHANMODES"]: if "e" not in self.server_support["CHANMODES"]:
raise runner.ChannelModeNotSupported("m", "mute") raise runner.ChannelModeNotSupported(self.char(), "mute")
clients = ("chanop", "qux") clients = ("chanop", "qux")
# Mute "qux" # Mute "qux"
self.joinChannel("chanop", "#chan") self.joinChannel("chanop", "#chan")
self.getMessages("chanop") self.getMessages("chanop")
self.sendLine("chanop", "MODE #chan +b m:qux!*@*") self.sendLine("chanop", f"MODE #chan +b {prefix}{self.char()}:qux!*@*")
replies = {msg.command for msg in self.getMessages("chanop")} replies = {msg.command for msg in self.getMessages("chanop")}
self.assertIn("MODE", replies) self.assertIn("MODE", replies)
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies) self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
@ -1472,11 +1484,13 @@ class MuteExtban(cases.BaseServerTestCase):
self.getMessages(client) self.getMessages(client)
# +e grants an exemption to +b # +e grants an exemption to +b
self.sendLine("chanop", "MODE #chan +e m:*!~evan@*") self.sendLine("chanop", f"MODE #chan +e {prefix}{self.char()}:*!~evan@*")
replies = {msg.command for msg in self.getMessages("chanop")} replies = {msg.command for msg in self.getMessages("chanop")}
self.assertIn("MODE", replies) self.assertIn("MODE", replies)
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies) self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
self.getMessages("qux")
# so "qux" can now talk # so "qux" can now talk
self.sendLine("qux", "PRIVMSG #chan :thanks for mute-excepting me") self.sendLine("qux", "PRIVMSG #chan :thanks for mute-excepting me")
replies = self.getMessages("qux") replies = self.getMessages("qux")
@ -1499,10 +1513,14 @@ class MuteExtban(cases.BaseServerTestCase):
""" """
clients = ("chanop", "bar") clients = ("chanop", "bar")
isupport = self.server_support
token = isupport.get("EXTBAN", "")
prefix, comma, types = token.partition(",")
self.connectClient("chanop", name="chanop") self.connectClient("chanop", name="chanop")
self.joinChannel("chanop", "#chan") self.joinChannel("chanop", "#chan")
self.getMessages("chanop") self.getMessages("chanop")
self.sendLine("chanop", "MODE #chan +b m:BAR!*@*") self.sendLine("chanop", f"MODE #chan +b {prefix}{self.char()}:BAR!*@*")
replies = {msg.command for msg in self.getMessages("chanop")} replies = {msg.command for msg in self.getMessages("chanop")}
self.assertIn("MODE", replies) self.assertIn("MODE", replies)
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies) self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
@ -1521,7 +1539,7 @@ class MuteExtban(cases.BaseServerTestCase):
self.assertEqual(self.getMessages("chanop"), []) self.assertEqual(self.getMessages("chanop"), [])
# remove mute with -b # remove mute with -b
self.sendLine("chanop", "MODE #chan -b m:bar!*@*") self.sendLine("chanop", f"MODE #chan -b {prefix}{self.char()}:bar!*@*")
replies = {msg.command for msg in self.getMessages("chanop")} replies = {msg.command for msg in self.getMessages("chanop")}
self.assertIn("MODE", replies) self.assertIn("MODE", replies)
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies) self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)

View File

@ -119,15 +119,38 @@ class ConnectionRegistrationTestCase(cases.BaseServerTestCase):
self.sendLine(2, "NICK foo") self.sendLine(2, "NICK foo")
self.sendLine(1, "USER username * * :Realname") self.sendLine(1, "USER username * * :Realname")
self.sendLine(2, "USER username * * :Realname") self.sendLine(2, "USER username * * :Realname")
m1 = self.getRegistrationMessage(1)
m2 = self.getRegistrationMessage(2) try:
m1 = self.getRegistrationMessage(1)
except (ConnectionClosed, ConnectionResetError):
# Unreal closes the connection, see
# https://bugs.unrealircd.org/view.php?id=5950
command1 = None
else:
command1 = m1.command
try:
m2 = self.getRegistrationMessage(2)
except (ConnectionClosed, ConnectionResetError):
# ditto
command2 = None
else:
command2 = m2.command
self.assertNotEqual( self.assertNotEqual(
(m1.command, m2.command), (command1, command2),
("001", "001"), ("001", "001"),
"Two concurrently registering requesting the same nickname " "Two concurrently registering requesting the same nickname "
"both got 001.", "both got 001.",
) )
self.assertIn(
"001",
(command1, command2),
"Two concurrently registering requesting the same nickname "
"neither got 001.",
)
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
def testIrc301CapLs(self): def testIrc301CapLs(self):
""" """

View File

@ -2,6 +2,8 @@
<http://ircv3.net/specs/extensions/echo-message-3.2.html> <http://ircv3.net/specs/extensions/echo-message-3.2.html>
""" """
import pytest
from irctest import cases from irctest import cases
from irctest.basecontrollers import NotImplementedByController from irctest.basecontrollers import NotImplementedByController
from irctest.irc_utils.junkdrawer import random_name from irctest.irc_utils.junkdrawer import random_name
@ -97,6 +99,7 @@ def _testEchoMessage(command, solo, server_time):
class EchoMessageTestCase(cases.BaseServerTestCase): class EchoMessageTestCase(cases.BaseServerTestCase):
@pytest.mark.arbitrary_client_tags
@cases.mark_capabilities( @cases.mark_capabilities(
"batch", "labeled-response", "echo-message", "message-tags" "batch", "labeled-response", "echo-message", "message-tags"
) )

View File

@ -282,7 +282,14 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
) )
self.getMessages(2) self.getMessages(2)
self.sendLine(1, "@label=12345;+draft/reply=123;+draft/react=l😃l TAGMSG bar") # Need to get a valid msgid because Unreal validates them
self.sendLine(1, "PRIVMSG bar :hi")
msgid = self.getMessage(1).tags["msgid"]
assert msgid == self.getMessage(2).tags["msgid"]
self.sendLine(
1, f"@label=12345;+draft/reply={msgid};+draft/react=l😃l TAGMSG bar"
)
m = self.getMessage(1) m = self.getMessage(1)
m2 = self.getMessage(2) m2 = self.getMessage(2)
@ -290,7 +297,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
self.assertMessageMatch( self.assertMessageMatch(
m2, m2,
command="TAGMSG", command="TAGMSG",
tags={"+draft/reply": "123", "+draft/react": "l😃l", **ANYDICT}, tags={"+draft/reply": msgid, "+draft/react": "l😃l", **ANYDICT},
) )
self.assertNotIn( self.assertNotIn(
"label", "label",
@ -308,7 +315,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
command="TAGMSG", command="TAGMSG",
tags={ tags={
"label": "12345", "label": "12345",
"+draft/reply": "123", "+draft/reply": msgid,
"+draft/react": "l😃l", "+draft/react": "l😃l",
**ANYDICT, **ANYDICT,
}, },
@ -338,7 +345,14 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
self.getMessages(2) self.getMessages(2)
self.getMessages(1) self.getMessages(1)
self.sendLine(1, "@label=12345;+draft/reply=123;+draft/react=l😃l TAGMSG #test") # Need to get a valid msgid because Unreal validates them
self.sendLine(1, "PRIVMSG #test :hi")
msgid = self.getMessage(1).tags["msgid"]
assert msgid == self.getMessage(2).tags["msgid"]
self.sendLine(
1, f"@label=12345;+draft/reply={msgid};+draft/react=l😃l TAGMSG #test"
)
ms = self.getMessage(1) ms = self.getMessage(1)
mt = self.getMessage(2) mt = self.getMessage(2)
@ -361,7 +375,9 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
# ensure sender correctly receives msg # ensure sender correctly receives msg
self.assertMessageMatch( self.assertMessageMatch(
ms, command="TAGMSG", tags={"label": "12345", **ANYDICT} ms,
command="TAGMSG",
tags={"label": "12345", "+draft/reply": msgid, **ANYDICT},
) )
@cases.mark_capabilities( @cases.mark_capabilities(

View File

@ -2,6 +2,8 @@
https://ircv3.net/specs/extensions/message-tags.html https://ircv3.net/specs/extensions/message-tags.html
""" """
import pytest
from irctest import cases from irctest import cases
from irctest.irc_utils.message_parser import parse_message from irctest.irc_utils.message_parser import parse_message
from irctest.numerics import ERR_INPUTTOOLONG from irctest.numerics import ERR_INPUTTOOLONG
@ -9,6 +11,7 @@ from irctest.patma import ANYDICT, ANYSTR, StrRe
class MessageTagsTestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class MessageTagsTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
@pytest.mark.arbitrary_client_tags
@cases.mark_capabilities("message-tags") @cases.mark_capabilities("message-tags")
def testBasic(self): def testBasic(self):
def getAllMessages(): def getAllMessages():
@ -107,6 +110,7 @@ class MessageTagsTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
self.assertNotIn("cat", msg.tags) self.assertNotIn("cat", msg.tags)
self.assertEqual(alice_msg.tags["msgid"], bob_msg.tags["msgid"]) self.assertEqual(alice_msg.tags["msgid"], bob_msg.tags["msgid"])
@pytest.mark.arbitrary_client_tags
@cases.mark_capabilities("message-tags") @cases.mark_capabilities("message-tags")
@cases.mark_specifications("ircdocs") @cases.mark_specifications("ircdocs")
def testLengthLimits(self): def testLengthLimits(self):

View File

@ -12,6 +12,7 @@ markers =
strict strict
deprecated deprecated
services services
arbitrary_client_tags
# capabilities # capabilities
account-tag account-tag