mirror of
https://github.com/progval/irctest.git
synced 2025-04-06 15:29:50 +00:00
Add tests for WHOWAS. (#138)
This commit is contained in:
5
.github/workflows/test-devel.yml
vendored
5
.github/workflows/test-devel.yml
vendored
@ -66,7 +66,7 @@ jobs:
|
|||||||
- name: Build Bahamut
|
- name: Build Bahamut
|
||||||
run: |
|
run: |
|
||||||
cd $GITHUB_WORKSPACE/Bahamut/
|
cd $GITHUB_WORKSPACE/Bahamut/
|
||||||
patch src/s_user.c < $GITHUB_WORKSPACE/bahamut_localhost.patch
|
patch src/s_user.c < $GITHUB_WORKSPACE/patches/bahamut_localhost.patch
|
||||||
echo "#undef THROTTLE_ENABLE" >> include/config.h
|
echo "#undef THROTTLE_ENABLE" >> include/config.h
|
||||||
libtoolize --force
|
libtoolize --force
|
||||||
aclocal
|
aclocal
|
||||||
@ -144,7 +144,7 @@ jobs:
|
|||||||
- name: Build InspIRCd
|
- name: Build InspIRCd
|
||||||
run: |
|
run: |
|
||||||
cd $GITHUB_WORKSPACE/inspircd/
|
cd $GITHUB_WORKSPACE/inspircd/
|
||||||
patch src/inspircd.cpp < $GITHUB_WORKSPACE/inspircd_mainloop.patch
|
patch src/inspircd.cpp < $GITHUB_WORKSPACE/patches/inspircd_mainloop.patch
|
||||||
./configure --prefix=$HOME/.local/inspircd --development
|
./configure --prefix=$HOME/.local/inspircd --development
|
||||||
make -j 4
|
make -j 4
|
||||||
make install
|
make install
|
||||||
@ -184,6 +184,7 @@ jobs:
|
|||||||
- name: Build ngircd
|
- name: Build ngircd
|
||||||
run: |
|
run: |
|
||||||
cd $GITHUB_WORKSPACE/ngircd
|
cd $GITHUB_WORKSPACE/ngircd
|
||||||
|
patch src/ngircd/client.c < $GITHUB_WORKSPACE/patches/ngircd_whowas_delay.patch
|
||||||
./autogen.sh
|
./autogen.sh
|
||||||
./configure --prefix=$HOME/.local/
|
./configure --prefix=$HOME/.local/
|
||||||
make -j 4
|
make -j 4
|
||||||
|
2
.github/workflows/test-devel_release.yml
vendored
2
.github/workflows/test-devel_release.yml
vendored
@ -57,7 +57,7 @@ jobs:
|
|||||||
- name: Build InspIRCd
|
- name: Build InspIRCd
|
||||||
run: |
|
run: |
|
||||||
cd $GITHUB_WORKSPACE/inspircd/
|
cd $GITHUB_WORKSPACE/inspircd/
|
||||||
patch src/inspircd.cpp < $GITHUB_WORKSPACE/inspircd_mainloop.patch
|
patch src/inspircd.cpp < $GITHUB_WORKSPACE/patches/inspircd_mainloop.patch
|
||||||
./configure --prefix=$HOME/.local/inspircd --development
|
./configure --prefix=$HOME/.local/inspircd --development
|
||||||
make -j 4
|
make -j 4
|
||||||
make install
|
make install
|
||||||
|
5
.github/workflows/test-stable.yml
vendored
5
.github/workflows/test-stable.yml
vendored
@ -66,7 +66,7 @@ jobs:
|
|||||||
- name: Build Bahamut
|
- name: Build Bahamut
|
||||||
run: |
|
run: |
|
||||||
cd $GITHUB_WORKSPACE/Bahamut/
|
cd $GITHUB_WORKSPACE/Bahamut/
|
||||||
patch src/s_user.c < $GITHUB_WORKSPACE/bahamut_localhost.patch
|
patch src/s_user.c < $GITHUB_WORKSPACE/patches/bahamut_localhost.patch
|
||||||
echo "#undef THROTTLE_ENABLE" >> include/config.h
|
echo "#undef THROTTLE_ENABLE" >> include/config.h
|
||||||
libtoolize --force
|
libtoolize --force
|
||||||
aclocal
|
aclocal
|
||||||
@ -184,7 +184,7 @@ jobs:
|
|||||||
- name: Build InspIRCd
|
- name: Build InspIRCd
|
||||||
run: |
|
run: |
|
||||||
cd $GITHUB_WORKSPACE/inspircd/
|
cd $GITHUB_WORKSPACE/inspircd/
|
||||||
patch src/inspircd.cpp < $GITHUB_WORKSPACE/inspircd_mainloop.patch
|
patch src/inspircd.cpp < $GITHUB_WORKSPACE/patches/inspircd_mainloop.patch
|
||||||
./configure --prefix=$HOME/.local/inspircd --development
|
./configure --prefix=$HOME/.local/inspircd --development
|
||||||
make -j 4
|
make -j 4
|
||||||
make install
|
make install
|
||||||
@ -224,6 +224,7 @@ jobs:
|
|||||||
- name: Build ngircd
|
- name: Build ngircd
|
||||||
run: |
|
run: |
|
||||||
cd $GITHUB_WORKSPACE/ngircd
|
cd $GITHUB_WORKSPACE/ngircd
|
||||||
|
patch src/ngircd/client.c < $GITHUB_WORKSPACE/patches/ngircd_whowas_delay.patch
|
||||||
./autogen.sh
|
./autogen.sh
|
||||||
./configure --prefix=$HOME/.local/
|
./configure --prefix=$HOME/.local/
|
||||||
make -j 4
|
make -j 4
|
||||||
|
4
Makefile
4
Makefile
@ -55,6 +55,7 @@ HYBRID_SELECTORS := \
|
|||||||
# testNoticeNonexistentChannel fails because of https://github.com/inspircd/inspircd/issues/1849
|
# testNoticeNonexistentChannel fails because of https://github.com/inspircd/inspircd/issues/1849
|
||||||
# testBotPrivateMessage and testBotChannelMessage fail because https://github.com/inspircd/inspircd/pull/1910 is not released yet
|
# testBotPrivateMessage and testBotChannelMessage fail because https://github.com/inspircd/inspircd/pull/1910 is not released yet
|
||||||
# testNamesInvalidChannel and testNamesNonexistingChannel fail because https://github.com/inspircd/inspircd/pull/1922 is not released yet.
|
# testNamesInvalidChannel and testNamesNonexistingChannel fail because https://github.com/inspircd/inspircd/pull/1922 is not released yet.
|
||||||
|
# WHOWAS tests fail because https://github.com/inspircd/inspircd/pull/1967 and https://github.com/inspircd/inspircd/pull/1968 are not released yet
|
||||||
INSPIRCD_SELECTORS := \
|
INSPIRCD_SELECTORS := \
|
||||||
not Ergo \
|
not Ergo \
|
||||||
and not deprecated \
|
and not deprecated \
|
||||||
@ -62,6 +63,7 @@ INSPIRCD_SELECTORS := \
|
|||||||
and not testNoticeNonexistentChannel \
|
and not testNoticeNonexistentChannel \
|
||||||
and not testBotPrivateMessage and not testBotChannelMessage \
|
and not testBotPrivateMessage and not testBotChannelMessage \
|
||||||
and not testNamesInvalidChannel and not testNamesNonexistingChannel \
|
and not testNamesInvalidChannel and not testNamesNonexistingChannel \
|
||||||
|
and not whowas \
|
||||||
$(EXTRA_SELECTORS)
|
$(EXTRA_SELECTORS)
|
||||||
|
|
||||||
# buffering tests fail because ircu2 discards the whole buffer on long lines (TODO: refine how we exclude these tests)
|
# buffering tests fail because ircu2 discards the whole buffer on long lines (TODO: refine how we exclude these tests)
|
||||||
@ -72,6 +74,7 @@ INSPIRCD_SELECTORS := \
|
|||||||
# testKickDefaultComment fails because it uses the nick of the kickee rather than the kicker.
|
# testKickDefaultComment fails because it uses the nick of the kickee rather than the kicker.
|
||||||
# testEmptyRealname fails because it uses a default value instead of ERR_NEEDMOREPARAMS.
|
# testEmptyRealname fails because it uses a default value instead of ERR_NEEDMOREPARAMS.
|
||||||
# HelpTestCase fails because it returns NOTICEs instead of numerics
|
# HelpTestCase fails because it returns NOTICEs instead of numerics
|
||||||
|
# testWhowasCountZero fails: https://github.com/UndernetIRC/ircu2/pull/19
|
||||||
IRCU2_SELECTORS := \
|
IRCU2_SELECTORS := \
|
||||||
not Ergo \
|
not Ergo \
|
||||||
and not deprecated \
|
and not deprecated \
|
||||||
@ -84,6 +87,7 @@ IRCU2_SELECTORS := \
|
|||||||
and not testKickDefaultComment \
|
and not testKickDefaultComment \
|
||||||
and not testEmptyRealname \
|
and not testEmptyRealname \
|
||||||
and not HelpTestCase \
|
and not HelpTestCase \
|
||||||
|
and not testWhowasCountZero \
|
||||||
$(EXTRA_SELECTORS)
|
$(EXTRA_SELECTORS)
|
||||||
|
|
||||||
# same justification as ircu2
|
# same justification as ircu2
|
||||||
|
@ -111,7 +111,7 @@ git clone https://github.com/inspircd/inspircd.git
|
|||||||
cd inspircd
|
cd inspircd
|
||||||
|
|
||||||
# optional, makes tests run considerably faster
|
# optional, makes tests run considerably faster
|
||||||
patch src/inspircd.cpp < ~/irctest/inspircd_mainloop.patch
|
patch src/inspircd.cpp < ~/irctest/patches/inspircd_mainloop.patch
|
||||||
|
|
||||||
./configure --prefix=$HOME/.local/ --development
|
./configure --prefix=$HOME/.local/ --development
|
||||||
make -j 4
|
make -j 4
|
||||||
|
@ -26,6 +26,9 @@ TEMPLATE_CONFIG = """
|
|||||||
Passive = yes # don't connect to it
|
Passive = yes # don't connect to it
|
||||||
ServiceMask = *Serv
|
ServiceMask = *Serv
|
||||||
|
|
||||||
|
[Options]
|
||||||
|
MorePrivacy = no # by default, always replies to WHOWAS with ERR_WASNOSUCHNICK
|
||||||
|
|
||||||
[Operator]
|
[Operator]
|
||||||
Name = operuser
|
Name = operuser
|
||||||
Password = operpassword
|
Password = operpassword
|
||||||
|
204
irctest/server_tests/whowas.py
Normal file
204
irctest/server_tests/whowas.py
Normal file
@ -0,0 +1,204 @@
|
|||||||
|
from irctest import cases
|
||||||
|
from irctest.exceptions import ConnectionClosed
|
||||||
|
from irctest.numerics import (
|
||||||
|
RPL_ENDOFWHOWAS,
|
||||||
|
RPL_WHOISACTUALLY,
|
||||||
|
RPL_WHOISSERVER,
|
||||||
|
RPL_WHOWASUSER,
|
||||||
|
)
|
||||||
|
from irctest.patma import ANYSTR, StrRe
|
||||||
|
|
||||||
|
|
||||||
|
class WhowasTestCase(cases.BaseServerTestCase):
|
||||||
|
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||||
|
def testWhowasNumerics(self):
|
||||||
|
"""
|
||||||
|
https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3
|
||||||
|
https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
|
||||||
|
"""
|
||||||
|
self.connectClient("nick1")
|
||||||
|
|
||||||
|
self.connectClient("nick2")
|
||||||
|
self.sendLine(2, "QUIT :bye")
|
||||||
|
try:
|
||||||
|
self.getMessages(2)
|
||||||
|
except ConnectionClosed:
|
||||||
|
pass
|
||||||
|
|
||||||
|
self.sendLine(1, "WHOWAS nick2")
|
||||||
|
|
||||||
|
messages = []
|
||||||
|
for _ in range(10):
|
||||||
|
messages.extend(self.getMessages(1))
|
||||||
|
if RPL_ENDOFWHOWAS in (m.command for m in messages):
|
||||||
|
break
|
||||||
|
|
||||||
|
last_message = messages.pop()
|
||||||
|
|
||||||
|
self.assertMessageMatch(
|
||||||
|
last_message,
|
||||||
|
command=RPL_ENDOFWHOWAS,
|
||||||
|
params=["nick1", "nick2", ANYSTR],
|
||||||
|
fail_msg=f"Last message was not RPL_ENDOFWHOWAS ({RPL_ENDOFWHOWAS})",
|
||||||
|
)
|
||||||
|
|
||||||
|
unexpected_messages = []
|
||||||
|
|
||||||
|
# Straight from the RFCs
|
||||||
|
for m in messages:
|
||||||
|
if m.command == RPL_WHOWASUSER:
|
||||||
|
host_re = "[0-9A-Za-z_:.-]+"
|
||||||
|
self.assertMessageMatch(
|
||||||
|
m,
|
||||||
|
params=[
|
||||||
|
"nick1",
|
||||||
|
"nick2",
|
||||||
|
StrRe("~?username"),
|
||||||
|
StrRe(host_re),
|
||||||
|
"*",
|
||||||
|
"Realname",
|
||||||
|
],
|
||||||
|
)
|
||||||
|
elif m.command == RPL_WHOISSERVER:
|
||||||
|
self.assertMessageMatch(
|
||||||
|
m, params=["nick1", "nick2", "My.Little.Server", ANYSTR]
|
||||||
|
)
|
||||||
|
elif m.command == RPL_WHOISACTUALLY:
|
||||||
|
# Technically not allowed by the RFCs, but Solanum uses it.
|
||||||
|
# Not checking the syntax here; WhoisTestCase does it.
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
unexpected_messages.append(m)
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
unexpected_messages, [], fail_msg="Unexpected numeric messages: {got}"
|
||||||
|
)
|
||||||
|
|
||||||
|
def _testWhowasMultiple(self, second_result, whowas_command):
|
||||||
|
"""
|
||||||
|
"The history is searched backward, returning the most recent entry first."
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
|
||||||
|
"""
|
||||||
|
# TODO: this test assumes the order is always: RPL_WHOWASUSER, then
|
||||||
|
# optional RPL_WHOISACTUALLY, then RPL_WHOISSERVER; but the RFCs
|
||||||
|
# don't specify the order.
|
||||||
|
self.connectClient("nick1")
|
||||||
|
|
||||||
|
self.connectClient("nick2", ident="ident2")
|
||||||
|
self.sendLine(2, "QUIT :bye")
|
||||||
|
try:
|
||||||
|
self.getMessages(2)
|
||||||
|
except ConnectionClosed:
|
||||||
|
pass
|
||||||
|
|
||||||
|
self.connectClient("nick2", ident="ident3")
|
||||||
|
self.sendLine(3, "QUIT :bye")
|
||||||
|
try:
|
||||||
|
self.getMessages(3)
|
||||||
|
except ConnectionClosed:
|
||||||
|
pass
|
||||||
|
|
||||||
|
self.sendLine(1, whowas_command)
|
||||||
|
|
||||||
|
messages = self.getMessages(1)
|
||||||
|
|
||||||
|
# nick2 with ident3
|
||||||
|
self.assertMessageMatch(
|
||||||
|
messages.pop(0),
|
||||||
|
command=RPL_WHOWASUSER,
|
||||||
|
params=[
|
||||||
|
"nick1",
|
||||||
|
"nick2",
|
||||||
|
StrRe("~?ident3"),
|
||||||
|
ANYSTR,
|
||||||
|
"*",
|
||||||
|
"Realname",
|
||||||
|
],
|
||||||
|
)
|
||||||
|
while messages[0].command in (RPL_WHOISACTUALLY, RPL_WHOISSERVER):
|
||||||
|
# don't care
|
||||||
|
messages.pop(0)
|
||||||
|
|
||||||
|
if second_result:
|
||||||
|
# nick2 with ident2
|
||||||
|
self.assertMessageMatch(
|
||||||
|
messages.pop(0),
|
||||||
|
command=RPL_WHOWASUSER,
|
||||||
|
params=[
|
||||||
|
"nick1",
|
||||||
|
"nick2",
|
||||||
|
StrRe("~?ident2"),
|
||||||
|
ANYSTR,
|
||||||
|
"*",
|
||||||
|
"Realname",
|
||||||
|
],
|
||||||
|
)
|
||||||
|
if messages[0].command == RPL_WHOISACTUALLY:
|
||||||
|
# don't care
|
||||||
|
messages.pop(0)
|
||||||
|
while messages[0].command in (RPL_WHOISACTUALLY, RPL_WHOISSERVER):
|
||||||
|
# don't care
|
||||||
|
messages.pop(0)
|
||||||
|
|
||||||
|
self.assertMessageMatch(
|
||||||
|
messages.pop(0),
|
||||||
|
command=RPL_ENDOFWHOWAS,
|
||||||
|
params=["nick1", "nick2", ANYSTR],
|
||||||
|
fail_msg=f"Last message was not RPL_ENDOFWHOWAS ({RPL_ENDOFWHOWAS})",
|
||||||
|
)
|
||||||
|
|
||||||
|
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||||
|
def testWhowasMultiple(self):
|
||||||
|
"""
|
||||||
|
"The history is searched backward, returning the most recent entry first."
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
|
||||||
|
"""
|
||||||
|
self._testWhowasMultiple(second_result=True, whowas_command="WHOWAS nick2")
|
||||||
|
|
||||||
|
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||||
|
def testWhowasCount1(self):
|
||||||
|
"""
|
||||||
|
"If there are multiple entries, up to <count> replies will be returned"
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
|
||||||
|
"""
|
||||||
|
self._testWhowasMultiple(second_result=False, whowas_command="WHOWAS nick2 1")
|
||||||
|
|
||||||
|
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||||
|
def testWhowasCount2(self):
|
||||||
|
"""
|
||||||
|
"If there are multiple entries, up to <count> replies will be returned"
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
|
||||||
|
"""
|
||||||
|
self._testWhowasMultiple(second_result=True, whowas_command="WHOWAS nick2 2")
|
||||||
|
|
||||||
|
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||||
|
def testWhowasCountNegative(self):
|
||||||
|
"""
|
||||||
|
"If a non-positive number is passed as being <count>, then a full search
|
||||||
|
is done."
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
|
||||||
|
"""
|
||||||
|
self._testWhowasMultiple(second_result=True, whowas_command="WHOWAS nick2 -1")
|
||||||
|
|
||||||
|
@cases.mark_specifications("RFC1459", "RFC2812")
|
||||||
|
def testWhowasCountZero(self):
|
||||||
|
"""
|
||||||
|
"If a non-positive number is passed as being <count>, then a full search
|
||||||
|
is done."
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
|
||||||
|
"""
|
||||||
|
self._testWhowasMultiple(second_result=True, whowas_command="WHOWAS nick2 0")
|
||||||
|
|
||||||
|
@cases.mark_specifications("RFC2812", deprecated=True)
|
||||||
|
def testWhowasWildcard(self):
|
||||||
|
"""
|
||||||
|
"Wildcards are allowed in the <target> parameter."
|
||||||
|
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
|
||||||
|
"""
|
||||||
|
self._testWhowasMultiple(second_result=True, whowas_command="WHOWAS *ck2")
|
19
patches/ngircd_whowas_delay.patch
Normal file
19
patches/ngircd_whowas_delay.patch
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
ngIRCd skips WHOWAS entries for users that were connected for less
|
||||||
|
than 30 seconds.
|
||||||
|
|
||||||
|
To avoid waiting 30s in every WHOWAS test, we need to remove this.
|
||||||
|
|
||||||
|
diff --git a/src/ngircd/client.c b/src/ngircd/client.c
|
||||||
|
index 67c02604..66e8e540 100644
|
||||||
|
--- a/src/ngircd/client.c
|
||||||
|
+++ b/src/ngircd/client.c
|
||||||
|
@@ -1490,9 +1490,6 @@ Client_RegisterWhowas( CLIENT *Client )
|
||||||
|
return;
|
||||||
|
|
||||||
|
now = time(NULL);
|
||||||
|
- /* Don't register clients that were connected less than 30 seconds. */
|
||||||
|
- if( now - Client->starttime < 30 )
|
||||||
|
- return;
|
||||||
|
|
||||||
|
slot = Last_Whowas + 1;
|
||||||
|
if( slot >= MAX_WHOWAS || slot < 0 ) slot = 0;
|
@ -104,7 +104,7 @@ software:
|
|||||||
separate_build_job: true
|
separate_build_job: true
|
||||||
build_script: |
|
build_script: |
|
||||||
cd $GITHUB_WORKSPACE/Bahamut/
|
cd $GITHUB_WORKSPACE/Bahamut/
|
||||||
patch src/s_user.c < $GITHUB_WORKSPACE/bahamut_localhost.patch
|
patch src/s_user.c < $GITHUB_WORKSPACE/patches/bahamut_localhost.patch
|
||||||
echo "#undef THROTTLE_ENABLE" >> include/config.h
|
echo "#undef THROTTLE_ENABLE" >> include/config.h
|
||||||
libtoolize --force
|
libtoolize --force
|
||||||
aclocal
|
aclocal
|
||||||
@ -152,7 +152,7 @@ software:
|
|||||||
separate_build_job: true
|
separate_build_job: true
|
||||||
build_script: &inspircd_build_script |
|
build_script: &inspircd_build_script |
|
||||||
cd $GITHUB_WORKSPACE/inspircd/
|
cd $GITHUB_WORKSPACE/inspircd/
|
||||||
patch src/inspircd.cpp < $GITHUB_WORKSPACE/inspircd_mainloop.patch
|
patch src/inspircd.cpp < $GITHUB_WORKSPACE/patches/inspircd_mainloop.patch
|
||||||
./configure --prefix=$HOME/.local/inspircd --development
|
./configure --prefix=$HOME/.local/inspircd --development
|
||||||
make -j 4
|
make -j 4
|
||||||
make install
|
make install
|
||||||
@ -217,6 +217,7 @@ software:
|
|||||||
separate_build_job: true
|
separate_build_job: true
|
||||||
build_script: |
|
build_script: |
|
||||||
cd $GITHUB_WORKSPACE/ngircd
|
cd $GITHUB_WORKSPACE/ngircd
|
||||||
|
patch src/ngircd/client.c < $GITHUB_WORKSPACE/patches/ngircd_whowas_delay.patch
|
||||||
./autogen.sh
|
./autogen.sh
|
||||||
./configure --prefix=$HOME/.local/
|
./configure --prefix=$HOME/.local/
|
||||||
make -j 4
|
make -j 4
|
||||||
|
Reference in New Issue
Block a user