mirror of
https://github.com/progval/irctest.git
synced 2025-04-05 06:49:47 +00:00
Change IRCv3 marks to reference capabilities instead of v3.1 / v3.2
This commit is contained in:
@ -20,7 +20,7 @@ from .numerics import (
|
||||
ERR_NOSUCHCHANNEL,
|
||||
ERR_TOOMANYCHANNELS,
|
||||
)
|
||||
from .specifications import Specifications
|
||||
from .specifications import Capabilities, IsupportTokens, Specifications
|
||||
|
||||
CHANNEL_JOIN_FAIL_NUMERICS = frozenset(
|
||||
[
|
||||
@ -489,7 +489,7 @@ class BaseServerTestCase(_IrcTestCase):
|
||||
)
|
||||
except AssertionError:
|
||||
if skip_if_cap_nak:
|
||||
raise runner.NotImplementedByController(", ".join(capabilities))
|
||||
raise runner.CapabilityNotSupported(" or ".join(capabilities))
|
||||
else:
|
||||
raise
|
||||
self.sendLine(client, "CAP END")
|
||||
@ -614,7 +614,7 @@ class OptionalityHelper:
|
||||
|
||||
def mark_specifications(*specifications, deprecated=False, strict=False):
|
||||
specifications = frozenset(
|
||||
Specifications.of_name(s) if isinstance(s, str) else s for s in specifications
|
||||
Specifications.from_name(s) if isinstance(s, str) else s for s in specifications
|
||||
)
|
||||
if None in specifications:
|
||||
raise ValueError("Invalid set of specifications: {}".format(specifications))
|
||||
@ -629,3 +629,35 @@ def mark_specifications(*specifications, deprecated=False, strict=False):
|
||||
return f
|
||||
|
||||
return decorator
|
||||
|
||||
|
||||
def mark_capabilities(*capabilities, deprecated=False, strict=False):
|
||||
capabilities = frozenset(
|
||||
Capabilities.from_name(c) if isinstance(c, str) else c for c in capabilities
|
||||
)
|
||||
if None in capabilities:
|
||||
raise ValueError("Invalid set of capabilities: {}".format(capabilities))
|
||||
|
||||
def decorator(f):
|
||||
for capability in capabilities:
|
||||
f = getattr(pytest.mark, capability.value)(f)
|
||||
# Support for any capability implies IRCv3
|
||||
f = pytest.mark.IRCv3(f)
|
||||
return f
|
||||
|
||||
return decorator
|
||||
|
||||
|
||||
def mark_isupport(*tokens, deprecated=False, strict=False):
|
||||
tokens = frozenset(
|
||||
IsupportTokens.from_name(c) if isinstance(c, str) else c for c in tokens
|
||||
)
|
||||
if None in tokens:
|
||||
raise ValueError("Invalid set of isupport tokens: {}".format(tokens))
|
||||
|
||||
def decorator(f):
|
||||
for token in tokens:
|
||||
f = getattr(pytest.mark, token.value)(f)
|
||||
return f
|
||||
|
||||
return decorator
|
||||
|
@ -3,12 +3,12 @@ from irctest.irc_utils.message_parser import Message
|
||||
|
||||
|
||||
class CapTestCase(cases.BaseClientTestCase, cases.ClientNegociationHelper):
|
||||
@cases.mark_specifications("IRCv3.1", "IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testSendCap(self):
|
||||
"""Send CAP LS 302 and read the result."""
|
||||
self.readCapLs()
|
||||
|
||||
@cases.mark_specifications("IRCv3.1", "IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testEmptyCapLs(self):
|
||||
"""Empty result to CAP LS. Client should send CAP END."""
|
||||
m = self.negotiateCapabilities([])
|
||||
|
@ -30,6 +30,11 @@ class CapabilityNotSupported(unittest.SkipTest):
|
||||
return "Unsupported capability: {}".format(self.args[0])
|
||||
|
||||
|
||||
class IsupportTokenNotSupported(unittest.SkipTest):
|
||||
def __str__(self):
|
||||
return "Unsupported ISUPPORT token: {}".format(self.args[0])
|
||||
|
||||
|
||||
class NotRequiredBySpecifications(unittest.SkipTest):
|
||||
def __str__(self):
|
||||
return "Tests not required by the set of tested specification(s)."
|
||||
|
@ -32,7 +32,7 @@ class AccountTagTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
self.sendLine(2, "CAP END")
|
||||
self.skipToWelcome(2)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN")
|
||||
def testPrivmsg(self):
|
||||
self.connectClient("foo", capabilities=["account-tag"], skip_if_cap_nak=True)
|
||||
|
@ -6,7 +6,7 @@ from irctest import cases
|
||||
|
||||
|
||||
class AwayNotifyTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_capabilities("away-notify")
|
||||
def testAwayNotify(self):
|
||||
"""Basic away-notify test."""
|
||||
self.connectClient("foo", capabilities=["away-notify"], skip_if_cap_nak=True)
|
||||
@ -29,7 +29,7 @@ class AwayNotifyTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
"Unexpected away-notify source: %s" % (awayNotify.prefix,),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("away-notify")
|
||||
def testAwayNotifyOnJoin(self):
|
||||
"""The away-notify specification states:
|
||||
"Clients will be sent an AWAY message [...] when a user joins
|
||||
|
@ -3,7 +3,7 @@ from irctest.runner import CapabilityNotSupported, ImplementationChoice
|
||||
|
||||
|
||||
class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testNoReq(self):
|
||||
"""Test the server handles gracefully clients which do not send
|
||||
REQs.
|
||||
@ -23,7 +23,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
m, command="001", fail_msg="Expected 001 after sending CAP END, got {msg}."
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testReqUnavailable(self):
|
||||
"""Test the server handles gracefully clients which request
|
||||
capabilities that are not available.
|
||||
@ -50,7 +50,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
m, command="001", fail_msg="Expected 001 after sending CAP END, got {msg}."
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testNakExactString(self):
|
||||
"""“The argument of the NAK subcommand MUST consist of at least the
|
||||
first 100 characters of the capability list in the REQ subcommand which
|
||||
@ -73,7 +73,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
"sending “CAP REQ :foo qux bar baz qux quux”, but got {msg}.",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testNakWhole(self):
|
||||
"""“The capability identifier set must be accepted as a whole, or
|
||||
rejected entirely.”
|
||||
@ -124,7 +124,7 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
"sending “CAP REQ :multi-prefix”, but got {msg}.",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testCapRemovalByClient(self):
|
||||
"""Test CAP LIST and removal of caps via CAP REQ :-tagname."""
|
||||
cap1 = "echo-message"
|
||||
|
@ -712,9 +712,13 @@ class testChannelCaseSensitivity(cases.BaseServerTestCase):
|
||||
|
||||
|
||||
class InviteTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("Modern")
|
||||
def testInvites(self):
|
||||
"""Test some basic functionality related to INVITE and the +i mode."""
|
||||
"""Test some basic functionality related to INVITE and the +i mode.
|
||||
|
||||
https://modern.ircdocs.horse/#invite-only-channel-mode
|
||||
https://modern.ircdocs.horse/#rplinviting-341
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.joinChannel(1, "#chan")
|
||||
self.sendLine(1, "MODE #chan +i")
|
||||
|
@ -128,9 +128,22 @@ class ConnectionRegistrationTestCase(cases.BaseServerTestCase):
|
||||
"both got 001.",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1", "IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testIrc301CapLs(self):
|
||||
"""IRCv3.1: “The LS subcommand is used to list the capabilities
|
||||
"""
|
||||
Current version:
|
||||
|
||||
"The LS subcommand is used to list the capabilities supported by the server.
|
||||
The client should send an LS subcommand with no other arguments to solicit
|
||||
a list of all capabilities."
|
||||
|
||||
"If a client has not indicated support for CAP LS 302 features,
|
||||
the server MUST NOT send these new features to the client."
|
||||
-- <https://ircv3.net/specs/core/capability-negotiation.html>
|
||||
|
||||
Before the v3.1 / v3.2 merge:
|
||||
|
||||
IRCv3.1: “The LS subcommand is used to list the capabilities
|
||||
supported by the server. The client should send an LS subcommand with
|
||||
no other arguments to solicit a list of all capabilities.”
|
||||
-- <http://ircv3.net/specs/core/capability-negotiation-3.1.html#the-cap-ls-subcommand>
|
||||
@ -156,7 +169,7 @@ class ConnectionRegistrationTestCase(cases.BaseServerTestCase):
|
||||
"request: {}".format(m),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testEmptyCapList(self):
|
||||
"""“If no capabilities are active, an empty parameter must be sent.”
|
||||
-- <http://ircv3.net/specs/core/capability-negotiation-3.1.html#the-cap-list-subcommand>
|
||||
|
@ -58,7 +58,7 @@ class DMEchoMessageTestCase(cases.BaseServerTestCase):
|
||||
|
||||
class EchoMessageTestCase(cases.BaseServerTestCase):
|
||||
def _testEchoMessage(command, solo, server_time):
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message")
|
||||
def f(self):
|
||||
"""<http://ircv3.net/specs/extensions/echo-message-3.2.html>"""
|
||||
self.addClient()
|
||||
|
@ -32,7 +32,7 @@ class MetadataTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
self.sendLine(2, "CAP END")
|
||||
self.skipToWelcome(2)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_capabilities("extended-join")
|
||||
def testNotLoggedIn(self):
|
||||
self.connectClient("foo", capabilities=["extended-join"], skip_if_cap_nak=True)
|
||||
self.joinChannel(1, "#chan")
|
||||
@ -47,7 +47,7 @@ class MetadataTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
"unregistered user joined, got: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_capabilities("extended-join")
|
||||
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN")
|
||||
def testLoggedIn(self):
|
||||
self.connectClient("foo", capabilities=["extended-join"], skip_if_cap_nak=True)
|
||||
|
@ -8,7 +8,7 @@ from irctest import cases
|
||||
|
||||
|
||||
class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("batch", "echo-message", "labeled-response")
|
||||
def testLabeledPrivmsgResponsesToMultipleClients(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -92,7 +92,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
m, command="BATCH", fail_msg="No BATCH echo received after sending one out"
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message", "labeled-response")
|
||||
def testLabeledPrivmsgResponsesToClient(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -152,7 +152,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message", "labeled-response")
|
||||
def testLabeledPrivmsgResponsesToChannel(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -220,7 +220,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message", "labeled-response")
|
||||
def testLabeledPrivmsgResponsesToSelf(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -263,7 +263,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
).format(number_of_labels),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message", "labeled-response")
|
||||
def testLabeledNoticeResponsesToClient(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -323,7 +323,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message", "labeled-response")
|
||||
def testLabeledNoticeResponsesToChannel(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -391,7 +391,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message", "labeled-response")
|
||||
def testLabeledNoticeResponsesToSelf(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -432,7 +432,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
).format(number_of_labels),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message", "labeled-response", "message-tags")
|
||||
def testLabeledTagMsgResponsesToClient(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -540,7 +540,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
fail_msg="React tag wasn't the same on the source user's TAGMSG: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message", "labeled-response", "message-tags")
|
||||
def testLabeledTagMsgResponsesToChannel(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -606,7 +606,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("echo-message", "labeled-response", "message-tags")
|
||||
def testLabeledTagMsgResponsesToSelf(self):
|
||||
self.connectClient(
|
||||
"foo",
|
||||
@ -647,7 +647,9 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
|
||||
).format(number_of_labels),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities(
|
||||
"echo-message", "labeled-response", "message-tags", "server-time"
|
||||
)
|
||||
def testBatchedJoinMessages(self):
|
||||
self.connectClient(
|
||||
"bar",
|
||||
|
@ -8,7 +8,7 @@ from irctest.numerics import ERR_INPUTTOOLONG
|
||||
|
||||
|
||||
class MessageTagsTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
@cases.mark_specifications("message-tags")
|
||||
@cases.mark_capabilities("message-tags")
|
||||
def testBasic(self):
|
||||
def getAllMessages():
|
||||
for name in ["alice", "bob", "carol", "dave"]:
|
||||
@ -87,7 +87,7 @@ class MessageTagsTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
self.assertTrue(alice_msg.tags["msgid"])
|
||||
self.assertEqual(alice_msg.tags["msgid"], bob_msg.tags["msgid"])
|
||||
|
||||
@cases.mark_specifications("message-tags")
|
||||
@cases.mark_capabilities("message-tags")
|
||||
def testLengthLimits(self):
|
||||
self.connectClient(
|
||||
"alice",
|
||||
|
@ -10,7 +10,7 @@ class MetadataTestCase(cases.BaseServerTestCase):
|
||||
valid_metadata_keys = {"valid_key1", "valid_key2"}
|
||||
invalid_metadata_keys = {"invalid_key1", "invalid_key2"}
|
||||
|
||||
@cases.mark_specifications("IRCv3.2", deprecated=True)
|
||||
@cases.mark_specifications("IRCv3", deprecated=True)
|
||||
def testInIsupport(self):
|
||||
"""“If METADATA is supported, it MUST be specified in RPL_ISUPPORT
|
||||
using the METADATA key.”
|
||||
@ -33,7 +33,7 @@ class MetadataTestCase(cases.BaseServerTestCase):
|
||||
)
|
||||
self.getMessages(1)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2", deprecated=True)
|
||||
@cases.mark_specifications("IRCv3", deprecated=True)
|
||||
def testGetOneUnsetValid(self):
|
||||
"""<http://ircv3.net/specs/core/metadata-3.2.html#metadata-get>"""
|
||||
self.connectClient("foo")
|
||||
@ -46,7 +46,7 @@ class MetadataTestCase(cases.BaseServerTestCase):
|
||||
"request to an unset valid METADATA key.",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2", deprecated=True)
|
||||
@cases.mark_specifications("IRCv3", deprecated=True)
|
||||
def testGetTwoUnsetValid(self):
|
||||
"""“Multiple keys may be given. The response will be either RPL_KEYVALUE,
|
||||
ERR_KEYINVALID or ERR_NOMATCHINGKEY for every key in order.”
|
||||
@ -83,7 +83,7 @@ class MetadataTestCase(cases.BaseServerTestCase):
|
||||
"did not respond to valid_key2 as second response: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2", deprecated=True)
|
||||
@cases.mark_specifications("IRCv3", deprecated=True)
|
||||
def testListNoSet(self):
|
||||
"""“This subcommand MUST list all currently-set metadata keys along
|
||||
with their values. The response will be zero or more RPL_KEYVALUE
|
||||
@ -100,7 +100,7 @@ class MetadataTestCase(cases.BaseServerTestCase):
|
||||
"762 (RPL_METADATAEND) but: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2", deprecated=True)
|
||||
@cases.mark_specifications("IRCv3", deprecated=True)
|
||||
def testListInvalidTarget(self):
|
||||
"""“In case of invalid target RPL_METADATAEND MUST NOT be sent.”
|
||||
-- <http://ircv3.net/specs/core/metadata-3.2.html#metadata-list>
|
||||
@ -187,13 +187,13 @@ class MetadataTestCase(cases.BaseServerTestCase):
|
||||
self.assertSetValue(target, key, value, displayable_value)
|
||||
self.assertGetValue(target, key, value, displayable_value)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2", deprecated=True)
|
||||
@cases.mark_specifications("IRCv3", deprecated=True)
|
||||
def testSetGetValid(self):
|
||||
"""<http://ircv3.net/specs/core/metadata-3.2.html>"""
|
||||
self.connectClient("foo")
|
||||
self.assertSetGetValue("*", "valid_key1", "myvalue")
|
||||
|
||||
@cases.mark_specifications("IRCv3.2", deprecated=True)
|
||||
@cases.mark_specifications("IRCv3", deprecated=True)
|
||||
def testSetGetZeroCharInValue(self):
|
||||
"""“Values are unrestricted, except that they MUST be UTF-8.”
|
||||
-- <http://ircv3.net/specs/core/metadata-3.2.html#metadata-restrictions>
|
||||
@ -201,7 +201,7 @@ class MetadataTestCase(cases.BaseServerTestCase):
|
||||
self.connectClient("foo")
|
||||
self.assertSetGetValue("*", "valid_key1", "zero->\0<-zero", "zero->\\0<-zero")
|
||||
|
||||
@cases.mark_specifications("IRCv3.2", deprecated=True)
|
||||
@cases.mark_specifications("IRCv3", deprecated=True)
|
||||
def testSetGetHeartInValue(self):
|
||||
"""“Values are unrestricted, except that they MUST be UTF-8.”
|
||||
-- <http://ircv3.net/specs/core/metadata-3.2.html#metadata-restrictions>
|
||||
@ -215,7 +215,7 @@ class MetadataTestCase(cases.BaseServerTestCase):
|
||||
"zero->{}<-zero".format(heart.encode()),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2", deprecated=True)
|
||||
@cases.mark_specifications("IRCv3", deprecated=True)
|
||||
def testSetInvalidUtf8(self):
|
||||
"""“Values are unrestricted, except that they MUST be UTF-8.”
|
||||
-- <http://ircv3.net/specs/core/metadata-3.2.html#metadata-restrictions>
|
||||
|
@ -66,7 +66,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
extra_format=(nick,),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testMonitorOneDisconnected(self):
|
||||
"""“If any of the targets being added are online, the server will
|
||||
generate RPL_MONONLINE numerics listing those targets that are
|
||||
@ -86,7 +87,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
pass
|
||||
self.assertMonoffline(1, "bar")
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testMonitorOneConnection(self):
|
||||
self.connectClient("foo")
|
||||
self.check_server_support()
|
||||
@ -95,7 +97,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
self.connectClient("bar")
|
||||
self.assertMononline(1, "bar")
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testMonitorOneConnected(self):
|
||||
"""“If any of the targets being added are offline, the server will
|
||||
generate RPL_MONOFFLINE numerics listing those targets that are
|
||||
@ -114,7 +117,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
pass
|
||||
self.assertMonoffline(1, "bar")
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testMonitorOneConnectionWithQuit(self):
|
||||
self.connectClient("foo")
|
||||
self.check_server_support()
|
||||
@ -130,7 +134,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
self.connectClient("bar")
|
||||
self.assertMononline(1, "bar")
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testMonitorConnectedAndDisconnected(self):
|
||||
"""“If any of the targets being added are online, the server will
|
||||
generate RPL_MONONLINE numerics listing those targets that are
|
||||
@ -185,7 +190,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
"“MONITOR + bar,baz” and “baz” is disconnected: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testUnmonitor(self):
|
||||
self.connectClient("foo")
|
||||
self.check_server_support()
|
||||
@ -210,7 +216,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
fail_msg="Got messages after disconnection of unmonitored " "nick: {got}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testMonitorForbidsMasks(self):
|
||||
"""“The MONITOR implementation also enhances user privacy by
|
||||
disallowing subscription to hostmasks, allowing users to avoid
|
||||
@ -247,7 +254,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
"was requested via hostmask connected: {}".format(m)
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testTwoMonitoringOneRemove(self):
|
||||
"""Tests the following scenario:
|
||||
* foo MONITORs qux
|
||||
@ -286,7 +294,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
extra_format=(messages,),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testMonitorList(self):
|
||||
def checkMonitorSubjects(messages, client_nick, expected_targets):
|
||||
# collect all the RPL_MONLIST nicks into a set:
|
||||
@ -320,7 +329,8 @@ class MonitorTestCase(cases.BaseServerTestCase):
|
||||
self.sendLine(1, "MONITOR L")
|
||||
checkMonitorSubjects(self.getMessages(1), "bar", {"bazbat"})
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.mark_isupport("MONITOR")
|
||||
def testNickChange(self):
|
||||
# see oragono issue #1076: nickname changes must trigger RPL_MONOFFLINE
|
||||
self.connectClient("bar")
|
||||
|
@ -7,7 +7,7 @@ from irctest import cases
|
||||
|
||||
|
||||
class MultiPrefixTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_capabilities("multi-prefix")
|
||||
def testMultiPrefix(self):
|
||||
"""“When requested, the multi-prefix client capability will cause the
|
||||
IRC server to send all possible prefixes which apply to a user in NAMES
|
||||
|
@ -12,7 +12,7 @@ base_caps = ["message-tags", "batch", "echo-message", "server-time", "labeled-re
|
||||
|
||||
|
||||
class MultilineTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
@cases.mark_specifications("multiline")
|
||||
@cases.mark_capabilities("draft/multiline")
|
||||
def testBasic(self):
|
||||
self.connectClient(
|
||||
"alice", capabilities=(base_caps + [CAP_NAME]), skip_if_cap_nak=True
|
||||
@ -79,7 +79,7 @@ class MultilineTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
self.assertNotIn(CONCAT_TAG, msg.tags)
|
||||
self.assertEqual(relayed_fmsgids, [msgid] + [None] * (len(fallback_relay) - 1))
|
||||
|
||||
@cases.mark_specifications("multiline")
|
||||
@cases.mark_capabilities("draft/multiline")
|
||||
def testBlankLines(self):
|
||||
self.connectClient(
|
||||
"alice", capabilities=(base_caps + [CAP_NAME]), skip_if_cap_nak=True
|
||||
|
@ -52,7 +52,7 @@ class RegressionsTestCase(cases.BaseServerTestCase):
|
||||
ms = self.getMessages(2)
|
||||
self.assertEqual(ms, [])
|
||||
|
||||
@cases.mark_specifications("IRCv3.2")
|
||||
@cases.mark_capabilities("message-tags", "batch", "echo-message", "server-time")
|
||||
def testTagCap(self):
|
||||
# regression test for oragono #754
|
||||
self.connectClient(
|
||||
|
@ -9,7 +9,7 @@ class RegistrationTestCase(cases.BaseServerTestCase):
|
||||
|
||||
|
||||
class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN")
|
||||
def testPlain(self):
|
||||
"""PLAIN authentication with correct username/password."""
|
||||
@ -54,7 +54,7 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
"({expects}), not {got}: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN")
|
||||
def testPlainNoAuthzid(self):
|
||||
"""“message = [authzid] UTF8NUL authcid UTF8NUL passwd
|
||||
@ -119,7 +119,7 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
"({expects}), not {got}: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
def testMechanismNotAvailable(self):
|
||||
"""“If authentication fails, a 904 or 905 numeric will be sent”
|
||||
-- <http://ircv3.net/specs/extensions/sasl-3.1.html#the-authenticate-command>
|
||||
@ -141,7 +141,7 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
fail_msg="Did not reply with 904 to “AUTHENTICATE FOO”: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN")
|
||||
def testPlainLarge(self):
|
||||
"""Test the client splits large AUTHENTICATE messages whose payload
|
||||
@ -202,7 +202,7 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
|
||||
# I don't know how to do it, because it would make the registration
|
||||
# message's length too big for it to be valid.
|
||||
|
||||
@cases.mark_specifications("IRCv3.1")
|
||||
@cases.mark_specifications("IRCv3")
|
||||
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN")
|
||||
def testPlainLargeEquals400(self):
|
||||
"""Test the client splits large AUTHENTICATE messages whose payload
|
||||
|
@ -5,16 +5,51 @@ import enum
|
||||
class Specifications(enum.Enum):
|
||||
RFC1459 = "RFC1459"
|
||||
RFC2812 = "RFC2812"
|
||||
IRC301 = "IRCv3.1"
|
||||
IRC302 = "IRCv3.2"
|
||||
IRCv3 = "IRCv3" # Mark with capabilities whenever possible
|
||||
Oragono = "Oragono"
|
||||
Multiline = "multiline"
|
||||
MessageTags = "message-tags"
|
||||
|
||||
Ircdocs = "ircdocs"
|
||||
"""Any document on ircdocs.horse (especially defs.ircdocs.horse),
|
||||
excluding modern.ircdocs.horse"""
|
||||
|
||||
Modern = "modern"
|
||||
|
||||
@classmethod
|
||||
def of_name(cls, name):
|
||||
def from_name(cls, name):
|
||||
name = name.upper()
|
||||
for spec in cls:
|
||||
if spec.value.upper() == name:
|
||||
return spec
|
||||
raise ValueError(name)
|
||||
|
||||
|
||||
@enum.unique
|
||||
class Capabilities(enum.Enum):
|
||||
AWAY_NOTIFY = "away-notify"
|
||||
BATCH = "batch"
|
||||
ECHO_MESSAGE = "echo-message"
|
||||
EXTENDED_JOIN = "extended-join"
|
||||
LABELED_RESPONSE = "labeled-response"
|
||||
MESSAGE_TAGS = "message-tags"
|
||||
MULTILINE = "draft/multiline"
|
||||
MULTI_PREFIX = "multi-prefix"
|
||||
SERVER_TIME = "server-time"
|
||||
|
||||
@classmethod
|
||||
def from_name(cls, name):
|
||||
try:
|
||||
return cls(name.lower())
|
||||
except ValueError:
|
||||
raise ValueError(name) from None
|
||||
|
||||
|
||||
@enum.unique
|
||||
class IsupportTokens(enum.Enum):
|
||||
MONITOR = "MONITOR"
|
||||
|
||||
@classmethod
|
||||
def from_name(cls, name):
|
||||
try:
|
||||
return cls(name.upper())
|
||||
except ValueError:
|
||||
raise ValueError(name) from None
|
||||
|
24
pytest.ini
24
pytest.ini
@ -1,11 +1,27 @@
|
||||
[pytest]
|
||||
markers =
|
||||
# specifications
|
||||
RFC1459
|
||||
RFC2812
|
||||
IRCv3.1
|
||||
IRCv3.2
|
||||
message-tags
|
||||
multiline
|
||||
IRCv3
|
||||
modern
|
||||
ircdocs
|
||||
Oragono
|
||||
|
||||
# misc marks
|
||||
strict
|
||||
deprecated
|
||||
|
||||
# capabilities
|
||||
away-notify
|
||||
batch
|
||||
echo-message
|
||||
extended-join
|
||||
labeled-response
|
||||
message-tags
|
||||
draft/multiline
|
||||
multi-prefix
|
||||
server-time
|
||||
|
||||
# isupport tokens
|
||||
MONITOR
|
||||
|
Reference in New Issue
Block a user