mirror of
https://github.com/progval/irctest.git
synced 2025-04-06 07:19:54 +00:00
Make all tests pass with Unreal (minus service tests)
This commit is contained in:
15
Makefile
15
Makefile
@ -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)'
|
||||||
|
@ -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,
|
||||||
|
@ -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()
|
||||||
|
@ -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()
|
||||||
|
@ -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]:
|
||||||
|
@ -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(
|
||||||
|
@ -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)
|
||||||
|
@ -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):
|
||||||
"""
|
"""
|
||||||
|
@ -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"
|
||||||
)
|
)
|
||||||
|
@ -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(
|
||||||
|
@ -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):
|
||||||
|
@ -12,6 +12,7 @@ markers =
|
|||||||
strict
|
strict
|
||||||
deprecated
|
deprecated
|
||||||
services
|
services
|
||||||
|
arbitrary_client_tags
|
||||||
|
|
||||||
# capabilities
|
# capabilities
|
||||||
account-tag
|
account-tag
|
||||||
|
Reference in New Issue
Block a user