From a9e66056409db6b1df57d80ae75db3eb5e6716cb Mon Sep 17 00:00:00 2001 From: Val Lorentz Date: Thu, 26 Aug 2021 21:04:45 +0200 Subject: [PATCH] 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. --- Makefile | 13 +- irctest/cases.py | 9 + irctest/controllers/inspircd.py | 3 + irctest/server_tests/invite.py | 314 +++++++++++++++++++++++++++++++- 4 files changed, 337 insertions(+), 2 deletions(-) diff --git a/Makefile b/Makefile index 40c7928..21a747d 100644 --- a/Makefile +++ b/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) \ diff --git a/irctest/cases.py b/irctest/cases.py index b94fdc8..569bc94 100644 --- a/irctest/cases.py +++ b/irctest/cases.py @@ -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}" diff --git a/irctest/controllers/inspircd.py b/irctest/controllers/inspircd.py index 1dfab8b..669f0bf 100644 --- a/irctest/controllers/inspircd.py +++ b/irctest/controllers/inspircd.py @@ -21,6 +21,9 @@ TEMPLATE_CONFIG = """ +# Disable 'NOTICE #chan :*** foo invited bar into the channel- + + # Services: " + + - 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)` + + + + Sent as a reply to the [`INVITE`](#invite-message) command to indicate + that the attempt was successful and the client with the nickname `` + has been invited to ``. + """ + 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 + " :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], + )