mirror of
https://github.com/progval/irctest.git
synced 2025-04-05 06:49:47 +00:00
Add exhaustive testing of INVITE. (#87)
* Add exhaustive testing of INVITE. Only tested with Modern, because no one implements the RFC syntax. * Mark testInviteUnopped* as strict tests. * Exclude testInviteInviteOnlyModern on Plexus4 * Add test for ERR_USERONCHANNEL.
This commit is contained in:
13
Makefile
13
Makefile
@ -35,8 +35,10 @@ ERGO_SELECTORS := \
|
||||
not deprecated \
|
||||
$(EXTRA_SELECTORS)
|
||||
|
||||
# testInviteUnoppedModern is the only strict test that Hybrid fails
|
||||
HYBRID_SELECTORS := \
|
||||
not Ergo \
|
||||
and not testInviteUnoppedModern \
|
||||
and not deprecated \
|
||||
$(EXTRA_SELECTORS)
|
||||
|
||||
@ -93,6 +95,15 @@ MAMMON_SELECTORS := \
|
||||
and not strict \
|
||||
$(EXTRA_SELECTORS)
|
||||
|
||||
# testInviteUnoppedModern is the only strict test that Plexus4 fails
|
||||
# testInviteInviteOnlyModern fails because Plexus4 allows non-op to invite if (and only if) the channel is not invite-only
|
||||
PLEXUS4_SELECTORS := \
|
||||
not Ergo \
|
||||
and not testInviteUnoppedModern \
|
||||
and not testInviteInviteOnlyModern \
|
||||
and not deprecated \
|
||||
$(EXTRA_SELECTORS)
|
||||
|
||||
# Limnoria can actually pass all the test so there is none to exclude.
|
||||
# `(foo or not foo)` serves as a `true` value so it doesn't break when
|
||||
# $(EXTRA_SELECTORS) is non-empty
|
||||
@ -239,7 +250,7 @@ plexus4:
|
||||
$(PYTEST) $(PYTEST_ARGS) \
|
||||
--controller irctest.controllers.plexus4 \
|
||||
-m 'not services' \
|
||||
-k "$(HYBRID_SELECTORS)"
|
||||
-k "$(PLEXUS4_SELECTORS)"
|
||||
|
||||
solanum:
|
||||
$(PYTEST) $(PYTEST_ARGS) \
|
||||
|
@ -144,6 +144,7 @@ class _IrcTestCase(Generic[TController]):
|
||||
nick: Optional[str] = None,
|
||||
fail_msg: Optional[str] = None,
|
||||
extra_format: Tuple = (),
|
||||
prefix: Union[None, str, patma.Operator] = None,
|
||||
**kwargs: Any,
|
||||
) -> Optional[str]:
|
||||
"""Returns an error message if the message doesn't match the given arguments,
|
||||
@ -161,6 +162,14 @@ class _IrcTestCase(Generic[TController]):
|
||||
msg=msg,
|
||||
)
|
||||
|
||||
if prefix and not patma.match_string(msg.prefix, prefix):
|
||||
fail_msg = (
|
||||
fail_msg or "expected prefix to match {expects}, got {got}: {msg}"
|
||||
)
|
||||
return fail_msg.format(
|
||||
*extra_format, got=msg.prefix, expects=prefix, msg=msg
|
||||
)
|
||||
|
||||
if params and not patma.match_list(list(msg.params), params):
|
||||
fail_msg = (
|
||||
fail_msg or "expected params to match {expects}, got {got}: {msg}"
|
||||
|
@ -21,6 +21,9 @@ TEMPLATE_CONFIG = """
|
||||
|
||||
<options casemapping="ascii">
|
||||
|
||||
# Disable 'NOTICE #chan :*** foo invited bar into the channel-
|
||||
<security announceinvites="none">
|
||||
|
||||
# Services:
|
||||
<bind address="{services_hostname}" port="{services_port}" type="servers">
|
||||
<link name="services.example.org"
|
||||
|
@ -1,5 +1,15 @@
|
||||
import pytest
|
||||
|
||||
from irctest import cases
|
||||
from irctest.numerics import ERR_INVITEONLYCHAN, ERR_NOSUCHNICK, RPL_INVITING
|
||||
from irctest.numerics import (
|
||||
ERR_CHANOPRIVSNEEDED,
|
||||
ERR_INVITEONLYCHAN,
|
||||
ERR_NOSUCHNICK,
|
||||
ERR_NOTONCHANNEL,
|
||||
ERR_USERONCHANNEL,
|
||||
RPL_INVITING,
|
||||
)
|
||||
from irctest.patma import ANYSTR, StrRe
|
||||
|
||||
|
||||
class InviteTestCase(cases.BaseServerTestCase):
|
||||
@ -98,3 +108,305 @@ class InviteTestCase(cases.BaseServerTestCase):
|
||||
"#chan, “foo” should have received “INVITE #chan bar” but "
|
||||
"got this instead: {msg}",
|
||||
)
|
||||
|
||||
def _testInvite(self, opped, invite_only, modern):
|
||||
"""
|
||||
"Only the user inviting and the user being invited will receive
|
||||
notification of the invitation."
|
||||
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.7
|
||||
|
||||
" 341 RPL_INVITING
|
||||
"<channel> <nick>"
|
||||
|
||||
- Returned by the server to indicate that the
|
||||
attempted INVITE message was successful and is
|
||||
being passed onto the end client."
|
||||
-- https://datatracker.ietf.org/doc/html/rfc1459
|
||||
-- https://datatracker.ietf.org/doc/html/rfc2812
|
||||
|
||||
"When the invite is successful, the server MUST send a `RPL_INVITING`
|
||||
numeric to the command issuer, and an `INVITE` message,
|
||||
with the issuer as prefix, to the target user."
|
||||
-- https://modern.ircdocs.horse/#invite-message
|
||||
|
||||
"### `RPL_INVITING (341)`
|
||||
|
||||
<client> <nick> <channel>
|
||||
|
||||
Sent as a reply to the [`INVITE`](#invite-message) command to indicate
|
||||
that the attempt was successful and the client with the nickname `<nick>`
|
||||
has been invited to `<channel>`.
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
self.getMessages(1)
|
||||
|
||||
if invite_only:
|
||||
self.sendLine(1, "MODE #chan +i")
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command="MODE",
|
||||
params=["#chan", "+i"],
|
||||
)
|
||||
|
||||
if not opped:
|
||||
self.sendLine(1, "MODE #chan -o foo")
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command="MODE",
|
||||
params=["#chan", "-o", "foo"],
|
||||
)
|
||||
|
||||
self.sendLine(1, "INVITE bar #chan")
|
||||
if modern:
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command=RPL_INVITING,
|
||||
params=["foo", "bar", "#chan"],
|
||||
fail_msg=f"After “foo” invited “bar” to a channel, “foo” should have "
|
||||
f"received “{RPL_INVITING} foo #chan bar” but got this instead: "
|
||||
f"{{msg}}",
|
||||
)
|
||||
else:
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command=RPL_INVITING,
|
||||
params=["#chan", "bar"],
|
||||
fail_msg=f"After “foo” invited “bar” to a channel, “foo” should have "
|
||||
f"received “{RPL_INVITING} #chan bar” but got this instead: {{msg}}",
|
||||
)
|
||||
|
||||
messages = self.getMessages(2)
|
||||
self.assertNotEqual(
|
||||
messages,
|
||||
[],
|
||||
fail_msg="After using “INVITE #chan bar”, “bar” received nothing.",
|
||||
)
|
||||
self.assertMessageMatch(
|
||||
messages[0],
|
||||
prefix=StrRe("foo!.*"),
|
||||
command="INVITE",
|
||||
params=["bar", "#chan"],
|
||||
fail_msg="After “foo” invited “bar”, “bar” should have received "
|
||||
"“INVITE bar #chan” but got this instead: {msg}",
|
||||
)
|
||||
|
||||
@pytest.mark.parametrize("invite_only", [True, False])
|
||||
@cases.mark_specifications("Modern")
|
||||
def testInviteModern(self, invite_only):
|
||||
self._testInvite(opped=True, invite_only=invite_only, modern=True)
|
||||
|
||||
@pytest.mark.parametrize("invite_only", [True, False])
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", deprecated=True)
|
||||
def testInviteRfc(self, invite_only):
|
||||
self._testInvite(opped=True, invite_only=invite_only, modern=False)
|
||||
|
||||
@cases.mark_specifications("Modern", strict=True)
|
||||
def testInviteUnoppedModern(self):
|
||||
"""Tests invites from unopped users on not-invite-only chans."""
|
||||
self._testInvite(opped=False, invite_only=False, modern=True)
|
||||
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", deprecated=True, strict=True)
|
||||
def testInviteUnoppedRfc(self, opped, invite_only):
|
||||
"""Tests invites from unopped users on not-invite-only chans."""
|
||||
self._testInvite(opped=False, invite_only=False, modern=False)
|
||||
|
||||
@cases.mark_specifications("RFC2812", "Modern")
|
||||
def testInviteNoNotificationForOtherMembers(self):
|
||||
"""
|
||||
"Other channel members are not notified."
|
||||
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.7
|
||||
|
||||
"Other channel members SHOULD NOT be notified."
|
||||
-- https://modern.ircdocs.horse/#invite-message
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.connectClient("baz")
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
self.getMessages(3)
|
||||
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
self.getMessages(1)
|
||||
|
||||
self.sendLine(3, "JOIN #chan")
|
||||
self.getMessages(3)
|
||||
|
||||
self.sendLine(1, "INVITE bar #chan")
|
||||
self.getMessages(1)
|
||||
|
||||
self.assertEqual(
|
||||
self.getMessages(3),
|
||||
[],
|
||||
fail_msg="After foo used “INVITE #chan bar”, other channel members "
|
||||
"were notified: {got}",
|
||||
)
|
||||
|
||||
def _testInviteInviteOnly(self, modern):
|
||||
"""
|
||||
"To invite a user to a channel which is invite only (MODE
|
||||
+i), the client sending the invite must be recognised as being a
|
||||
channel operator on the given channel."
|
||||
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.7
|
||||
|
||||
"When the channel has invite-only
|
||||
flag set, only channel operators may issue INVITE command."
|
||||
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.7
|
||||
|
||||
"When the channel has [invite-only](#invite-only-channel-mode) mode set,
|
||||
only channel operators may issue INVITE command.
|
||||
Otherwise, the server MUST reject the command with the `ERR_CHANOPRIVSNEEDED`
|
||||
numeric."
|
||||
-- https://modern.ircdocs.horse/#invite-message
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
self.getMessages(1)
|
||||
|
||||
self.sendLine(1, "MODE #chan +i")
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command="MODE",
|
||||
params=["#chan", "+i"],
|
||||
)
|
||||
|
||||
self.sendLine(1, "MODE #chan -o foo")
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command="MODE",
|
||||
params=["#chan", "-o", "foo"],
|
||||
)
|
||||
|
||||
self.sendLine(1, "INVITE bar #chan")
|
||||
if modern:
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command=ERR_CHANOPRIVSNEEDED,
|
||||
params=["foo", "#chan", ANYSTR],
|
||||
fail_msg=f"After “foo” invited “bar” to a channel to an invite-only "
|
||||
f"channel without being opped, “foo” should have received "
|
||||
f"“{ERR_CHANOPRIVSNEEDED} foo #chan :*” but got this instead: {{msg}}",
|
||||
)
|
||||
else:
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command=ERR_CHANOPRIVSNEEDED,
|
||||
params=["#chan", ANYSTR],
|
||||
fail_msg=f"After “foo” invited “bar” to a channel to an invite-only "
|
||||
f"channel without being opped, “foo” should have received "
|
||||
f"“{ERR_CHANOPRIVSNEEDED} #chan :*” but got this instead: {{msg}}",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("Modern")
|
||||
def testInviteInviteOnlyModern(self):
|
||||
self._testInviteInviteOnly(modern=True)
|
||||
|
||||
@cases.mark_specifications("RFC1459", "RFC2812", deprecated=True)
|
||||
def testInviteInviteOnlyRfc(self):
|
||||
self._testInviteInviteOnly(modern=False)
|
||||
|
||||
@cases.mark_specifications("RFC2812", "Modern")
|
||||
def _testInviteOnlyFromUsersInChannel(self, modern):
|
||||
"""
|
||||
"if the channel exists, only members of the channel are allowed
|
||||
to invite other users"
|
||||
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.7
|
||||
|
||||
" 442 ERR_NOTONCHANNEL
|
||||
"<channel> :You're not on that channel"
|
||||
|
||||
- Returned by the server whenever a client tries to
|
||||
perform a channel affecting command for which the
|
||||
client isn't a member.
|
||||
"
|
||||
-- https://datatracker.ietf.org/doc/html/rfc2812
|
||||
|
||||
|
||||
" Only members of the channel are allowed to invite other users.
|
||||
Otherwise, the server MUST reject the command with the `ERR_NOTONCHANNEL`
|
||||
numeric."
|
||||
-- https://modern.ircdocs.horse/#invite-message
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.connectClient("baz")
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
self.getMessages(3)
|
||||
|
||||
# Create the channel
|
||||
self.sendLine(3, "JOIN #chan")
|
||||
self.getMessages(3)
|
||||
|
||||
self.sendLine(1, "INVITE bar #chan")
|
||||
if modern:
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command=ERR_NOTONCHANNEL,
|
||||
params=["foo", "#chan", ANYSTR],
|
||||
fail_msg=f"After “foo” invited “bar” to a channel it is not on "
|
||||
f"#chan, “foo” should have received "
|
||||
f"“ERR_NOTONCHANNEL ({ERR_NOTONCHANNEL}) foo #chan :*” but "
|
||||
f"got this instead: {{msg}}",
|
||||
)
|
||||
else:
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command=ERR_NOTONCHANNEL,
|
||||
params=["#chan", ANYSTR],
|
||||
fail_msg=f"After “foo” invited “bar” to a channel it is not on "
|
||||
f"#chan, “foo” should have received "
|
||||
f"“ERR_NOTONCHANNEL ({ERR_NOTONCHANNEL}) #chan :*” but "
|
||||
f"got this instead: {{msg}}",
|
||||
)
|
||||
|
||||
messages = self.getMessages(2)
|
||||
self.assertEqual(
|
||||
messages,
|
||||
[],
|
||||
fail_msg="After using “INVITE #chan bar” while the emitter is "
|
||||
"not in #chan, “bar” received something.",
|
||||
)
|
||||
|
||||
@cases.mark_specifications("Modern")
|
||||
def testInviteOnlyFromUsersInChannelModern(self):
|
||||
self._testInviteOnlyFromUsersInChannel(modern=True)
|
||||
|
||||
@cases.mark_specifications("RFC2812", deprecated=True)
|
||||
def testInviteOnlyFromUsersInChannelRfc(self):
|
||||
self._testInviteOnlyFromUsersInChannel(modern=False)
|
||||
|
||||
@cases.mark_specifications("Modern")
|
||||
def testInviteAlreadyInChannel(self):
|
||||
"""
|
||||
"If the user is already on the target channel,
|
||||
the server MUST reject the command with the `ERR_USERONCHANNEL` numeric."
|
||||
-- https://github.com/ircdocs/modern-irc/pull/80
|
||||
"""
|
||||
self.connectClient("foo")
|
||||
self.connectClient("bar")
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
|
||||
self.sendLine(1, "JOIN #chan")
|
||||
self.sendLine(2, "JOIN #chan")
|
||||
self.getMessages(1)
|
||||
self.getMessages(2)
|
||||
self.getMessages(1)
|
||||
|
||||
self.sendLine(1, "INVITE bar #chan")
|
||||
|
||||
self.assertMessageMatch(
|
||||
self.getMessage(1),
|
||||
command=ERR_USERONCHANNEL,
|
||||
params=["foo", "bar", "#chan", ANYSTR],
|
||||
)
|
||||
|
Reference in New Issue
Block a user