mirror of
https://github.com/progval/irctest.git
synced 2025-04-05 14:59:49 +00:00
210 lines
7.6 KiB
Python
210 lines
7.6 KiB
Python
"""
|
|
Tests section 4.1 of RFC 1459.
|
|
<https://tools.ietf.org/html/rfc1459#section-4.1>
|
|
"""
|
|
|
|
from irctest import cases
|
|
from irctest.client_mock import ConnectionClosed
|
|
|
|
|
|
class PasswordedConnectionRegistrationTestCase(cases.BaseServerTestCase):
|
|
password = "testpassword"
|
|
|
|
@cases.mark_specifications("RFC1459", "RFC2812")
|
|
def testPassBeforeNickuser(self):
|
|
self.addClient()
|
|
self.sendLine(1, "PASS {}".format(self.password))
|
|
self.sendLine(1, "NICK foo")
|
|
self.sendLine(1, "USER username * * :Realname")
|
|
|
|
m = self.getRegistrationMessage(1)
|
|
self.assertMessageMatch(
|
|
m,
|
|
command="001",
|
|
fail_msg="Did not get 001 after correct PASS+NICK+USER: {msg}",
|
|
)
|
|
|
|
@cases.mark_specifications("RFC1459", "RFC2812")
|
|
def testNoPassword(self):
|
|
self.addClient()
|
|
self.sendLine(1, "NICK foo")
|
|
self.sendLine(1, "USER username * * :Realname")
|
|
m = self.getRegistrationMessage(1)
|
|
self.assertNotEqual(
|
|
m.command, "001", msg="Got 001 after NICK+USER but missing PASS"
|
|
)
|
|
|
|
@cases.mark_specifications("RFC1459", "RFC2812")
|
|
def testWrongPassword(self):
|
|
self.addClient()
|
|
self.sendLine(1, "PASS {}".format(self.password + "garbage"))
|
|
self.sendLine(1, "NICK foo")
|
|
self.sendLine(1, "USER username * * :Realname")
|
|
m = self.getRegistrationMessage(1)
|
|
self.assertNotEqual(
|
|
m.command, "001", msg="Got 001 after NICK+USER but incorrect PASS"
|
|
)
|
|
|
|
@cases.mark_specifications("RFC1459", "RFC2812", strict=True)
|
|
def testPassAfterNickuser(self):
|
|
"""“The password can and must be set before any attempt to register
|
|
the connection is made.”
|
|
-- <https://tools.ietf.org/html/rfc1459#section-4.1.1>
|
|
|
|
“The optional password can and MUST be set before any attempt to
|
|
register the connection is made.
|
|
Currently this requires that user send a PASS command before
|
|
sending the NICK/USER combination.”
|
|
-- <https://tools.ietf.org/html/rfc2812#section-3.1.1>
|
|
"""
|
|
self.addClient()
|
|
self.sendLine(1, "NICK foo")
|
|
self.sendLine(1, "USER username * * :Realname")
|
|
self.sendLine(1, "PASS {}".format(self.password))
|
|
m = self.getRegistrationMessage(1)
|
|
self.assertNotEqual(m.command, "001", "Got 001 after PASS sent after NICK+USER")
|
|
|
|
|
|
class ConnectionRegistrationTestCase(cases.BaseServerTestCase):
|
|
@cases.mark_specifications("RFC1459")
|
|
def testQuitDisconnects(self):
|
|
"""“The server must close the connection to a client which sends a
|
|
QUIT message.”
|
|
-- <https://tools.ietf.org/html/rfc1459#section-4.1.3>
|
|
"""
|
|
self.connectClient("foo")
|
|
self.getMessages(1)
|
|
self.sendLine(1, "QUIT")
|
|
with self.assertRaises(ConnectionClosed):
|
|
self.getMessages(1) # Fetch remaining messages
|
|
self.getMessages(1)
|
|
|
|
@cases.mark_specifications("RFC2812")
|
|
def testQuitErrors(self):
|
|
"""“A client session is terminated with a quit message. The server
|
|
acknowledges this by sending an ERROR message to the client.”
|
|
-- <https://tools.ietf.org/html/rfc2812#section-3.1.7>
|
|
"""
|
|
self.connectClient("foo")
|
|
self.getMessages(1)
|
|
self.sendLine(1, "QUIT")
|
|
try:
|
|
commands = {m.command for m in self.getMessages(1)}
|
|
except ConnectionClosed:
|
|
assert False, "Connection closed without ERROR."
|
|
self.assertIn(
|
|
"ERROR", commands, fail_msg="Did not receive ERROR as a reply to QUIT."
|
|
)
|
|
|
|
def testNickCollision(self):
|
|
"""A user connects and requests the same nickname as an already
|
|
registered user.
|
|
"""
|
|
self.connectClient("foo")
|
|
self.addClient()
|
|
self.sendLine(2, "NICK foo")
|
|
self.sendLine(2, "USER username * * :Realname")
|
|
m = self.getRegistrationMessage(2)
|
|
self.assertNotEqual(
|
|
m.command,
|
|
"001",
|
|
"Received 001 after registering with the nick of a " "registered user.",
|
|
)
|
|
|
|
def testEarlyNickCollision(self):
|
|
"""Two users register simultaneously with the same nick."""
|
|
self.addClient()
|
|
self.addClient()
|
|
self.sendLine(1, "NICK foo")
|
|
self.sendLine(2, "NICK foo")
|
|
self.sendLine(1, "USER username * * :Realname")
|
|
self.sendLine(2, "USER username * * :Realname")
|
|
|
|
try:
|
|
m1 = self.getRegistrationMessage(1)
|
|
except (ConnectionClosed, ConnectionResetError):
|
|
# Unreal closes the connection, see
|
|
# https://bugs.unrealircd.org/view.php?id=5950
|
|
command1 = None
|
|
else:
|
|
command1 = m1.command
|
|
|
|
try:
|
|
m2 = self.getRegistrationMessage(2)
|
|
except (ConnectionClosed, ConnectionResetError):
|
|
# ditto
|
|
command2 = None
|
|
else:
|
|
command2 = m2.command
|
|
|
|
self.assertNotEqual(
|
|
(command1, command2),
|
|
("001", "001"),
|
|
"Two concurrently registering requesting the same nickname "
|
|
"both got 001.",
|
|
)
|
|
|
|
self.assertIn(
|
|
"001",
|
|
(command1, command2),
|
|
"Two concurrently registering requesting the same nickname "
|
|
"neither got 001.",
|
|
)
|
|
|
|
@cases.mark_specifications("IRCv3")
|
|
def testIrc301CapLs(self):
|
|
"""
|
|
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>
|
|
|
|
IRCv3.2: “Servers MUST NOT send messages described by this document if
|
|
the client only supports version 3.1.”
|
|
-- <http://ircv3.net/specs/core/capability-negotiation-3.2.html#version-in-cap-ls>
|
|
""" # noqa
|
|
self.addClient()
|
|
self.sendLine(1, "CAP LS")
|
|
m = self.getRegistrationMessage(1)
|
|
self.assertNotEqual(
|
|
m.params[2],
|
|
"*",
|
|
m,
|
|
fail_msg="Server replied with multi-line CAP LS to a "
|
|
"“CAP LS” (ie. IRCv3.1) request: {msg}",
|
|
)
|
|
self.assertFalse(
|
|
any("=" in cap for cap in m.params[2].split()),
|
|
"Server replied with a name-value capability in "
|
|
"CAP LS reply as a response to “CAP LS” (ie. IRCv3.1) "
|
|
"request: {}".format(m),
|
|
)
|
|
|
|
@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>
|
|
""" # noqa
|
|
self.addClient()
|
|
self.sendLine(1, "CAP LIST")
|
|
m = self.getRegistrationMessage(1)
|
|
self.assertMessageMatch(
|
|
m,
|
|
command="CAP",
|
|
params=["*", "LIST", ""],
|
|
fail_msg="Sending “CAP LIST” as first message got a reply "
|
|
"that is not “CAP * LIST :”: {msg}",
|
|
)
|