mirror of
https://github.com/progval/irctest.git
synced 2025-04-03 22:09:46 +00:00
Split channel_operations.py by command.
It was messy.
This commit is contained in:
@ -54,7 +54,7 @@ Linking to an external document saying "Here is how everyone does it" is.
|
||||
|
||||
If reference documents / documentations are long or not trivial,
|
||||
**try to quote the specific part being tested**.
|
||||
See `irctest/server_tests/test_channel_operations.py` for example.
|
||||
See `irctest/server_tests/kick.py` for example.
|
||||
|
||||
Tests for **pending/draft specifications are welcome**.
|
||||
|
||||
|
61
irctest/server_tests/channel.py
Normal file
61
irctest/server_tests/channel.py
Normal file
@ -0,0 +1,61 @@
|
||||
import pytest
|
||||
|
||||
from irctest import cases, client_mock, runner
|
||||
|
||||
|
||||
class TestChannelCaseSensitivity(cases.BaseServerTestCase):
|
||||
@pytest.mark.parametrize(
|
||||
"casemapping,name1,name2",
|
||||
[
|
||||
("ascii", "#Foo", "#foo"),
|
||||
("rfc1459", "#Foo", "#foo"),
|
||||
("rfc1459", "#F]|oo{", "#f}\\oo["),
|
||||
("rfc1459", "#F}o\\o[", "#f]o|o{"),
|
||||
],
|
||||
)
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", strict=True)
|
||||
def testChannelsEquivalent(self, casemapping, name1, name2):
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
if self.server_support["CASEMAPPING"] != casemapping:
|
||||
raise runner.NotImplementedByController(
|
||||
"Casemapping {} not implemented".format(casemapping)
|
||||
)
|
||||
self.joinClient(1, name1)
|
||||
self.joinClient(2, name2)
|
||||
try:
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="JOIN", nick="bar")
|
||||
except client_mock.NoMessageException:
|
||||
raise AssertionError(
|
||||
"Channel names {} and {} are not equivalent.".format(name1, name2)
|
||||
)
|
||||
|
||||
@pytest.mark.parametrize(
|
||||
"casemapping,name1,name2",
|
||||
[
|
||||
("ascii", "#Foo", "#fooa"),
|
||||
("rfc1459", "#Foo", "#fooa"),
|
||||
],
|
||||
)
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", strict=True)
|
||||
def testChannelsNotEquivalent(self, casemapping, name1, name2):
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
if self.server_support["CASEMAPPING"] != casemapping:
|
||||
raise runner.NotImplementedByController(
|
||||
"Casemapping {} not implemented".format(casemapping)
|
||||
)
|
||||
self.joinClient(1, name1)
|
||||
self.joinClient(2, name2)
|
||||
try:
|
||||
m = self.getMessage(1)
|
||||
except client_mock.NoMessageException:
|
||||
pass
|
||||
else:
|
||||
self.assertMessageMatch(
|
||||
m, command="JOIN", nick="bar"
|
||||
) # This should always be true
|
||||
raise AssertionError(
|
||||
"Channel names {} and {} are equivalent.".format(name1, name2)
|
||||
)
|
File diff suppressed because it is too large
Load Diff
100
irctest/server_tests/invite.py
Normal file
100
irctest/server_tests/invite.py
Normal file
@ -0,0 +1,100 @@
|
||||
from irctest import cases
|
||||
from irctest.numerics import ERR_INVITEONLYCHAN, ERR_NOSUCHNICK, RPL_INVITING
|
||||
|
||||
|
||||
class InviteTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("Modern")
|
||||
def testInvites(self):
|
||||
"""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")
|
||||
self.getMessages(1)
|
||||
self.sendLine(1, "INVITE bar #chan")
|
||||
m = self.getMessage(1)
|
||||
self.assertEqual(m.command, ERR_NOSUCHNICK)
|
||||
|
||||
self.connectClient("bar")
|
||||
self.sendLine(2, "JOIN #chan")
|
||||
m = self.getMessage(2)
|
||||
self.assertEqual(m.command, ERR_INVITEONLYCHAN)
|
||||
|
||||
self.sendLine(1, "INVITE bar #chan")
|
||||
m = self.getMessage(1)
|
||||
# modern/ircv3 param order: inviter, invitee, channel
|
||||
self.assertMessageMatch(m, command=RPL_INVITING, params=["foo", "bar", "#chan"])
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(m, command="INVITE", params=["bar", "#chan"])
|
||||
self.assertTrue(m.prefix.startswith("foo")) # nickmask of inviter
|
||||
|
||||
# we were invited, so join should succeed now
|
||||
self.joinChannel(2, "#chan")
|
||||
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", deprecated=True)
|
||||
def testInviteNonExistingChannelTransmitted(self):
|
||||
"""“There is no requirement that the channel the target user is being
|
||||
invited to must exist or be a valid channel.”
|
||||
-- <https://tools.ietf.org/html/rfc1459#section-4.2.7>
|
||||
and <https://tools.ietf.org/html/rfc2812#section-3.2.7>
|
||||
|
||||
“Only the user inviting and the user being invited will receive
|
||||
notification of the invitation.”
|
||||
-- <https://tools.ietf.org/html/rfc2812#section-3.2.7>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
self.sendLine(1, "INVITE #chan bar")
|
||||
self.getMessages(1)
|
||||
messages = self.getMessages(2)
|
||||
self.assertNotEqual(
|
||||
messages,
|
||||
[],
|
||||
fail_msg="After using “INVITE #chan bar” while #chan does "
|
||||
"not exist, “bar” received nothing.",
|
||||
)
|
||||
self.assertMessageMatch(
|
||||
messages[0],
|
||||
command="INVITE",
|
||||
params=["#chan", "bar"],
|
||||
fail_msg="After “foo” invited “bar” do non-existing channel "
|
||||
"#chan, “bar” should have received “INVITE #chan bar” but "
|
||||
"got this instead: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", deprecated=True)
|
||||
def testInviteNonExistingChannelEchoed(self):
|
||||
"""“There is no requirement that the channel the target user is being
|
||||
invited to must exist or be a valid channel.”
|
||||
-- <https://tools.ietf.org/html/rfc1459#section-4.2.7>
|
||||
and <https://tools.ietf.org/html/rfc2812#section-3.2.7>
|
||||
|
||||
“Only the user inviting and the user being invited will receive
|
||||
notification of the invitation.”
|
||||
-- <https://tools.ietf.org/html/rfc2812#section-3.2.7>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
self.sendLine(1, "INVITE #chan bar")
|
||||
messages = self.getMessages(1)
|
||||
self.assertNotEqual(
|
||||
messages,
|
||||
[],
|
||||
fail_msg="After using “INVITE #chan bar” while #chan does "
|
||||
"not exist, the author received nothing.",
|
||||
)
|
||||
self.assertMessageMatch(
|
||||
messages[0],
|
||||
command="INVITE",
|
||||
params=["#chan", "bar"],
|
||||
fail_msg="After “foo” invited “bar” do non-existing channel "
|
||||
"#chan, “foo” should have received “INVITE #chan bar” but "
|
||||
"got this instead: {msg}",
|
||||
)
|
112
irctest/server_tests/join.py
Normal file
112
irctest/server_tests/join.py
Normal file
@ -0,0 +1,112 @@
|
||||
from irctest import cases
|
||||
from irctest.irc_utils import ambiguities
|
||||
|
||||
|
||||
class JoinTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", strict=True)
|
||||
def testJoinAllMessages(self):
|
||||
"""“If a JOIN is successful, the user receives a JOIN message as
|
||||
confirmation and is then sent the channel's topic (using RPL_TOPIC) and
|
||||
the list of users who are on the channel (using RPL_NAMREPLY), which
|
||||
MUST include the user joining.”
|
||||
-- <https://tools.ietf.org/html/rfc2812#section-3.2.1>
|
||||
|
||||
“If a JOIN is successful, the user is then sent the channel's topic
|
||||
(using RPL_TOPIC) and the list of users who are on the channel (using
|
||||
RPL_NAMREPLY), which must include the user joining.”
|
||||
-- <https://tools.ietf.org/html/rfc1459#section-4.2.1>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
received_commands = {m.command for m in self.getMessages(1)}
|
||||
expected_commands = {"353", "366"} # RPL_NAMREPLY # RPL_ENDOFNAMES
|
||||
self.assertTrue(
|
||||
expected_commands.issubset(received_commands),
|
||||
"Server sent {} commands, but at least {} were expected.".format(
|
||||
received_commands, expected_commands
|
||||
),
|
||||
)
|
||||
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testJoinNamreply(self):
|
||||
"""“353 RPL_NAMREPLY
|
||||
"( "=" / "*" / "@" ) <channel>
|
||||
:[ "@" / "+" ] <nick> *( " " [ "@" / "+" ] <nick> )”
|
||||
-- <https://tools.ietf.org/html/rfc2812#section-5.2>
|
||||
|
||||
This test makes a user join and check what is sent to them.
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
|
||||
for m in self.getMessages(1):
|
||||
if m.command == "353":
|
||||
self.assertIn(
|
||||
len(m.params),
|
||||
(3, 4),
|
||||
m,
|
||||
fail_msg="RPL_NAM_REPLY with number of arguments "
|
||||
"<3 or >4: {msg}",
|
||||
)
|
||||
params = ambiguities.normalize_namreply_params(m.params)
|
||||
self.assertIn(
|
||||
params[1],
|
||||
"=*@",
|
||||
m,
|
||||
fail_msg="Bad channel prefix: {item} not in {list}: {msg}",
|
||||
)
|
||||
self.assertEqual(
|
||||
params[2],
|
||||
"#chan",
|
||||
m,
|
||||
fail_msg="Bad channel name: {got} instead of " "{expects}: {msg}",
|
||||
)
|
||||
self.assertIn(
|
||||
params[3],
|
||||
{"foo", "@foo", "+foo"},
|
||||
m,
|
||||
fail_msg="Bad user list: should contain only user "
|
||||
'"foo" with an optional "+" or "@" prefix, but got: '
|
||||
"{msg}",
|
||||
)
|
||||
|
||||
def testJoinTwice(self):
|
||||
self.connectClient("foo")
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="JOIN", params=["#chan"])
|
||||
self.getMessages(1)
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
# Note that there may be no message. Both RFCs require replies only
|
||||
# if the join is successful, or has an error among the given set.
|
||||
for m in self.getMessages(1):
|
||||
if m.command == "353":
|
||||
self.assertIn(
|
||||
len(m.params),
|
||||
(3, 4),
|
||||
m,
|
||||
fail_msg="RPL_NAM_REPLY with number of arguments "
|
||||
"<3 or >4: {msg}",
|
||||
)
|
||||
params = ambiguities.normalize_namreply_params(m.params)
|
||||
self.assertIn(
|
||||
params[1],
|
||||
"=*@",
|
||||
m,
|
||||
fail_msg="Bad channel prefix: {item} not in {list}: {msg}",
|
||||
)
|
||||
self.assertEqual(
|
||||
params[2],
|
||||
"#chan",
|
||||
m,
|
||||
fail_msg="Bad channel name: {got} instead of " "{expects}: {msg}",
|
||||
)
|
||||
self.assertIn(
|
||||
params[3],
|
||||
{"foo", "@foo", "+foo"},
|
||||
m,
|
||||
fail_msg='Bad user list after user "foo" joined twice '
|
||||
"the same channel: should contain only user "
|
||||
'"foo" with an optional "+" or "@" prefix, but got: '
|
||||
"{msg}",
|
||||
)
|
178
irctest/server_tests/kick.py
Normal file
178
irctest/server_tests/kick.py
Normal file
@ -0,0 +1,178 @@
|
||||
import pytest
|
||||
|
||||
from irctest import cases, client_mock, runner
|
||||
from irctest.numerics import (
|
||||
ERR_CHANOPRIVSNEEDED,
|
||||
ERR_NOSUCHCHANNEL,
|
||||
ERR_NOTONCHANNEL,
|
||||
RPL_NAMREPLY,
|
||||
)
|
||||
from irctest.patma import ANYSTR
|
||||
|
||||
|
||||
class KickTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testKickSendsMessages(self):
|
||||
"""“Once a user has joined a channel, he receives information about
|
||||
all commands his server receives affecting the channel. This
|
||||
includes […] KICK”
|
||||
-- <https://tools.ietf.org/html/rfc1459#section-4.2.1>
|
||||
and <https://tools.ietf.org/html/rfc2812#section-3.2.1>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.joinChannel(1, "#chan")
|
||||
|
||||
self.connectClient("bar")
|
||||
self.joinChannel(2, "#chan")
|
||||
|
||||
self.connectClient("baz")
|
||||
self.joinChannel(3, "#chan")
|
||||
|
||||
# TODO: check foo is an operator
|
||||
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
self.getMessages(3)
|
||||
self.sendLine(1, "KICK #chan bar :bye")
|
||||
try:
|
||||
m = self.getMessage(1)
|
||||
if m.command == "482":
|
||||
raise runner.ImplementationChoice(
|
||||
"Channel creators are not opped by default."
|
||||
)
|
||||
self.assertMessageMatch(m, command="KICK")
|
||||
except client_mock.NoMessageException:
|
||||
# The RFCs do not say KICK must be echoed
|
||||
pass
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(m, command="KICK", params=["#chan", "bar", "bye"])
|
||||
m = self.getMessage(3)
|
||||
self.assertMessageMatch(m, command="KICK", params=["#chan", "bar", "bye"])
|
||||
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testKickPrivileges(self):
|
||||
"""Test who has the ability to kick / what error codes are sent
|
||||
for invalid kicks."""
|
||||
self.connectClient("foo")
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
self.getMessages(1)
|
||||
|
||||
self.connectClient("bar")
|
||||
self.sendLine(2, "JOIN #chan")
|
||||
|
||||
messages = self.getMessages(2)
|
||||
names = set()
|
||||
for message in messages:
|
||||
if message.command == RPL_NAMREPLY:
|
||||
names.update(set(message.params[-1].split()))
|
||||
# assert foo is opped
|
||||
self.assertIn("@foo", names, f"unexpected names: {names}")
|
||||
|
||||
self.connectClient("baz")
|
||||
|
||||
self.sendLine(3, "KICK #chan bar")
|
||||
replies = set(m.command for m in self.getMessages(3))
|
||||
self.assertTrue(
|
||||
ERR_NOTONCHANNEL in replies
|
||||
or ERR_CHANOPRIVSNEEDED in replies
|
||||
or ERR_NOSUCHCHANNEL in replies,
|
||||
f"did not receive acceptable error code for kick from outside channel: "
|
||||
f"{replies}",
|
||||
)
|
||||
|
||||
self.joinChannel(3, "#chan")
|
||||
self.getMessages(3)
|
||||
self.sendLine(3, "KICK #chan bar")
|
||||
replies = set(m.command for m in self.getMessages(3))
|
||||
# now we're a channel member so we should receive ERR_CHANOPRIVSNEEDED
|
||||
self.assertIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
self.sendLine(1, "MODE #chan +o baz")
|
||||
self.getMessages(1)
|
||||
# should be able to kick an unprivileged user:
|
||||
self.sendLine(3, "KICK #chan bar")
|
||||
# should be able to kick an operator:
|
||||
self.sendLine(3, "KICK #chan foo")
|
||||
baz_replies = set(m.command for m in self.getMessages(3))
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, baz_replies)
|
||||
kick_targets = [m.params[1] for m in self.getMessages(1) if m.command == "KICK"]
|
||||
# foo should see bar and foo being kicked
|
||||
self.assertTrue(
|
||||
any(target.startswith("foo") for target in kick_targets),
|
||||
f"unexpected kick targets: {kick_targets}",
|
||||
)
|
||||
self.assertTrue(
|
||||
any(target.startswith("bar") for target in kick_targets),
|
||||
f"unexpected kick targets: {kick_targets}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testKickNonexistentChannel(self):
|
||||
"""“Kick command [...] Numeric replies: [...] ERR_NOSUCHCHANNEL."""
|
||||
self.connectClient("foo")
|
||||
self.sendLine(1, "KICK #chan nick")
|
||||
m = self.getMessage(1)
|
||||
# should return ERR_NOSUCHCHANNEL
|
||||
self.assertMessageMatch(m, command="403")
|
||||
|
||||
@pytest.mark.parametrize("multiple_targets", [True, False])
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testDoubleKickMessages(self, multiple_targets):
|
||||
"""“The server MUST NOT send KICK messages with multiple channels or
|
||||
users to clients. This is necessarily to maintain backward
|
||||
compatibility with old client software.”
|
||||
-- https://tools.ietf.org/html/rfc2812#section-3.2.8
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.joinChannel(1, "#chan")
|
||||
|
||||
self.connectClient("bar")
|
||||
self.joinChannel(2, "#chan")
|
||||
|
||||
self.connectClient("baz")
|
||||
self.joinChannel(3, "#chan")
|
||||
|
||||
self.connectClient("qux")
|
||||
self.joinChannel(4, "#chan")
|
||||
|
||||
# TODO: check foo is an operator
|
||||
|
||||
# Synchronize
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
self.getMessages(3)
|
||||
self.getMessages(4)
|
||||
|
||||
if multiple_targets:
|
||||
self.sendLine(1, "KICK #chan,#chan bar,baz :bye")
|
||||
else:
|
||||
self.sendLine(1, "KICK #chan bar,baz :bye")
|
||||
try:
|
||||
m = self.getMessage(1)
|
||||
if m.command == "482":
|
||||
raise runner.OptionalExtensionNotSupported(
|
||||
"Channel creators are not opped by default."
|
||||
)
|
||||
if m.command in {"401", "403"}:
|
||||
raise runner.NotImplementedByController("Multi-target KICK")
|
||||
except client_mock.NoMessageException:
|
||||
# The RFCs do not say KICK must be echoed
|
||||
pass
|
||||
|
||||
mgroup = self.getMessages(4)
|
||||
self.assertGreaterEqual(len(mgroup), 2)
|
||||
m1, m2 = mgroup[:2]
|
||||
|
||||
self.assertMessageMatch(m1, command="KICK", params=["#chan", ANYSTR, "bye"])
|
||||
self.assertMessageMatch(m2, command="KICK", params=["#chan", ANYSTR, "bye"])
|
||||
|
||||
if (m1.params[1] == "bar" and m2.params[1] == "baz") or (
|
||||
m1.params[1] == "baz" and m2.params[1] == "bar"
|
||||
):
|
||||
... # success
|
||||
else:
|
||||
raise AssertionError(
|
||||
"Middle params [{}, {}] are not correct.".format(
|
||||
m1.params[1], m2.params[1]
|
||||
)
|
||||
)
|
69
irctest/server_tests/list.py
Normal file
69
irctest/server_tests/list.py
Normal file
@ -0,0 +1,69 @@
|
||||
from irctest import cases
|
||||
|
||||
|
||||
class ListTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testListEmpty(self):
|
||||
"""<https://tools.ietf.org/html/rfc1459#section-4.2.6>
|
||||
<https://tools.ietf.org/html/rfc2812#section-3.2.6>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.getMessages(1)
|
||||
self.sendLine(2, "LIST")
|
||||
m = self.getMessage(2)
|
||||
if m.command == "321":
|
||||
# skip RPL_LISTSTART
|
||||
m = self.getMessage(2)
|
||||
self.assertNotEqual(
|
||||
m.command,
|
||||
"322", # RPL_LIST
|
||||
"LIST response gives (at least) one channel, whereas there " "is none.",
|
||||
)
|
||||
self.assertMessageMatch(
|
||||
m,
|
||||
command="323", # RPL_LISTEND
|
||||
fail_msg="Second reply to LIST is not 322 (RPL_LIST) "
|
||||
"or 323 (RPL_LISTEND), or but: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testListOne(self):
|
||||
"""When a channel exists, LIST should get it in a reply.
|
||||
<https://tools.ietf.org/html/rfc1459#section-4.2.6>
|
||||
<https://tools.ietf.org/html/rfc2812#section-3.2.6>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
self.getMessages(1)
|
||||
self.sendLine(2, "LIST")
|
||||
m = self.getMessage(2)
|
||||
if m.command == "321":
|
||||
# skip RPL_LISTSTART
|
||||
m = self.getMessage(2)
|
||||
self.assertNotEqual(
|
||||
m.command,
|
||||
"323", # RPL_LISTEND
|
||||
fail_msg="LIST response ended (ie. 323, aka RPL_LISTEND) "
|
||||
"without listing any channel, whereas there is one.",
|
||||
)
|
||||
self.assertMessageMatch(
|
||||
m,
|
||||
command="322", # RPL_LIST
|
||||
fail_msg="Second reply to LIST is not 322 (RPL_LIST), "
|
||||
"nor 323 (RPL_LISTEND) but: {msg}",
|
||||
)
|
||||
m = self.getMessage(2)
|
||||
self.assertNotEqual(
|
||||
m.command,
|
||||
"322", # RPL_LIST
|
||||
fail_msg="LIST response gives (at least) two channels, "
|
||||
"whereas there is only one.",
|
||||
)
|
||||
self.assertMessageMatch(
|
||||
m,
|
||||
command="323", # RPL_LISTEND
|
||||
fail_msg="Third reply to LIST is not 322 (RPL_LIST) "
|
||||
"or 323 (RPL_LISTEND), or but: {msg}",
|
||||
)
|
699
irctest/server_tests/mode.py
Normal file
699
irctest/server_tests/mode.py
Normal file
@ -0,0 +1,699 @@
|
||||
import math
|
||||
import time
|
||||
|
||||
from irctest import cases, runner
|
||||
from irctest.irc_utils.junkdrawer import ircv3_timestamp_to_unixtime
|
||||
from irctest.numerics import (
|
||||
ERR_BADCHANNELKEY,
|
||||
ERR_BANNEDFROMCHAN,
|
||||
ERR_CANNOTSENDTOCHAN,
|
||||
ERR_CHANOPRIVSNEEDED,
|
||||
ERR_INVALIDKEY,
|
||||
ERR_INVALIDMODEPARAM,
|
||||
ERR_UNKNOWNERROR,
|
||||
RPL_NAMREPLY,
|
||||
)
|
||||
from irctest.patma import ANYLIST, StrRe
|
||||
|
||||
MODERN_CAPS = [
|
||||
"server-time",
|
||||
"message-tags",
|
||||
"batch",
|
||||
"labeled-response",
|
||||
"echo-message",
|
||||
"account-tag",
|
||||
]
|
||||
|
||||
|
||||
class KeyTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testKeyNormal(self):
|
||||
self.connectClient("bar")
|
||||
self.joinChannel(1, "#chan")
|
||||
self.sendLine(1, "MODE #chan +k beer")
|
||||
self.getMessages(1)
|
||||
|
||||
self.connectClient("qux")
|
||||
self.getMessages(2)
|
||||
self.sendLine(2, "JOIN #chan")
|
||||
reply = self.getMessages(2)
|
||||
self.assertNotIn("JOIN", {msg.command for msg in reply})
|
||||
self.assertIn(ERR_BADCHANNELKEY, {msg.command for msg in reply})
|
||||
|
||||
self.sendLine(2, "JOIN #chan beer")
|
||||
reply = self.getMessages(2)
|
||||
self.assertMessageMatch(reply[0], command="JOIN", params=["#chan"])
|
||||
|
||||
@cases.mark_specifications("RFC2812", "Modern")
|
||||
def testKeyValidation(self):
|
||||
"""
|
||||
key = 1*23( %x01-05 / %x07-08 / %x0C / %x0E-1F / %x21-7F )
|
||||
; any 7-bit US_ASCII character,
|
||||
; except NUL, CR, LF, FF, h/v TABs, and " "
|
||||
-- https://tools.ietf.org/html/rfc2812#page-8
|
||||
|
||||
"Servers may validate the value (eg. to forbid spaces, as they make it harder
|
||||
to use the key in `JOIN` messages). If the value is invalid, they SHOULD
|
||||
return [`ERR_INVALIDMODEPARAM`](#errinvalidmodeparam-696).
|
||||
However, clients MUST be able to handle any of the following:
|
||||
|
||||
* [`ERR_INVALIDMODEPARAM`](#errinvalidmodeparam-696)
|
||||
* [`ERR_INVALIDKEY`](#errinvalidkey-525)
|
||||
* `MODE` echoed with a different key (eg. truncated or stripped of invalid
|
||||
characters)
|
||||
* the key changed ignored, and no `MODE` echoed if no other mode change
|
||||
was valid.
|
||||
"
|
||||
-- https://modern.ircdocs.horse/#key-channel-mode
|
||||
-- https://github.com/ircdocs/modern-irc/pull/111
|
||||
"""
|
||||
self.connectClient("bar")
|
||||
self.joinChannel(1, "#chan")
|
||||
self.sendLine(1, "MODE #chan +k :passphrase with spaces")
|
||||
|
||||
# The spec requires no space; but doesn't say what to do
|
||||
# if there is one.
|
||||
# Let's check the various alternatives
|
||||
|
||||
replies = self.getMessages(1)
|
||||
self.assertNotIn(
|
||||
ERR_UNKNOWNERROR,
|
||||
{msg.command for msg in replies},
|
||||
fail_msg="Sending an invalid key (with a space) caused an "
|
||||
"ERR_UNKNOWNERROR instead of being handled explicitly "
|
||||
"(eg. ERR_INVALIDMODEPARAM or truncation): {msg}",
|
||||
)
|
||||
|
||||
if {ERR_INVALIDMODEPARAM, ERR_INVALIDKEY} & {msg.command for msg in replies}:
|
||||
# First option: ERR_INVALIDMODEPARAM (eg. Ergo) or ERR_INVALIDKEY
|
||||
# (eg. ircu2)
|
||||
return
|
||||
|
||||
if not replies:
|
||||
# MODE was ignored entirely
|
||||
self.connectClient("foo")
|
||||
self.sendLine(2, "JOIN #chan")
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(2), command="JOIN", params=["#chan"]
|
||||
)
|
||||
return
|
||||
|
||||
# Second and third options: truncating the key (eg. UnrealIRCd)
|
||||
# or replacing spaces (eg. Charybdis)
|
||||
mode_commands = [msg for msg in replies if msg.command == "MODE"]
|
||||
self.assertGreaterEqual(
|
||||
len(mode_commands),
|
||||
1,
|
||||
fail_msg="Sending an invalid key (with a space) triggered "
|
||||
"neither ERR_UNKNOWNERROR, ERR_INVALIDMODEPARAM, ERR_INVALIDKEY, "
|
||||
" or a MODE. Only these: {}",
|
||||
extra_format=(replies,),
|
||||
)
|
||||
self.assertLessEqual(
|
||||
len(mode_commands),
|
||||
1,
|
||||
fail_msg="Sending an invalid key (with a space) triggered "
|
||||
"multiple MODE responses: {}",
|
||||
extra_format=(replies,),
|
||||
)
|
||||
|
||||
mode_command = mode_commands[0]
|
||||
if mode_command.params == ["#chan", "+k", "passphrase"]:
|
||||
key = "passphrase"
|
||||
elif mode_command.params == ["#chan", "+k", "passphrasewithspaces"]:
|
||||
key = "passphrasewithspaces"
|
||||
elif mode_command.params == ["#chan", "+k", "passphrase with spaces"]:
|
||||
raise self.failureException("Invalid key (with a space) was not rejected.")
|
||||
|
||||
self.connectClient("foo")
|
||||
self.sendLine(2, f"JOIN #chan {key}")
|
||||
self.assertMessageMatch(self.getMessage(2), command="JOIN", params=["#chan"])
|
||||
|
||||
|
||||
class AuditoriumTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("Ergo")
|
||||
def testAuditorium(self):
|
||||
self.connectClient("bar", name="bar", capabilities=MODERN_CAPS)
|
||||
self.joinChannel("bar", "#auditorium")
|
||||
self.getMessages("bar")
|
||||
self.sendLine("bar", "MODE #auditorium +u")
|
||||
modelines = [msg for msg in self.getMessages("bar") if msg.command == "MODE"]
|
||||
self.assertEqual(len(modelines), 1)
|
||||
self.assertMessageMatch(modelines[0], params=["#auditorium", "+u"])
|
||||
|
||||
self.connectClient("guest1", name="guest1", capabilities=MODERN_CAPS)
|
||||
self.joinChannel("guest1", "#auditorium")
|
||||
self.getMessages("guest1")
|
||||
# chanop should get a JOIN message
|
||||
join_msgs = [msg for msg in self.getMessages("bar") if msg.command == "JOIN"]
|
||||
self.assertEqual(len(join_msgs), 1)
|
||||
self.assertMessageMatch(join_msgs[0], nick="guest1", params=["#auditorium"])
|
||||
|
||||
self.connectClient("guest2", name="guest2", capabilities=MODERN_CAPS)
|
||||
self.joinChannel("guest2", "#auditorium")
|
||||
self.getMessages("guest2")
|
||||
# chanop should get a JOIN message
|
||||
join_msgs = [msg for msg in self.getMessages("bar") if msg.command == "JOIN"]
|
||||
self.assertEqual(len(join_msgs), 1)
|
||||
join_msg = join_msgs[0]
|
||||
self.assertMessageMatch(join_msg, nick="guest2", params=["#auditorium"])
|
||||
# oragono/oragono#1642 ; msgid should be populated,
|
||||
# and the time tag should be sane
|
||||
self.assertTrue(join_msg.tags.get("msgid"))
|
||||
self.assertLessEqual(
|
||||
math.fabs(time.time() - ircv3_timestamp_to_unixtime(join_msg.tags["time"])),
|
||||
60.0,
|
||||
)
|
||||
# fellow unvoiced participant should not
|
||||
unvoiced_join_msgs = [
|
||||
msg for msg in self.getMessages("guest1") if msg.command == "JOIN"
|
||||
]
|
||||
self.assertEqual(len(unvoiced_join_msgs), 0)
|
||||
|
||||
self.connectClient("guest3", name="guest3", capabilities=MODERN_CAPS)
|
||||
self.joinChannel("guest3", "#auditorium")
|
||||
self.getMessages("guest3")
|
||||
|
||||
self.sendLine("bar", "PRIVMSG #auditorium hi")
|
||||
echo_message = [
|
||||
msg for msg in self.getMessages("bar") if msg.command == "PRIVMSG"
|
||||
][0]
|
||||
self.assertEqual(echo_message, self.getMessages("guest1")[0])
|
||||
self.assertEqual(echo_message, self.getMessages("guest2")[0])
|
||||
self.assertEqual(echo_message, self.getMessages("guest3")[0])
|
||||
|
||||
# unvoiced users can speak
|
||||
self.sendLine("guest1", "PRIVMSG #auditorium :hi you")
|
||||
echo_message = [
|
||||
msg for msg in self.getMessages("guest1") if msg.command == "PRIVMSG"
|
||||
][0]
|
||||
self.assertEqual(self.getMessages("bar"), [echo_message])
|
||||
self.assertEqual(self.getMessages("guest2"), [echo_message])
|
||||
self.assertEqual(self.getMessages("guest3"), [echo_message])
|
||||
|
||||
def names(client):
|
||||
self.sendLine(client, "NAMES #auditorium")
|
||||
result = set()
|
||||
for msg in self.getMessages(client):
|
||||
if msg.command == RPL_NAMREPLY:
|
||||
result.update(msg.params[-1].split())
|
||||
return result
|
||||
|
||||
self.assertEqual(names("bar"), {"@bar", "guest1", "guest2", "guest3"})
|
||||
self.assertEqual(names("guest1"), {"@bar"})
|
||||
self.assertEqual(names("guest2"), {"@bar"})
|
||||
self.assertEqual(names("guest3"), {"@bar"})
|
||||
|
||||
self.sendLine("bar", "MODE #auditorium +v guest1")
|
||||
modeLine = [msg for msg in self.getMessages("bar") if msg.command == "MODE"][0]
|
||||
self.assertEqual(self.getMessages("guest1"), [modeLine])
|
||||
self.assertEqual(self.getMessages("guest2"), [modeLine])
|
||||
self.assertEqual(self.getMessages("guest3"), [modeLine])
|
||||
self.assertEqual(names("bar"), {"@bar", "+guest1", "guest2", "guest3"})
|
||||
self.assertEqual(names("guest2"), {"@bar", "+guest1"})
|
||||
self.assertEqual(names("guest3"), {"@bar", "+guest1"})
|
||||
|
||||
self.sendLine("guest1", "PART #auditorium")
|
||||
part = [msg for msg in self.getMessages("guest1") if msg.command == "PART"][0]
|
||||
# everyone should see voiced PART
|
||||
self.assertEqual(self.getMessages("bar")[0], part)
|
||||
self.assertEqual(self.getMessages("guest2")[0], part)
|
||||
self.assertEqual(self.getMessages("guest3")[0], part)
|
||||
|
||||
self.joinChannel("guest1", "#auditorium")
|
||||
self.getMessages("guest1")
|
||||
self.getMessages("bar")
|
||||
|
||||
self.sendLine("guest2", "PART #auditorium")
|
||||
part = [msg for msg in self.getMessages("guest2") if msg.command == "PART"][0]
|
||||
self.assertEqual(self.getMessages("bar"), [part])
|
||||
# part should be hidden from unvoiced participants
|
||||
self.assertEqual(self.getMessages("guest1"), [])
|
||||
self.assertEqual(self.getMessages("guest3"), [])
|
||||
|
||||
self.sendLine("guest3", "QUIT")
|
||||
self.assertDisconnected("guest3")
|
||||
# quit should be hidden from unvoiced participants
|
||||
self.assertEqual(
|
||||
len([msg for msg in self.getMessages("bar") if msg.command == "QUIT"]), 1
|
||||
)
|
||||
self.assertEqual(
|
||||
len([msg for msg in self.getMessages("guest1") if msg.command == "QUIT"]), 0
|
||||
)
|
||||
|
||||
|
||||
class BanMode(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testBan(self):
|
||||
"""Basic ban operation"""
|
||||
self.connectClient("chanop", name="chanop")
|
||||
self.joinChannel("chanop", "#chan")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", "MODE #chan +b bar!*@*")
|
||||
self.assertMessageMatch(self.getMessage("chanop"), command="MODE")
|
||||
|
||||
self.connectClient(
|
||||
"Bar", name="bar", capabilities=["echo-message"], skip_if_cap_nak=True
|
||||
)
|
||||
self.getMessages("bar")
|
||||
self.sendLine("bar", "JOIN #chan")
|
||||
self.assertMessageMatch(self.getMessage("bar"), command=ERR_BANNEDFROMCHAN)
|
||||
|
||||
self.sendLine("chanop", "MODE #chan -b bar!*@*")
|
||||
self.assertMessageMatch(self.getMessage("chanop"), command="MODE")
|
||||
|
||||
self.sendLine("bar", "JOIN #chan")
|
||||
self.assertMessageMatch(self.getMessage("bar"), command="JOIN")
|
||||
|
||||
@cases.mark_specifications("Ergo")
|
||||
def testCaseInsensitive(self):
|
||||
"""Some clients allow unsetting modes if their argument matches
|
||||
up to normalization"""
|
||||
self.connectClient("chanop", name="chanop")
|
||||
self.joinChannel("chanop", "#chan")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", "MODE #chan +b BAR!*@*")
|
||||
self.assertMessageMatch(self.getMessage("chanop"), command="MODE")
|
||||
|
||||
self.connectClient("Bar", name="bar", capabilities=["echo-message"])
|
||||
self.getMessages("bar")
|
||||
self.sendLine("bar", "JOIN #chan")
|
||||
self.assertMessageMatch(self.getMessage("bar"), command=ERR_BANNEDFROMCHAN)
|
||||
|
||||
self.sendLine("chanop", "MODE #chan -b bar!*@*")
|
||||
self.assertMessageMatch(self.getMessage("chanop"), command="MODE")
|
||||
|
||||
self.sendLine("bar", "JOIN #chan")
|
||||
self.assertMessageMatch(self.getMessage("bar"), command="JOIN")
|
||||
|
||||
|
||||
class ModeratedMode(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testModeratedMode(self):
|
||||
# test the +m channel mode
|
||||
self.connectClient("chanop", name="chanop")
|
||||
self.joinChannel("chanop", "#chan")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", "MODE #chan +m")
|
||||
replies = self.getMessages("chanop")
|
||||
modeLines = [line for line in replies if line.command == "MODE"]
|
||||
self.assertMessageMatch(modeLines[0], command="MODE", params=["#chan", "+m"])
|
||||
|
||||
self.connectClient("baz", name="baz")
|
||||
self.joinChannel("baz", "#chan")
|
||||
self.getMessages("chanop")
|
||||
# this message should be suppressed completely by +m
|
||||
self.sendLine("baz", "PRIVMSG #chan :hi from baz")
|
||||
replies = self.getMessages("baz")
|
||||
reply_cmds = {reply.command for reply in replies}
|
||||
self.assertIn(ERR_CANNOTSENDTOCHAN, reply_cmds)
|
||||
self.assertEqual(self.getMessages("chanop"), [])
|
||||
|
||||
# grant +v, user should be able to send messages
|
||||
self.sendLine("chanop", "MODE #chan +v baz")
|
||||
self.getMessages("chanop")
|
||||
self.getMessages("baz")
|
||||
self.sendLine("baz", "PRIVMSG #chan :hi again from baz")
|
||||
self.getMessages("baz")
|
||||
relays = self.getMessages("chanop")
|
||||
relay = relays[0]
|
||||
self.assertMessageMatch(
|
||||
relay, command="PRIVMSG", params=["#chan", "hi again from baz"]
|
||||
)
|
||||
|
||||
|
||||
@cases.mark_services
|
||||
class RegisteredOnlySpeakMode(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("Ergo")
|
||||
def testRegisteredOnlySpeakMode(self):
|
||||
self.controller.registerUser(self, "evan", "sesame")
|
||||
|
||||
# test the +M (only registered users and ops can speak) channel mode
|
||||
self.connectClient("chanop", name="chanop")
|
||||
self.joinChannel("chanop", "#chan")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", "MODE #chan +M")
|
||||
replies = self.getMessages("chanop")
|
||||
modeLines = [line for line in replies if line.command == "MODE"]
|
||||
self.assertMessageMatch(modeLines[0], command="MODE", params=["#chan", "+M"])
|
||||
|
||||
self.connectClient("baz", name="baz")
|
||||
self.joinChannel("baz", "#chan")
|
||||
self.getMessages("chanop")
|
||||
# this message should be suppressed completely by +M
|
||||
self.sendLine("baz", "PRIVMSG #chan :hi from baz")
|
||||
replies = self.getMessages("baz")
|
||||
reply_cmds = {reply.command for reply in replies}
|
||||
self.assertIn(ERR_CANNOTSENDTOCHAN, reply_cmds)
|
||||
self.assertEqual(self.getMessages("chanop"), [])
|
||||
|
||||
# +v exempts users from the registration requirement:
|
||||
self.sendLine("chanop", "MODE #chan +v baz")
|
||||
self.getMessages("chanop")
|
||||
self.getMessages("baz")
|
||||
self.sendLine("baz", "PRIVMSG #chan :hi again from baz")
|
||||
replies = self.getMessages("baz")
|
||||
# baz should not receive an error (or an echo)
|
||||
self.assertEqual(replies, [])
|
||||
replies = self.getMessages("chanop")
|
||||
self.assertMessageMatch(
|
||||
replies[0], command="PRIVMSG", params=["#chan", "hi again from baz"]
|
||||
)
|
||||
|
||||
self.connectClient(
|
||||
"evan",
|
||||
name="evan",
|
||||
account="evan",
|
||||
password="sesame",
|
||||
capabilities=["sasl"],
|
||||
)
|
||||
self.joinChannel("evan", "#chan")
|
||||
self.getMessages("baz")
|
||||
self.sendLine("evan", "PRIVMSG #chan :hi from evan")
|
||||
replies = self.getMessages("evan")
|
||||
# evan should not receive an error (or an echo)
|
||||
self.assertEqual(replies, [])
|
||||
replies = self.getMessages("baz")
|
||||
self.assertMessageMatch(
|
||||
replies[0], command="PRIVMSG", params=["#chan", "hi from evan"]
|
||||
)
|
||||
|
||||
|
||||
class OpModerated(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("Ergo")
|
||||
def testOpModerated(self):
|
||||
# test the +U channel mode
|
||||
self.connectClient("chanop", name="chanop", capabilities=MODERN_CAPS)
|
||||
self.joinChannel("chanop", "#chan")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", "MODE #chan +U")
|
||||
replies = {msg.command for msg in self.getMessages("chanop")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
self.connectClient("baz", name="baz", capabilities=MODERN_CAPS)
|
||||
self.joinChannel("baz", "#chan")
|
||||
self.sendLine("baz", "PRIVMSG #chan :hi from baz")
|
||||
echo = self.getMessages("baz")[0]
|
||||
self.assertMessageMatch(
|
||||
echo, command="PRIVMSG", params=["#chan", "hi from baz"]
|
||||
)
|
||||
self.assertEqual(
|
||||
[msg for msg in self.getMessages("chanop") if msg.command == "PRIVMSG"],
|
||||
[echo],
|
||||
)
|
||||
|
||||
self.connectClient("qux", name="qux", capabilities=MODERN_CAPS)
|
||||
self.joinChannel("qux", "#chan")
|
||||
self.sendLine("qux", "PRIVMSG #chan :hi from qux")
|
||||
echo = self.getMessages("qux")[0]
|
||||
self.assertMessageMatch(
|
||||
echo, command="PRIVMSG", params=["#chan", "hi from qux"]
|
||||
)
|
||||
# message is relayed to chanop but not to unprivileged
|
||||
self.assertEqual(
|
||||
[msg for msg in self.getMessages("chanop") if msg.command == "PRIVMSG"],
|
||||
[echo],
|
||||
)
|
||||
self.assertEqual(
|
||||
[msg for msg in self.getMessages("baz") if msg.command == "PRIVMSG"], []
|
||||
)
|
||||
|
||||
self.sendLine("chanop", "MODE #chan +v qux")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("qux", "PRIVMSG #chan :hi again from qux")
|
||||
echo = [msg for msg in self.getMessages("qux") if msg.command == "PRIVMSG"][0]
|
||||
self.assertMessageMatch(
|
||||
echo, command="PRIVMSG", params=["#chan", "hi again from qux"]
|
||||
)
|
||||
self.assertEqual(
|
||||
[msg for msg in self.getMessages("chanop") if msg.command == "PRIVMSG"],
|
||||
[echo],
|
||||
)
|
||||
self.assertEqual(
|
||||
[msg for msg in self.getMessages("baz") if msg.command == "PRIVMSG"], [echo]
|
||||
)
|
||||
|
||||
|
||||
class MuteExtban(cases.BaseServerTestCase):
|
||||
"""https://defs.ircdocs.horse/defs/isupport.html#extban
|
||||
|
||||
It magically guesses what char the IRCd uses for mutes."""
|
||||
|
||||
def char(self):
|
||||
if self.controller.extban_mute_char is None:
|
||||
raise runner.ExtbanNotSupported("", "mute")
|
||||
else:
|
||||
return self.controller.extban_mute_char
|
||||
|
||||
@cases.mark_specifications("Ergo")
|
||||
def testISupport(self):
|
||||
self.connectClient(1) # Fetches ISUPPORT
|
||||
isupport = self.server_support
|
||||
token = isupport["EXTBAN"]
|
||||
prefix, comma, types = token.partition(",")
|
||||
self.assertIn(self.char(), types, f"Missing '{self.char()}' in ISUPPORT EXTBAN")
|
||||
self.assertEqual(prefix, "")
|
||||
self.assertEqual(comma, ",")
|
||||
|
||||
@cases.mark_specifications("ircdocs")
|
||||
def testMuteExtban(self):
|
||||
"""Basic usage of mute"""
|
||||
self.connectClient("chanop", name="chanop")
|
||||
|
||||
isupport = self.server_support
|
||||
token = isupport.get("EXTBAN", "")
|
||||
prefix, comma, types = token.partition(",")
|
||||
if self.char() not in types:
|
||||
raise runner.ExtbanNotSupported(self.char(), "mute")
|
||||
|
||||
clients = ("chanop", "bar")
|
||||
|
||||
# Mute "bar"
|
||||
self.joinChannel("chanop", "#chan")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", f"MODE #chan +b {prefix}{self.char()}:bar!*@*")
|
||||
replies = {msg.command for msg in self.getMessages("chanop")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
self.connectClient("bar", name="bar", capabilities=["echo-message"])
|
||||
self.joinChannel("bar", "#chan")
|
||||
|
||||
for client in clients:
|
||||
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!*@*",
|
||||
StrRe("chanop(!.*)?"),
|
||||
*ANYLIST,
|
||||
],
|
||||
)
|
||||
self.getMessages("bar")
|
||||
|
||||
# "bar" talks: rejected
|
||||
self.sendLine("bar", "PRIVMSG #chan :hi from bar")
|
||||
replies = self.getMessages("bar")
|
||||
replies_cmds = {msg.command for msg in replies}
|
||||
self.assertNotIn("PRIVMSG", replies_cmds)
|
||||
self.assertIn(ERR_CANNOTSENDTOCHAN, replies_cmds)
|
||||
self.assertEqual(self.getMessages("chanop"), [])
|
||||
|
||||
# remove mute on "bar" with -b
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", f"MODE #chan -b {prefix}{self.char()}:bar!*@*")
|
||||
replies = {msg.command for msg in self.getMessages("chanop")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
# "bar" can now talk
|
||||
self.sendLine("bar", "PRIVMSG #chan :hi again from bar")
|
||||
replies = self.getMessages("bar")
|
||||
replies_cmds = {msg.command for msg in replies}
|
||||
self.assertIn("PRIVMSG", replies_cmds)
|
||||
self.assertNotIn(ERR_CANNOTSENDTOCHAN, replies_cmds)
|
||||
self.assertEqual(
|
||||
self.getMessages("chanop"),
|
||||
[msg for msg in replies if msg.command == "PRIVMSG"],
|
||||
)
|
||||
|
||||
@cases.mark_specifications("ircdocs")
|
||||
def testMuteExtbanVoiced(self):
|
||||
"""Checks +v overrides the mute"""
|
||||
self.connectClient("chanop", name="chanop")
|
||||
|
||||
isupport = self.server_support
|
||||
token = isupport.get("EXTBAN", "")
|
||||
prefix, comma, types = token.partition(",")
|
||||
if self.char() not in types:
|
||||
raise runner.ExtbanNotSupported(self.char(), "mute")
|
||||
|
||||
clients = ("chanop", "qux")
|
||||
|
||||
# Mute "qux"
|
||||
self.joinChannel("chanop", "#chan")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", f"MODE #chan +b {prefix}{self.char()}:qux!*@*")
|
||||
replies = {msg.command for msg in self.getMessages("chanop")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
self.connectClient(
|
||||
"qux", name="qux", ident="evan", capabilities=["echo-message"]
|
||||
)
|
||||
self.joinChannel("qux", "#chan")
|
||||
|
||||
for client in clients:
|
||||
self.getMessages(client)
|
||||
|
||||
# "qux" talks: rejected
|
||||
self.sendLine("qux", "PRIVMSG #chan :hi from qux")
|
||||
replies = self.getMessages("qux")
|
||||
replies_cmds = {msg.command for msg in replies}
|
||||
self.assertNotIn("PRIVMSG", replies_cmds)
|
||||
self.assertIn(ERR_CANNOTSENDTOCHAN, replies_cmds)
|
||||
self.assertEqual(self.getMessages("chanop"), [])
|
||||
|
||||
for client in clients:
|
||||
self.getMessages(client)
|
||||
|
||||
# +v grants an exemption to +b
|
||||
self.sendLine("chanop", "MODE #chan +v qux")
|
||||
replies = {msg.command for msg in self.getMessages("chanop")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
# so "qux" can now talk
|
||||
self.sendLine("qux", "PRIVMSG #chan :hi again from qux")
|
||||
replies = self.getMessages("qux")
|
||||
replies_cmds = {msg.command for msg in replies}
|
||||
self.assertIn("PRIVMSG", replies_cmds)
|
||||
self.assertNotIn(ERR_CANNOTSENDTOCHAN, replies_cmds)
|
||||
self.assertEqual(
|
||||
self.getMessages("chanop"),
|
||||
[msg for msg in replies if msg.command == "PRIVMSG"],
|
||||
)
|
||||
|
||||
@cases.mark_specifications("ircdocs")
|
||||
def testMuteExtbanExempt(self):
|
||||
"""Checks +e overrides the mute
|
||||
|
||||
<https://defs.ircdocs.horse/defs/chanmodes.html#e-ban-exception>"""
|
||||
self.connectClient("chanop", name="chanop")
|
||||
|
||||
isupport = self.server_support
|
||||
token = isupport.get("EXTBAN", "")
|
||||
prefix, comma, types = token.partition(",")
|
||||
if self.char() not in types:
|
||||
raise runner.ExtbanNotSupported(self.char(), "mute")
|
||||
if "e" not in self.server_support["CHANMODES"]:
|
||||
raise runner.ChannelModeNotSupported(self.char(), "mute")
|
||||
|
||||
clients = ("chanop", "qux")
|
||||
|
||||
# Mute "qux"
|
||||
self.joinChannel("chanop", "#chan")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", f"MODE #chan +b {prefix}{self.char()}:qux!*@*")
|
||||
replies = {msg.command for msg in self.getMessages("chanop")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
self.connectClient(
|
||||
"qux", name="qux", ident="evan", capabilities=["echo-message"]
|
||||
)
|
||||
self.joinChannel("qux", "#chan")
|
||||
|
||||
for client in clients:
|
||||
self.getMessages(client)
|
||||
|
||||
# "qux" talks: rejected
|
||||
self.sendLine("qux", "PRIVMSG #chan :hi from qux")
|
||||
replies = self.getMessages("qux")
|
||||
replies_cmds = {msg.command for msg in replies}
|
||||
self.assertNotIn("PRIVMSG", replies_cmds)
|
||||
self.assertIn(ERR_CANNOTSENDTOCHAN, replies_cmds)
|
||||
self.assertEqual(self.getMessages("chanop"), [])
|
||||
|
||||
for client in clients:
|
||||
self.getMessages(client)
|
||||
|
||||
# +e grants an exemption to +b
|
||||
self.sendLine("chanop", f"MODE #chan +e {prefix}{self.char()}:*!~evan@*")
|
||||
replies = {msg.command for msg in self.getMessages("chanop")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
self.getMessages("qux")
|
||||
|
||||
# so "qux" can now talk
|
||||
self.sendLine("qux", "PRIVMSG #chan :thanks for mute-excepting me")
|
||||
replies = self.getMessages("qux")
|
||||
replies_cmds = {msg.command for msg in replies}
|
||||
self.assertIn("PRIVMSG", replies_cmds)
|
||||
self.assertNotIn(ERR_CANNOTSENDTOCHAN, replies_cmds)
|
||||
self.assertEqual(
|
||||
self.getMessages("chanop"),
|
||||
[msg for msg in replies if msg.command == "PRIVMSG"],
|
||||
)
|
||||
|
||||
@cases.mark_specifications("Ergo")
|
||||
def testCapitalization(self):
|
||||
"""
|
||||
Regression test for oragono #1370: mutes not correctly enforced against
|
||||
users with capital letters in their NUH
|
||||
|
||||
For consistency with regular -b, which allows unsetting up to
|
||||
normalization
|
||||
"""
|
||||
clients = ("chanop", "bar")
|
||||
|
||||
self.connectClient("chanop", name="chanop")
|
||||
|
||||
isupport = self.server_support
|
||||
token = isupport.get("EXTBAN", "")
|
||||
prefix, comma, types = token.partition(",")
|
||||
|
||||
self.joinChannel("chanop", "#chan")
|
||||
self.getMessages("chanop")
|
||||
self.sendLine("chanop", f"MODE #chan +b {prefix}{self.char()}:BAR!*@*")
|
||||
replies = {msg.command for msg in self.getMessages("chanop")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
self.connectClient("Bar", name="bar", capabilities=["echo-message"])
|
||||
self.joinChannel("bar", "#chan")
|
||||
|
||||
for client in clients:
|
||||
self.getMessages(client)
|
||||
|
||||
self.sendLine("bar", "PRIVMSG #chan :hi from bar")
|
||||
replies = self.getMessages("bar")
|
||||
replies_cmds = {msg.command for msg in replies}
|
||||
self.assertNotIn("PRIVMSG", replies_cmds)
|
||||
self.assertIn(ERR_CANNOTSENDTOCHAN, replies_cmds)
|
||||
self.assertEqual(self.getMessages("chanop"), [])
|
||||
|
||||
# remove mute with -b
|
||||
self.sendLine("chanop", f"MODE #chan -b {prefix}{self.char()}:bar!*@*")
|
||||
replies = {msg.command for msg in self.getMessages("chanop")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
# "bar" can talk again
|
||||
self.sendLine("bar", "PRIVMSG #chan :hi again from bar")
|
||||
replies = self.getMessages("bar")
|
||||
replies_cmds = {msg.command for msg in replies}
|
||||
self.assertIn("PRIVMSG", replies_cmds)
|
||||
self.assertNotIn(ERR_CANNOTSENDTOCHAN, replies_cmds)
|
||||
self.assertEqual(
|
||||
self.getMessages("chanop"),
|
||||
[msg for msg in replies if msg.command == "PRIVMSG"],
|
||||
)
|
49
irctest/server_tests/names.py
Normal file
49
irctest/server_tests/names.py
Normal file
@ -0,0 +1,49 @@
|
||||
from irctest import cases
|
||||
from irctest.numerics import RPL_ENDOFNAMES
|
||||
from irctest.patma import ANYSTR
|
||||
|
||||
|
||||
class NamesTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
|
||||
def testNamesInvalidChannel(self):
|
||||
"""
|
||||
"There is no error reply for bad channel names."
|
||||
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
|
||||
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
|
||||
|
||||
"If the channel name is invalid or the channel does not exist,
|
||||
one `RPL_ENDOFNAMES` numeric containing the given channel name
|
||||
should be returned."
|
||||
-- https://modern.ircdocs.horse/#names-message
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.getMessages(1)
|
||||
|
||||
self.sendLine(1, "NAMES invalid")
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command=RPL_ENDOFNAMES,
|
||||
params=["foo", "invalid", ANYSTR],
|
||||
)
|
||||
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
|
||||
def testNamesNonexistingChannel(self):
|
||||
"""
|
||||
"There is no error reply for bad channel names."
|
||||
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
|
||||
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
|
||||
|
||||
"If the channel name is invalid or the channel does not exist,
|
||||
one `RPL_ENDOFNAMES` numeric containing the given channel name
|
||||
should be returned."
|
||||
-- https://modern.ircdocs.horse/#names-message
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.getMessages(1)
|
||||
|
||||
self.sendLine(1, "NAMES #nonexisting")
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command=RPL_ENDOFNAMES,
|
||||
params=["foo", "#nonexisting", ANYSTR],
|
||||
)
|
148
irctest/server_tests/part.py
Normal file
148
irctest/server_tests/part.py
Normal file
@ -0,0 +1,148 @@
|
||||
import time
|
||||
|
||||
from irctest import cases
|
||||
|
||||
|
||||
class PartTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testPartNotInEmptyChannel(self):
|
||||
"""“442 ERR_NOTONCHANNEL
|
||||
"<channel> :You're not on that channel"
|
||||
|
||||
- Returned by the server whenever a client tries to
|
||||
perform a channel effecting command for which the
|
||||
client isn't a member.”
|
||||
-- <https://tools.ietf.org/html/rfc1459#section-6.1>
|
||||
and <https://tools.ietf.org/html/rfc2812#section-5.2>
|
||||
|
||||
According to RFCs, ERR_NOSUCHCHANNEL should only be used for invalid
|
||||
channel names:
|
||||
“403 ERR_NOSUCHCHANNEL
|
||||
"<channel name> :No such channel"
|
||||
|
||||
- Used to indicate the given channel name is invalid.”
|
||||
-- <https://tools.ietf.org/html/rfc1459#section-6.1>
|
||||
and <https://tools.ietf.org/html/rfc2812#section-5.2>
|
||||
|
||||
However, many implementations use 479 instead, so let's allow it.
|
||||
<http://danieloaks.net/irc-defs/defs/ircnumerics.html#403>
|
||||
<http://danieloaks.net/irc-defs/defs/ircnumerics.html#479>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.sendLine(1, "PART #chan")
|
||||
m = self.getMessage(1)
|
||||
self.assertIn(
|
||||
m.command,
|
||||
{"442", "403"},
|
||||
m, # ERR_NOTONCHANNEL, ERR_NOSUCHCHANNEL
|
||||
fail_msg="Expected ERR_NOTONCHANNEL (442) or "
|
||||
"ERR_NOSUCHCHANNEL (403) after PARTing an empty channel "
|
||||
"one is not on, but got: {msg}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testPartNotInNonEmptyChannel(self):
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
self.getMessages(1) # Synchronize
|
||||
self.sendLine(2, "PART #chan")
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(
|
||||
m,
|
||||
command="442", # ERR_NOTONCHANNEL
|
||||
fail_msg="Expected ERR_NOTONCHANNEL (442) "
|
||||
"after PARTing a non-empty channel "
|
||||
"one is not on, but got: {msg}",
|
||||
)
|
||||
self.assertEqual(self.getMessages(2), [])
|
||||
|
||||
testPartNotInNonEmptyChannel.__doc__ = testPartNotInEmptyChannel.__doc__
|
||||
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testBasicPart(self):
|
||||
self.connectClient("bar")
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="JOIN", params=["#chan"])
|
||||
|
||||
self.connectClient("baz")
|
||||
self.sendLine(2, "JOIN #chan")
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(m, command="JOIN", params=["#chan"])
|
||||
|
||||
# skip the rest of the JOIN burst:
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
|
||||
self.sendLine(1, "PART #chan")
|
||||
# both the PART'ing client and the other channel member should receive
|
||||
# a PART line:
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="PART")
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(m, command="PART")
|
||||
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testBasicPartRfc2812(self):
|
||||
"""
|
||||
“If a "Part Message" is given, this will be sent
|
||||
instead of the default message, the nickname.”
|
||||
"""
|
||||
self.connectClient("bar")
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="JOIN", params=["#chan"])
|
||||
|
||||
self.connectClient("baz")
|
||||
self.sendLine(2, "JOIN #chan")
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(m, command="JOIN", params=["#chan"])
|
||||
|
||||
# skip the rest of the JOIN burst:
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
|
||||
# Despite `anti_spam_exit_message_time = 0`, hybrid does not immediately
|
||||
# allow custom PART reasons.
|
||||
time.sleep(1)
|
||||
|
||||
self.sendLine(1, "PART #chan :bye everyone")
|
||||
# both the PART'ing client and the other channel member should receive
|
||||
# a PART line:
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="PART", params=["#chan", "bye everyone"])
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(m, command="PART", params=["#chan", "bye everyone"])
|
||||
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testPartMessage(self):
|
||||
"""
|
||||
“If a "Part Message" is given, this will be sent
|
||||
instead of the default message, the nickname.”
|
||||
"""
|
||||
self.connectClient("bar")
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="JOIN", params=["#chan"])
|
||||
|
||||
self.connectClient("baz")
|
||||
self.sendLine(2, "JOIN #chan")
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(m, command="JOIN", params=["#chan"])
|
||||
|
||||
# skip the rest of the JOIN burst:
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
|
||||
# Despite `anti_spam_exit_message_time = 0`, hybrid does not immediately
|
||||
# allow custom PART reasons.
|
||||
time.sleep(1)
|
||||
|
||||
self.sendLine(1, "PART #chan :bye everyone")
|
||||
# both the PART'ing client and the other channel member should receive
|
||||
# a PART line:
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="PART", params=["#chan", "bye everyone"])
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(m, command="PART", params=["#chan", "bye everyone"])
|
60
irctest/server_tests/quit.py
Normal file
60
irctest/server_tests/quit.py
Normal file
@ -0,0 +1,60 @@
|
||||
import time
|
||||
|
||||
from irctest import cases
|
||||
from irctest.numerics import ERR_CANNOTSENDTOCHAN
|
||||
from irctest.patma import StrRe
|
||||
|
||||
|
||||
class ChannelQuitTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testQuit(self):
|
||||
"""“Once a user has joined a channel, he receives information about
|
||||
all commands his server receives affecting the channel. This
|
||||
includes [...] QUIT”
|
||||
<https://tools.ietf.org/html/rfc2812#section-3.2.1>
|
||||
"""
|
||||
self.connectClient("bar")
|
||||
self.joinChannel(1, "#chan")
|
||||
self.connectClient("qux")
|
||||
self.sendLine(2, "JOIN #chan")
|
||||
self.getMessages(2)
|
||||
|
||||
self.getMessages(1)
|
||||
|
||||
# Despite `anti_spam_exit_message_time = 0`, hybrid does not immediately
|
||||
# allow custom PART reasons.
|
||||
time.sleep(1)
|
||||
|
||||
self.sendLine(2, "QUIT :qux out")
|
||||
self.getMessages(2)
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="QUIT", params=[StrRe(".*qux out.*")])
|
||||
self.assertTrue(m.prefix.startswith("qux")) # nickmask of quitter
|
||||
|
||||
|
||||
class NoCTCPTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("Ergo")
|
||||
def testQuit(self):
|
||||
self.connectClient("bar")
|
||||
self.joinChannel(1, "#chan")
|
||||
self.sendLine(1, "MODE #chan +C")
|
||||
self.getMessages(1)
|
||||
|
||||
self.connectClient("qux")
|
||||
self.joinChannel(2, "#chan")
|
||||
self.getMessages(2)
|
||||
|
||||
self.sendLine(1, "PRIVMSG #chan :\x01ACTION hi\x01")
|
||||
self.getMessages(1)
|
||||
ms = self.getMessages(2)
|
||||
self.assertEqual(len(ms), 1)
|
||||
self.assertMessageMatch(
|
||||
ms[0], command="PRIVMSG", params=["#chan", "\x01ACTION hi\x01"]
|
||||
)
|
||||
|
||||
self.sendLine(1, "PRIVMSG #chan :\x01PING 1473523796 918320\x01")
|
||||
ms = self.getMessages(1)
|
||||
self.assertEqual(len(ms), 1)
|
||||
self.assertMessageMatch(ms[0], command=ERR_CANNOTSENDTOCHAN)
|
||||
ms = self.getMessages(2)
|
||||
self.assertEqual(ms, [])
|
199
irctest/server_tests/topic.py
Normal file
199
irctest/server_tests/topic.py
Normal file
@ -0,0 +1,199 @@
|
||||
from irctest import cases, client_mock, runner
|
||||
from irctest.numerics import ERR_CHANOPRIVSNEEDED, RPL_NOTOPIC, RPL_TOPIC, RPL_TOPICTIME
|
||||
|
||||
|
||||
class TopicTestCase(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testTopic(self):
|
||||
"""“Once a user has joined a channel, he receives information about
|
||||
all commands his server receives affecting the channel. This
|
||||
includes […] TOPIC”
|
||||
-- <https://tools.ietf.org/html/rfc1459#section-4.2.1>
|
||||
and <https://tools.ietf.org/html/rfc2812#section-3.2.1>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.joinChannel(1, "#chan")
|
||||
|
||||
self.connectClient("bar")
|
||||
self.joinChannel(2, "#chan")
|
||||
|
||||
# clear waiting msgs about cli 2 joining the channel
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
|
||||
# TODO: check foo is opped OR +t is unset
|
||||
|
||||
self.sendLine(1, "TOPIC #chan :T0P1C")
|
||||
try:
|
||||
m = self.getMessage(1)
|
||||
if m.command == "482":
|
||||
raise runner.ImplementationChoice(
|
||||
"Channel creators are not opped by default, and "
|
||||
"channel modes to no allow regular users to change "
|
||||
"topic."
|
||||
)
|
||||
self.assertMessageMatch(m, command="TOPIC")
|
||||
except client_mock.NoMessageException:
|
||||
# The RFCs do not say TOPIC must be echoed
|
||||
pass
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(m, command="TOPIC", params=["#chan", "T0P1C"])
|
||||
|
||||
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||
def testTopicMode(self):
|
||||
"""“Once a user has joined a channel, he receives information about
|
||||
all commands his server receives affecting the channel. This
|
||||
includes […] TOPIC”
|
||||
-- <https://tools.ietf.org/html/rfc1459#section-4.2.1>
|
||||
and <https://tools.ietf.org/html/rfc2812#section-3.2.1>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.joinChannel(1, "#chan")
|
||||
|
||||
self.connectClient("bar")
|
||||
self.joinChannel(2, "#chan")
|
||||
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
|
||||
# TODO: check foo is opped
|
||||
|
||||
self.sendLine(1, "MODE #chan +t")
|
||||
self.getMessages(1)
|
||||
|
||||
self.getMessages(2)
|
||||
self.sendLine(2, "TOPIC #chan :T0P1C")
|
||||
m = self.getMessage(2)
|
||||
self.assertMessageMatch(
|
||||
m, command="482", fail_msg="Non-op user was not refused use of TOPIC: {msg}"
|
||||
)
|
||||
self.assertEqual(self.getMessages(1), [])
|
||||
|
||||
self.sendLine(1, "MODE #chan -t")
|
||||
self.getMessages(1)
|
||||
self.sendLine(2, "TOPIC #chan :T0P1C")
|
||||
try:
|
||||
m = self.getMessage(2)
|
||||
self.assertNotEqual(
|
||||
m.command,
|
||||
"482",
|
||||
msg="User was refused TOPIC whereas +t was not " "set: {}".format(m),
|
||||
)
|
||||
except client_mock.NoMessageException:
|
||||
# The RFCs do not say TOPIC must be echoed
|
||||
pass
|
||||
m = self.getMessage(1)
|
||||
self.assertMessageMatch(m, command="TOPIC", params=["#chan", "T0P1C"])
|
||||
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testTopicNonexistentChannel(self):
|
||||
"""RFC2812 specifies ERR_NOTONCHANNEL as the correct response to TOPIC
|
||||
on a nonexistent channel. The modern spec prefers ERR_NOSUCHCHANNEL.
|
||||
|
||||
<https://tools.ietf.org/html/rfc2812#section-3.2.4>
|
||||
<http://modern.ircdocs.horse/#topic-message>
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.sendLine(1, "TOPIC #chan")
|
||||
m = self.getMessage(1)
|
||||
# either 403 ERR_NOSUCHCHANNEL or 443 ERR_NOTONCHANNEL
|
||||
self.assertIn(m.command, ("403", "443"))
|
||||
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testUnsetTopicResponses(self):
|
||||
"""Test various cases related to RPL_NOTOPIC with set and unset topics."""
|
||||
self.connectClient("bar")
|
||||
self.sendLine(1, "JOIN #test")
|
||||
messages = self.getMessages(1)
|
||||
# shouldn't send RPL_NOTOPIC for a new channel
|
||||
self.assertNotIn(RPL_NOTOPIC, [m.command for m in messages])
|
||||
|
||||
self.connectClient("baz")
|
||||
self.sendLine(2, "JOIN #test")
|
||||
messages = self.getMessages(2)
|
||||
# topic is still unset, shouldn't send RPL_NOTOPIC on initial join
|
||||
self.assertNotIn(RPL_NOTOPIC, [m.command for m in messages])
|
||||
|
||||
self.sendLine(2, "TOPIC #test")
|
||||
messages = self.getMessages(2)
|
||||
# explicit TOPIC should receive RPL_NOTOPIC
|
||||
self.assertIn(RPL_NOTOPIC, [m.command for m in messages])
|
||||
|
||||
self.getMessages(1)
|
||||
|
||||
self.sendLine(1, "TOPIC #test :new topic")
|
||||
# client 1 should get the new TOPIC line echoed
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1), command="TOPIC", params=["#test", "new topic"]
|
||||
)
|
||||
# client 2 should get the new TOPIC line too
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(2), command="TOPIC", params=["#test", "new topic"]
|
||||
)
|
||||
|
||||
# unset the topic:
|
||||
self.sendLine(1, "TOPIC #test :")
|
||||
# client 1 should get the new TOPIC line echoed, which has the empty arg
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1), command="TOPIC", params=["#test", ""]
|
||||
)
|
||||
# client 2 should get the new TOPIC line to
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(2), command="TOPIC", params=["#test", ""]
|
||||
)
|
||||
|
||||
self.connectClient("qux")
|
||||
self.sendLine(3, "join #test")
|
||||
messages = self.getMessages(3)
|
||||
# topic is once again unset, shouldn't send RPL_NOTOPIC on initial join
|
||||
self.assertNotIn(RPL_NOTOPIC, [m.command for m in messages])
|
||||
|
||||
|
||||
class TopicPrivileges(cases.BaseServerTestCase):
|
||||
@cases.mark_specifications("RFC2812")
|
||||
def testTopicPrivileges(self):
|
||||
# test the +t channel mode, which prevents unprivileged users
|
||||
# from changing the topic
|
||||
self.connectClient("bar", name="bar")
|
||||
self.joinChannel("bar", "#chan")
|
||||
self.getMessages("bar")
|
||||
self.sendLine("bar", "MODE #chan +t")
|
||||
replies = {msg.command for msg in self.getMessages("bar")}
|
||||
# success response is undefined, may be MODE or may be 324 RPL_CHANNELMODEIS,
|
||||
# depending on whether this was a no-op
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
self.sendLine("bar", "TOPIC #chan :new topic")
|
||||
replies = {msg.command for msg in self.getMessages("bar")}
|
||||
self.assertIn("TOPIC", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
self.connectClient("qux", name="qux")
|
||||
self.joinChannel("qux", "#chan")
|
||||
self.getMessages("qux")
|
||||
self.sendLine("qux", "TOPIC #chan :new topic")
|
||||
replies = {msg.command for msg in self.getMessages("qux")}
|
||||
self.assertIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
self.assertNotIn("TOPIC", replies)
|
||||
|
||||
self.sendLine("bar", "MODE #chan +v qux")
|
||||
replies = {msg.command for msg in self.getMessages("bar")}
|
||||
self.assertIn("MODE", replies)
|
||||
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
|
||||
# regression test: +v cannot change the topic of a +t channel
|
||||
self.sendLine("qux", "TOPIC #chan :new topic")
|
||||
replies = {msg.command for msg in self.getMessages("qux")}
|
||||
self.assertIn(ERR_CHANOPRIVSNEEDED, replies)
|
||||
self.assertNotIn("TOPIC", replies)
|
||||
|
||||
# test that RPL_TOPIC and RPL_TOPICTIME are sent on join
|
||||
self.connectClient("buzz", name="buzz")
|
||||
self.sendLine("buzz", "JOIN #chan")
|
||||
replies = self.getMessages("buzz")
|
||||
rpl_topic = [msg for msg in replies if msg.command == RPL_TOPIC][0]
|
||||
self.assertMessageMatch(
|
||||
rpl_topic, command=RPL_TOPIC, params=["buzz", "#chan", "new topic"]
|
||||
)
|
||||
self.assertEqual(
|
||||
len([msg for msg in replies if msg.command == RPL_TOPICTIME]), 1
|
||||
)
|
Reference in New Issue
Block a user