109 Commits

Author SHA1 Message Date
0816232c1c Fix sync issue 2022-11-16 15:39:33 +01:00
3319920250 Check behavior of PRIVMSG when banned 2022-11-16 14:38:35 +01:00
fd0b050686 Add support for Dlk-Services (#176) 2022-11-14 22:58:30 +01:00
d0645ab1a8 dashboard: Use qualified class names in multi-module views 2022-11-12 11:49:14 +01:00
65d7e0e506 whowas: Update quotes and links to Modern spec
In particular, this takes https://github.com/ircdocs/modern-irc/pull/196
into account.
2022-10-22 15:49:30 +02:00
690aaf24a1 Bump flake8 version
Fixes support for importlib_metadata 5.0.0,
https://github.com/PyCQA/flake8/issues/1701
2022-10-22 12:34:46 +02:00
40385c112b add a test for AWAY :\r\n (#175) 2022-09-18 13:27:48 -04:00
9d4212504b Add tests for TIME. (#127) 2022-09-11 17:18:10 +02:00
cae3aec338 workflows: Remove special-casing of Anope 2022-09-10 15:15:29 +02:00
c1442c4301 unrealircd: Use lock around startup/shutdown instead of proot
to ensure no unrealircd instance is starting up while another clears
$PREFIX/tmp/

While proot allows full parallelism and is less error-prone, it takes
a long time to start; and segfaults on my Armbian system.
2022-09-10 14:56:20 +02:00
507f5b7426 Use pathlib to work with temporary config dirs 2022-09-10 14:17:19 +02:00
dbdadec677 test that WHO ignores +i for bare nicknames (#171) 2022-08-26 19:01:41 +02:00
6290825c64 README: Remove reference to setup.py 2022-08-20 18:05:47 +02:00
f1c9218fbb Bump Go version for Ergo 2022-08-04 21:24:48 +02:00
6b6017b40c testStarNick: Replace unreliable workaround for irc2 2022-06-27 20:54:04 +02:00
601f49a9ef Fix infinite loop when server is slow (eg. Bahamut) 2022-06-27 20:53:50 +02:00
e205cc1531 bahamut: pre-initialize entropy to avoid freezing on GH Actions 2022-06-19 16:48:26 +02:00
8a4f254a21 Reduce parallelism on other servers as well 2022-06-18 22:01:36 +02:00
81dac6f582 bahamut: lower mainloop delay, and reduce parallelism to make tests less flaky 2022-06-18 20:26:53 +02:00
53710779f0 Prevent tests from blocking for too long
Bahamut frequently gets stuck, and waiting 6h is a waste of time.
2022-06-11 02:08:58 +02:00
058fab85b0 test incorrect channel keys (#169) 2022-05-29 09:49:21 +02:00
683f7c0a15 Fix support of Unreal 5 2022-05-13 22:30:31 +02:00
0f100a5c80 Work around Unreal >=6.0.4 sending RPL_WHOISSPECIAL by default
085490d780
2022-05-13 22:12:40 +02:00
83017483ba test +R user mode as implemented in Ergo (#168) 2022-05-13 19:49:40 +02:00
627f0b6415 Try fixing flakyness of Plexus4 and others 2022-05-01 11:56:09 +02:00
7ccf5a4f9e Check for 'U' ELIST param in testListUsers (#164) 2022-04-28 20:39:20 +02:00
641bea5f0a bot_mode: Make draft/ prefix optional (#167)
The spec is ratified.
2022-04-28 20:38:49 +02:00
8c73ac2b75 patma: Add support for operators in keys
Will be used to match either '@bot' or '@draft/bot'.
2022-04-28 20:12:18 +02:00
ca35069487 Replace remote download of irc2 with a git clone
To avoid flakiness and hitting the irc.org servers too hard
2022-04-26 22:56:01 +02:00
011bdff7e4 Fix ELIST detection 2022-04-26 22:22:36 +02:00
c4d86aef4e bump black to fix click dependency issue
https://github.com/psf/black/ #2964
2022-04-26 21:46:06 +02:00
c0af9bc0a8 add a regression test for ergochat/ergo#1928
LIST on a nonexistent channel does not get an error response.
2022-04-26 21:46:06 +02:00
a15025a276 Add tests for JOIN with some invalid channels in the target param (#163) 2022-04-16 12:15:56 +02:00
a923353ec4 Add test for ban exception mode (+e) (#162) 2022-04-16 08:12:27 +02:00
45dd42e682 Replace incorrect uses of NotImplementedByController exception (#161) 2022-04-15 16:01:36 +02:00
5122c04826 Add tests for the two invite lists (#149)
* Add tests for the two invite lists

* Add workaround for Hybrid

* Skip testInviteList on ircu2

* Fix merge
2022-04-14 21:28:12 +02:00
9bc331483a deploy_to_netlify.py: Fix crash on the first commit of a PR (#160) 2022-04-14 20:21:49 +02:00
2cd5fc1dca dashboard: Add a page for each implementation (#159) 2022-04-14 19:56:06 +02:00
778510e021 Bump Unreal to 6.0.3 and remove ELIST workarounds (#158)
Workarounds that are only still needed for Unreal 5 and and Hybrid/Plexus
2022-04-13 20:54:11 +02:00
8e2670df54 unreal: Prevent download of geoIP database on first startup (#156) 2022-04-13 20:19:07 +02:00
1e01cb3286 Fix CI (#157)
Broken by recent merges
2022-04-13 19:57:16 +02:00
83867dad32 testWrongPassword: Add stricter check of the reply's command (#144) 2022-04-13 18:59:34 +02:00
94cd2d5437 Merge pull request #143 from progval/elist
Add tests for ELIST
2022-04-13 18:58:12 +02:00
a39ce7f19b Merge branch 'master' into elist 2022-04-13 18:57:46 +02:00
363b62cc80 Add tests for LINKS (#147) 2022-04-13 18:56:29 +02:00
6539ed881a Add tests for NAMES (#145) 2022-04-13 18:54:42 +02:00
3ab31ca4de Add tests for WHOWAS as specified in modern-irc (#142)
https://github.com/ircdocs/modern-irc/pull/170
2022-04-13 18:52:12 +02:00
82928bc6fc Sort results 2022-04-12 22:53:50 +02:00
47db85f026 Fix typo 2022-04-12 22:53:02 +02:00
2bc68a2208 Use xfail instead of deselection for known failures (#155) 2022-04-12 22:36:28 +02:00
10b6f8d6da Remove useless 'OptionalityHelper'. 2022-04-12 18:48:03 +02:00
fc4e31e099 dashboard: Omit irrelevant tests from specific tables 2022-04-12 18:33:52 +02:00
d90264ca9f dashboard: fix pagination 2022-04-12 18:33:02 +02:00
0d64e5c1e2 Merge pull request #154 from progval/docstrings
Toplevel docstring maintainance + show them on dashboard
2022-04-10 16:22:57 +02:00
09c31f428a Format the index as columns when possible
To avoid wasting space.
2022-04-10 15:55:53 +02:00
e92aee012b Fix CI 2022-04-10 15:55:53 +02:00
358b6c2213 dashboard: Show module and class docstrings 2022-04-10 15:55:27 +02:00
a3f0d42248 Remove Ergo-specific mark on channel-rename 2022-04-10 15:55:27 +02:00
397509a282 Move CAP tests to the right module 2022-04-10 15:55:27 +02:00
107af942e9 Add top-level docstrings to all modules
Will be used on the dashboard index in a future commit
2022-04-10 15:55:27 +02:00
93c454c99b Don't use an alias for prod deployment
It prevents deployment to the main domain
2022-04-10 12:11:32 +02:00
d24f0b4f12 Add support for Nefarious (#151) 2022-04-10 11:37:35 +02:00
ca9ec1733c Fix comment 2022-04-10 11:31:26 +02:00
a7d3fadd8b Fix crash on scheduled workflows 2022-04-10 11:08:59 +02:00
edf3e5904b Produce a dashboard website after running tests (#152) 2022-04-10 10:40:39 +02:00
3083aeeb24 fix processing of multiline CAP LS 302 output (#153)
connectClient implicitly assumed that the CAP LS 302 output would be
a single registration message. This caused incorrect skipping of some tests
with `skip_if_cap_nak=True`, for example
RegisterEmailVerifiedTestCase.testAfterConnect on Ergo.

Technically there is no need for connectClient to send CAP LS before CAP REQ;
however, this provides additional test coverage for the syntactic correctness
of the CAP LS output in multiple server configurations, so we might as well
keep it.
2022-04-10 08:39:30 +02:00
ebd7edcc74 workflows: Replace spaces from artifact names
It made them impractical to use as file names.
2022-04-09 08:59:50 +02:00
9a19416731 INVITE: Fix misunderstanding of the RFCs (#148)
They make the first argument of numerics implicit, so there is actually
no difference with Modern
2022-03-31 15:53:51 +02:00
f52f21897b Bump Go version 2022-03-30 20:32:56 +02:00
af001fad2e Add tests for ELIST 2022-03-27 17:08:46 +02:00
a9a7a2a187 list: Modernize tests a bit 2022-03-27 17:08:40 +02:00
72a12ff5ce Add support for 'faketime', to avoid long sleeps in upcoming ELIST tests 2022-03-27 17:08:40 +02:00
3f483243d9 Minor readability improvement 2022-03-27 17:07:29 +02:00
491f92ca60 Use proot with unreal, to make it parallelizable (#146) 2022-03-23 21:26:41 +01:00
7608ea5145 Fix flaky LUSERS tests on Unreal 2022-03-20 22:07:07 +01:00
256a8641ec Add test for multi-target WHOWAS (#141)
* Add test for multi-target WHOWAS

I don't think anyone implements it; let's see

* Skip on Bahamut
2022-03-20 11:36:51 +01:00
f606c075f7 Add tests for error cases of WHOWAS. (#139) 2022-03-19 22:12:25 +01:00
b63ead9546 Bump versions used on the CI. (#140) 2022-03-19 21:39:26 +01:00
7b38c2be8a Add tests for WHOWAS. (#138) 2022-03-19 20:20:50 +01:00
c47b057546 Fix inconsistent arg order 2022-03-19 16:53:14 +01:00
2af62461bc Add test for mismatch on both command and param 2022-03-19 16:34:39 +01:00
69c5dca4b9 Add client tests for SASL with non-ASCII passwords (#137) 2022-03-19 16:09:27 +01:00
ee8f60d6c2 Add test for ISUPPORT PREFIX. (#128) 2022-03-09 20:01:34 +01:00
8356ace014 Shorten ListRemainder's repr() when possible. 2022-03-05 10:12:09 +01:00
2a4e71eccd patma: Fix inconsistencies between ANYSTR and AnyOptStr 2022-03-05 10:12:09 +01:00
66c457f6ce patma: Fix repr() inconsistencies and add tests 2022-03-05 10:12:09 +01:00
7e112359a2 secret channel test (#135)
* silent.py tests for channels with mode +s appearing in LIST only when the user is connected to that channel

* Added assertions for exact content of lines with command RPL_LIST and checks for exact number of RPL_LIST replies

* fix linter errors

* only validate the first two parameters of RPL_LIST

* rename to secret channel test, add citation

* ignore ngircd pseudo-channel

* attempt to fix charybdis/solanum and ircu issues

* review fixes

Co-authored-by: William Rehwinkel <willrehwinkel@gmail.com>
2022-03-04 21:58:05 +01:00
da005d7d24 Add tests for WHOX. (#131) 2022-02-21 21:43:22 +01:00
79c65cf248 Generalize ANYSTR to ListRemainder
So it can match specific strings and have a minimum length.
This can be used to match ISUPPORT-like messages.
2022-02-19 11:55:03 +01:00
d34175d6a8 Fix message matching on empty prefix/params/tags/... 2022-02-19 11:54:44 +01:00
6b1084face Add support for pytest 7.0.0 2022-02-04 20:48:28 +01:00
1371979ccd lusers: Add a variant that ircu2 and snircd can pass + add stricter tests (#126) 2022-01-15 00:19:58 +01:00
88a8f8ad8d Add tests for INFO (#121)
* Add tests for INFO

* Workaround remote INFO being oper-only on some ircds

* Skip testInfoNosuchserver on Ergo

* info: Mark tests with target as deprecated.
2022-01-10 23:55:42 +01:00
255ef1e469 Add tests for the HELP and HELPOP commands (#117)
* Add tests for the HELP and HELPOP commands

* Make testHelpUnknownSubject accept lowercase

* Add support for Hybrid and Plexus4's normalization
2022-01-10 23:55:24 +01:00
cac4428cbd regression test for ergochat/ergo#1898 (#130) 2022-01-10 23:22:46 +01:00
8240cd95cf regression test for ergochat/ergo#1876 (#125) 2022-01-10 21:35:17 +01:00
e8486913a0 workflows: allow go version to float (#129) 2022-01-02 21:54:14 +01:00
c826dd6c2e Bump Go version used to build Ergo 2022-01-02 12:40:24 +01:00
6c393c4e00 Add tests for WHO (#122)
* Add tests for WHO

* Make the mask in RPL_ENDOFWHO case-insensitive + skip test when there is a space in the mask

* Remove 'o' flag of WHO, it's not consistently implemented

* Skip matches on username and realname (for now?)

* Add workarounds from irc2 and ircu2.

* Add test for 'WHO *'.

* Exclude mask tests in test_who.py for Bahamut
2021-12-23 17:15:10 +01:00
05e78802ca Add support for Unreal 6 (#123)
List of issues we had:
 
* echo wallops missing `!user@host` [wallops: Send a full NUH as prefix in echoed WALLOPS unrealircd/unrealircd#186](https://github.com/unrealircd/unrealircd/pull/186)
* RPL_MONONLINE is (re)sent on nick case change - https://bugs.unrealircd.org/view.php?id=6013
* MONITOR accepts masks - https://bugs.unrealircd.org/view.php?id=6014
* KICK doesn't support multiple channels anymore, despite unchanged TARGMAX - https://bugs.unrealircd.org/view.php?id=6015
2021-12-19 00:45:51 +01:00
16533de157 Fix invalid nick 2021-12-10 17:14:26 +01:00
d29c0035e6 test that ERR_UNKNOWNCOMMAND is labeled
If an invalid command is sent with a valid label, it should receive
ERR_UNKNOWNCOMMAND with a label (comparable to other error responses).
2021-12-03 10:15:32 +01:00
18befc9e96 inspircd: Increase limit of connections on insp4
There used to be no limit by default, but 460220fbf5 set it to 3.
2021-11-25 20:32:14 +01:00
2684e7edb7 Enable services tests for hybrid and plexus on the CI (#120)
* Enable services tests for hybrid and plexus on the CI

* Workaround the broken Github CI's host config
2021-11-20 12:15:07 +01:00
b895539bdd Update links to WHOIS spec. 2021-11-12 22:34:58 +01:00
e89584b28e Make black ignore irctest/scram/ 2021-11-12 22:34:48 +01:00
9ade524447 Bump Limnoria version to make it pass tests 2021-11-06 22:55:01 +01:00
39587c3c49 Add testBanList 2021-11-06 09:49:12 +01:00
3b96b5992c sts: Don't send the port on secure connections 2021-11-06 09:48:05 +01:00
111 changed files with 6066 additions and 1029 deletions

120
.github/deploy_to_netlify.py vendored Executable file
View File

@ -0,0 +1,120 @@
#!/usr/bin/env python3
import json
import os
import pprint
import re
import subprocess
import sys
import urllib.request
event_name = os.environ["GITHUB_EVENT_NAME"]
is_pull_request = is_push = False
if event_name.startswith("pull_request"):
is_pull_request = True
elif event_name.startswith("push"):
is_push = True
elif event_name.startswith("schedule"):
# Don't publish; scheduled workflows run against the latest commit of every
# implementation, so they are likely to have failed tests for the wrong reasons
sys.exit(0)
else:
print("Unexpected event name:", event_name)
with open(os.environ["GITHUB_EVENT_PATH"]) as fd:
github_event = json.load(fd)
pprint.pprint(github_event)
context_suffix = ""
command = ["netlify", "deploy", "--dir=dashboard/"]
if is_pull_request:
pr_number = github_event["number"]
sha = github_event.get("after") or github_event["pull_request"]["head"]["sha"]
# Aliases can't exceed 37 chars
command.extend(["--alias", f"pr-{pr_number}-{sha[0:10]}"])
context_suffix = " (pull_request)"
elif is_push:
ref = github_event["ref"]
m = re.match("refs/heads/(.*)", ref)
if m:
branch = m.group(1)
sha = github_event["head_commit"]["id"]
if branch in ("main", "master"):
command.extend(["--prod"])
else:
command.extend(["--alias", f"br-{branch[0:23]}-{sha[0:10]}"])
context_suffix = " (push)"
else:
# TODO
pass
print("Running", command)
proc = subprocess.run(command, capture_output=True)
output = proc.stdout.decode()
assert proc.returncode == 0, (output, proc.stderr.decode())
print(output)
m = re.search("https://[^ ]*--[^ ]*netlify.app", output)
assert m
netlify_site_url = m.group(0)
target_url = f"{netlify_site_url}/index.xhtml"
print("Published to", netlify_site_url)
def send_status() -> None:
statuses_url = github_event["repository"]["statuses_url"].format(sha=sha)
payload = {
"state": "success",
"context": f"Dashboard{context_suffix}",
"description": "Table of all test results",
"target_url": target_url,
}
request = urllib.request.Request(
statuses_url,
data=json.dumps(payload).encode(),
headers={
"Authorization": f'token {os.environ["GITHUB_TOKEN"]}',
"Content-Type": "text/json",
"Accept": "application/vnd.github+json",
},
)
response = urllib.request.urlopen(request)
assert response.status == 201, response.read()
send_status()
def send_pr_comment() -> None:
comments_url = github_event["pull_request"]["_links"]["comments"]["href"]
payload = {
"body": f"[Test results]({target_url})",
}
request = urllib.request.Request(
comments_url,
data=json.dumps(payload).encode(),
headers={
"Authorization": f'token {os.environ["GITHUB_TOKEN"]}',
"Content-Type": "text/json",
"Accept": "application/vnd.github+json",
},
)
response = urllib.request.urlopen(request)
assert response.status == 201, response.read()
if is_pull_request:
send_pr_comment()

View File

@ -5,18 +5,22 @@ jobs:
build-anope: build-anope:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2
- name: Create directories - name: Create directories
run: cd ~/; mkdir -p .local/ go/ run: cd ~/; mkdir -p .local/ go/
- name: Cache Anope - name: Cache dependencies
uses: actions/cache@v2 uses: actions/cache@v2
with: with:
key: 3-${{ runner.os }}-anope-2.0.9 key: 3-${{ runner.os }}-anope-devel
path: '~/.cache path: '~/.cache
${{ github.workspace }}/anope ${ github.workspace }/anope
' '
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Checkout Anope - name: Checkout Anope
uses: actions/checkout@v2 uses: actions/checkout@v2
with: with:
@ -24,7 +28,7 @@ jobs:
ref: 2.0.9 ref: 2.0.9
repository: anope/anope repository: anope/anope
- name: Build Anope - name: Build Anope
run: |- run: |
cd $GITHUB_WORKSPACE/anope/ cd $GITHUB_WORKSPACE/anope/
cp $GITHUB_WORKSPACE/data/anope/* . cp $GITHUB_WORKSPACE/data/anope/* .
CFLAGS=-O0 ./Config -quick CFLAGS=-O0 ./Config -quick
@ -66,7 +70,8 @@ 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
patch src/s_bsd.c < $GITHUB_WORKSPACE/patches/bahamut_mainloop.patch
echo "#undef THROTTLE_ENABLE" >> include/config.h echo "#undef THROTTLE_ENABLE" >> include/config.h
libtoolize --force libtoolize --force
aclocal aclocal
@ -144,7 +149,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 +189,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
@ -297,13 +303,13 @@ jobs:
uses: actions/setup-python@v2 uses: actions/setup-python@v2
with: with:
python-version: 3.7 python-version: 3.7
- name: Checkout UnrealIRCd - name: Checkout UnrealIRCd 6
uses: actions/checkout@v2 uses: actions/checkout@v2
with: with:
path: unrealircd path: unrealircd
ref: unreal52 ref: unreal60_dev
repository: unrealircd/unrealircd repository: unrealircd/unrealircd
- name: Build UnrealIRCd - name: Build UnrealIRCd 6
run: | run: |
cd $GITHUB_WORKSPACE/unrealircd/ cd $GITHUB_WORKSPACE/unrealircd/
cp $GITHUB_WORKSPACE/data/unreal/* . cp $GITHUB_WORKSPACE/data/unreal/* .
@ -314,6 +320,8 @@ jobs:
CFLAGS="-O0 -march=x86-64" CXXFLAGS="$CFLAGS" ./Config -quick CFLAGS="-O0 -march=x86-64" CXXFLAGS="$CFLAGS" ./Config -quick
make -j 4 make -j 4
make install make install
# Prevent download of geoIP database on first startup
sed -i 's/loadmodule "geoip_classic";//' ~/.local/unrealircd/conf/modules.default.conf
- name: Make artefact tarball - name: Make artefact tarball
run: cd ~; tar -czf artefacts-unrealircd.tar.gz .local/ go/ run: cd ~; tar -czf artefacts-unrealircd.tar.gz .local/ go/
- name: Upload build artefacts - name: Upload build artefacts
@ -322,9 +330,55 @@ jobs:
name: installed-unrealircd name: installed-unrealircd
path: ~/artefacts-*.tar.gz path: ~/artefacts-*.tar.gz
retention-days: 1 retention-days: 1
build-unrealircd-5:
runs-on: ubuntu-latest
steps:
- name: Create directories
run: cd ~/; mkdir -p .local/ go/
- name: Cache dependencies
uses: actions/cache@v2
with:
key: 3-${{ runner.os }}-unrealircd-5-devel
path: '~/.cache
${ github.workspace }/unrealircd
'
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Checkout UnrealIRCd 5
uses: actions/checkout@v2
with:
path: unrealircd
ref: unreal52
repository: unrealircd/unrealircd
- name: Build UnrealIRCd 5
run: |
cd $GITHUB_WORKSPACE/unrealircd/
cp $GITHUB_WORKSPACE/data/unreal/* .
# Need to use a specific -march, because GitHub has inconsistent
# architectures across workers, which result in random SIGILL with some
# worker combinations
sudo apt install libsodium-dev libargon2-dev
CFLAGS="-O0 -march=x86-64" CXXFLAGS="$CFLAGS" ./Config -quick
make -j 4
make install
# Prevent download of geoIP database on first startup
sed -i 's/loadmodule "geoip_classic";//' ~/.local/unrealircd/conf/modules.default.conf
- name: Make artefact tarball
run: cd ~; tar -czf artefacts-unrealircd-5.tar.gz .local/ go/
- name: Upload build artefacts
uses: actions/upload-artifact@v2
with:
name: installed-unrealircd-5
path: ~/artefacts-*.tar.gz
retention-days: 1
publish-test-results: publish-test-results:
if: success() || failure() if: success() || failure()
name: Publish Unit Tests Results name: Publish Dashboard
needs: needs:
- test-bahamut - test-bahamut
- test-bahamut-anope - test-bahamut-anope
@ -335,6 +389,7 @@ jobs:
- test-inspircd-anope - test-inspircd-anope
- test-ircu2 - test-ircu2
- test-limnoria - test-limnoria
- test-nefarious
- test-ngircd - test-ngircd
- test-ngircd-anope - test-ngircd-anope
- test-ngircd-atheme - test-ngircd-atheme
@ -342,8 +397,10 @@ jobs:
- test-solanum - test-solanum
- test-sopel - test-sopel
- test-unrealircd - test-unrealircd
- test-unrealircd-5
- test-unrealircd-anope - test-unrealircd-anope
- test-unrealircd-atheme - test-unrealircd-atheme
- test-unrealircd-dlk
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
@ -351,27 +408,23 @@ jobs:
uses: actions/download-artifact@v2 uses: actions/download-artifact@v2
with: with:
path: artifacts path: artifacts
- if: github.event_name == 'pull_request' - name: Install dashboard dependencies
name: Publish Unit Test Results run: |-
uses: actions/github-script@v4 python -m pip install --upgrade pip
with: pip install defusedxml docutils -r requirements.txt
result-encoding: string - name: Generate dashboard
script: | run: |-
let body = ''; shopt -s globstar
const options = {}; python3 -m irctest.dashboard.format dashboard/ artifacts/**/*.xml
options.listeners = { echo '/ /index.xhtml' > dashboard/_redirects
stdout: (data) => { - name: Install netlify-cli
body += data.toString(); run: npm i -g netlify-cli
} - env:
}; GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
await exec.exec('bash', ['-c', 'shopt -s globstar; python3 report.py artifacts/**/*.xml'], options); NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
github.issues.createComment({ NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID }}
issue_number: context.issue.number, name: Deploy to Netlify
owner: context.repo.owner, run: ./.github/deploy_to_netlify.py
repo: context.repo.repo,
body: body,
});
return body;
test-bahamut: test-bahamut:
needs: needs:
- build-bahamut - build-bahamut
@ -389,8 +442,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -398,11 +451,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
bahamut bahamut
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results bahamut (devel) name: pytest-results_bahamut_devel
path: pytest.xml path: pytest.xml
test-bahamut-anope: test-bahamut-anope:
needs: needs:
@ -427,8 +481,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -436,11 +490,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
bahamut-anope bahamut-anope
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results bahamut-anope (devel) name: pytest-results_bahamut-anope_devel
path: pytest.xml path: pytest.xml
test-bahamut-atheme: test-bahamut-atheme:
needs: needs:
@ -459,8 +514,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -468,11 +523,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
bahamut-atheme bahamut-atheme
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results bahamut-atheme (devel) name: pytest-results_bahamut-atheme_devel
path: pytest.xml path: pytest.xml
test-ergo: test-ergo:
needs: [] needs: []
@ -491,15 +547,15 @@ jobs:
repository: ergochat/ergo repository: ergochat/ergo
- uses: actions/setup-go@v2 - uses: actions/setup-go@v2
with: with:
go-version: ~1.16 go-version: ^1.19.0
- run: go version - run: go version
- name: Build Ergo - name: Build Ergo
run: | run: |
cd $GITHUB_WORKSPACE/ergo/ cd $GITHUB_WORKSPACE/ergo/
make build make build
make install make install
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -507,15 +563,17 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/go/sbin:~/go/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/go/sbin:~/go/bin:$PATH
make ergo make ergo
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ergo (devel) name: pytest-results_ergo_devel
path: pytest.xml path: pytest.xml
test-hybrid: test-hybrid:
needs: needs:
- build-hybrid - build-hybrid
- build-anope
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
@ -528,22 +586,28 @@ jobs:
with: with:
name: installed-hybrid name: installed-hybrid
path: '~' path: '~'
- name: Download build artefacts
uses: actions/download-artifact@v2
with:
name: installed-anope
path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
pip install pytest pytest-xdist -r requirements.txt pip install pytest pytest-xdist -r requirements.txt
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
hybrid hybrid
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results hybrid (devel) name: pytest-results_hybrid_devel
path: pytest.xml path: pytest.xml
test-inspircd: test-inspircd:
needs: needs:
@ -562,8 +626,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -571,11 +635,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH
make inspircd make inspircd
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results inspircd (devel) name: pytest-results_inspircd_devel
path: pytest.xml path: pytest.xml
test-inspircd-anope: test-inspircd-anope:
needs: needs:
@ -600,8 +665,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -609,11 +674,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH make
inspircd-anope inspircd-anope
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results inspircd-anope (devel) name: pytest-results_inspircd-anope_devel
path: pytest.xml path: pytest.xml
test-ircu2: test-ircu2:
needs: [] needs: []
@ -638,8 +704,8 @@ jobs:
./configure --prefix=$HOME/.local/ --with-maxcon=1024 --enable-debug ./configure --prefix=$HOME/.local/ --with-maxcon=1024 --enable-debug
make -j 4 make -j 4
make install make install
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -647,11 +713,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
ircu2 ircu2
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ircu2 (devel) name: pytest-results_ircu2_devel
path: pytest.xml path: pytest.xml
test-limnoria: test-limnoria:
needs: [] needs: []
@ -665,8 +732,8 @@ jobs:
- name: Install dependencies - name: Install dependencies
run: pip install git+https://github.com/ProgVal/Limnoria.git@testing cryptography run: pip install git+https://github.com/ProgVal/Limnoria.git@testing cryptography
pyxmpp2-scram pyxmpp2-scram
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -674,11 +741,50 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
limnoria limnoria
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results limnoria (devel) name: pytest-results_limnoria_devel
path: pytest.xml
test-nefarious:
needs: []
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Checkout nefarious
uses: actions/checkout@v2
with:
path: nefarious
ref: master
repository: evilnet/nefarious2
- name: Build nefarious
run: |
cd $GITHUB_WORKSPACE/nefarious
./configure --prefix=$HOME/.local/ --enable-debug
make -j 4
make install
cp $GITHUB_WORKSPACE/data/nefarious/* $HOME/.local/lib
- name: Install system dependencies
run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies
run: |-
python -m pip install --upgrade pip
pip install pytest pytest-xdist -r requirements.txt
- name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
nefarious
timeout-minutes: 30
- if: always()
name: Publish results
uses: actions/upload-artifact@v2
with:
name: pytest-results_nefarious_devel
path: pytest.xml path: pytest.xml
test-ngircd: test-ngircd:
needs: needs:
@ -697,8 +803,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -706,11 +812,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH
make ngircd make ngircd
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ngircd (devel) name: pytest-results_ngircd_devel
path: pytest.xml path: pytest.xml
test-ngircd-anope: test-ngircd-anope:
needs: needs:
@ -735,8 +842,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -744,11 +851,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH make
ngircd-anope ngircd-anope
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ngircd-anope (devel) name: pytest-results_ngircd-anope_devel
path: pytest.xml path: pytest.xml
test-ngircd-atheme: test-ngircd-atheme:
needs: needs:
@ -767,8 +875,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -776,11 +884,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH
make ngircd-atheme make ngircd-atheme
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ngircd-atheme (devel) name: pytest-results_ngircd-atheme_devel
path: pytest.xml path: pytest.xml
test-plexus4: test-plexus4:
needs: needs:
@ -805,8 +914,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -814,11 +923,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
plexus4 plexus4
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results plexus4 (devel) name: pytest-results_plexus4_devel
path: pytest.xml path: pytest.xml
test-solanum: test-solanum:
needs: needs:
@ -837,8 +947,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -846,11 +956,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
solanum solanum
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results solanum (devel) name: pytest-results_solanum_devel
path: pytest.xml path: pytest.xml
test-sopel: test-sopel:
needs: [] needs: []
@ -863,8 +974,8 @@ jobs:
python-version: 3.7 python-version: 3.7
- name: Install dependencies - name: Install dependencies
run: pip install git+https://github.com/sopel-irc/sopel.git run: pip install git+https://github.com/sopel-irc/sopel.git
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -872,11 +983,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
sopel sopel
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results sopel (devel) name: pytest-results_sopel_devel
path: pytest.xml path: pytest.xml
test-unrealircd: test-unrealircd:
needs: needs:
@ -895,8 +1007,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -904,11 +1016,45 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH
make unrealircd make unrealircd
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results unrealircd (devel) name: pytest-results_unrealircd_devel
path: pytest.xml
test-unrealircd-5:
needs:
- build-unrealircd-5
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Download build artefacts
uses: actions/download-artifact@v2
with:
name: installed-unrealircd-5
path: '~'
- name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install system dependencies
run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies
run: |-
python -m pip install --upgrade pip
pip install pytest pytest-xdist -r requirements.txt
- name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH
make unrealircd-5
timeout-minutes: 30
- if: always()
name: Publish results
uses: actions/upload-artifact@v2
with:
name: pytest-results_unrealircd-5_devel
path: pytest.xml path: pytest.xml
test-unrealircd-anope: test-unrealircd-anope:
needs: needs:
@ -933,8 +1079,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -942,11 +1088,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH make
unrealircd-anope unrealircd-anope
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results unrealircd-anope (devel) name: pytest-results_unrealircd-anope_devel
path: pytest.xml path: pytest.xml
test-unrealircd-atheme: test-unrealircd-atheme:
needs: needs:
@ -965,8 +1112,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -974,11 +1121,58 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH
make unrealircd-atheme make unrealircd-atheme
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results unrealircd-atheme (devel) name: pytest-results_unrealircd-atheme_devel
path: pytest.xml
test-unrealircd-dlk:
needs:
- build-unrealircd
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Download build artefacts
uses: actions/download-artifact@v2
with:
name: installed-unrealircd
path: '~'
- name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Checkout Dlk
uses: actions/checkout@v2
with:
path: Dlk-Services
ref: main
repository: DalekIRC/Dalek-Services
- name: Build Dlk
run: |
pip install pifpaf
wget -q https://raw.githubusercontent.com/wp-cli/builds/gh-pages/phar/wp-cli.phar
wget -q https://wordpress.org/latest.zip -O wordpress-latest.zip
- name: Install system dependencies
run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies
run: |-
python -m pip install --upgrade pip
pip install pytest pytest-xdist -r requirements.txt
- name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH
IRCTEST_DLK_PATH="${{ github.workspace }}/Dlk-Services" IRCTEST_WP_CLI_PATH="${{
github.workspace }}/wp-cli.phar" IRCTEST_WP_ZIP_PATH="${{ github.workspace
}}/wordpress-latest.zip" make unrealircd-dlk
timeout-minutes: 30
- if: always()
name: Publish results
uses: actions/upload-artifact@v2
with:
name: pytest-results_unrealircd-dlk_devel
path: pytest.xml path: pytest.xml
name: irctest with devel versions name: irctest with devel versions
'on': 'on':

View File

@ -5,18 +5,22 @@ jobs:
build-anope: build-anope:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2
- name: Create directories - name: Create directories
run: cd ~/; mkdir -p .local/ go/ run: cd ~/; mkdir -p .local/ go/
- name: Cache Anope - name: Cache dependencies
uses: actions/cache@v2 uses: actions/cache@v2
with: with:
key: 3-${{ runner.os }}-anope-2.0.9 key: 3-${{ runner.os }}-anope-devel_release
path: '~/.cache path: '~/.cache
${{ github.workspace }}/anope ${ github.workspace }/anope
' '
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Checkout Anope - name: Checkout Anope
uses: actions/checkout@v2 uses: actions/checkout@v2
with: with:
@ -24,7 +28,7 @@ jobs:
ref: 2.0.9 ref: 2.0.9
repository: anope/anope repository: anope/anope
- name: Build Anope - name: Build Anope
run: |- run: |
cd $GITHUB_WORKSPACE/anope/ cd $GITHUB_WORKSPACE/anope/
cp $GITHUB_WORKSPACE/data/anope/* . cp $GITHUB_WORKSPACE/data/anope/* .
CFLAGS=-O0 ./Config -quick CFLAGS=-O0 ./Config -quick
@ -57,7 +61,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
@ -71,7 +75,7 @@ jobs:
retention-days: 1 retention-days: 1
publish-test-results: publish-test-results:
if: success() || failure() if: success() || failure()
name: Publish Unit Tests Results name: Publish Dashboard
needs: needs:
- test-inspircd - test-inspircd
- test-inspircd-anope - test-inspircd-anope
@ -83,27 +87,23 @@ jobs:
uses: actions/download-artifact@v2 uses: actions/download-artifact@v2
with: with:
path: artifacts path: artifacts
- if: github.event_name == 'pull_request' - name: Install dashboard dependencies
name: Publish Unit Test Results run: |-
uses: actions/github-script@v4 python -m pip install --upgrade pip
with: pip install defusedxml docutils -r requirements.txt
result-encoding: string - name: Generate dashboard
script: | run: |-
let body = ''; shopt -s globstar
const options = {}; python3 -m irctest.dashboard.format dashboard/ artifacts/**/*.xml
options.listeners = { echo '/ /index.xhtml' > dashboard/_redirects
stdout: (data) => { - name: Install netlify-cli
body += data.toString(); run: npm i -g netlify-cli
} - env:
}; GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
await exec.exec('bash', ['-c', 'shopt -s globstar; python3 report.py artifacts/**/*.xml'], options); NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
github.issues.createComment({ NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID }}
issue_number: context.issue.number, name: Deploy to Netlify
owner: context.repo.owner, run: ./.github/deploy_to_netlify.py
repo: context.repo.repo,
body: body,
});
return body;
test-inspircd: test-inspircd:
needs: needs:
- build-inspircd - build-inspircd
@ -121,8 +121,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -130,11 +130,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH
make inspircd make inspircd
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results inspircd (devel_release) name: pytest-results_inspircd_devel_release
path: pytest.xml path: pytest.xml
test-inspircd-anope: test-inspircd-anope:
needs: needs:
@ -159,8 +160,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -168,11 +169,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH make
inspircd-anope inspircd-anope
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results inspircd-anope (devel_release) name: pytest-results_inspircd-anope_devel_release
path: pytest.xml path: pytest.xml
test-inspircd-atheme: test-inspircd-atheme:
needs: needs:
@ -191,8 +193,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -200,11 +202,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH
make inspircd-atheme make inspircd-atheme
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results inspircd-atheme (devel_release) name: pytest-results_inspircd-atheme_devel_release
path: pytest.xml path: pytest.xml
name: irctest with devel_release versions name: irctest with devel_release versions
'on': 'on':

View File

@ -5,18 +5,22 @@ jobs:
build-anope: build-anope:
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2
- name: Create directories - name: Create directories
run: cd ~/; mkdir -p .local/ go/ run: cd ~/; mkdir -p .local/ go/
- name: Cache Anope - name: Cache dependencies
uses: actions/cache@v2 uses: actions/cache@v2
with: with:
key: 3-${{ runner.os }}-anope-2.0.9 key: 3-${{ runner.os }}-anope-stable
path: '~/.cache path: '~/.cache
${{ github.workspace }}/anope ${ github.workspace }/anope
' '
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Checkout Anope - name: Checkout Anope
uses: actions/checkout@v2 uses: actions/checkout@v2
with: with:
@ -24,7 +28,7 @@ jobs:
ref: 2.0.9 ref: 2.0.9
repository: anope/anope repository: anope/anope
- name: Build Anope - name: Build Anope
run: |- run: |
cd $GITHUB_WORKSPACE/anope/ cd $GITHUB_WORKSPACE/anope/
cp $GITHUB_WORKSPACE/data/anope/* . cp $GITHUB_WORKSPACE/data/anope/* .
CFLAGS=-O0 ./Config -quick CFLAGS=-O0 ./Config -quick
@ -61,12 +65,13 @@ jobs:
uses: actions/checkout@v2 uses: actions/checkout@v2
with: with:
path: Bahamut path: Bahamut
ref: v2.2.0 ref: v2.2.1
repository: DALnet/Bahamut repository: DALnet/Bahamut
- 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
patch src/s_bsd.c < $GITHUB_WORKSPACE/patches/bahamut_mainloop.patch
echo "#undef THROTTLE_ENABLE" >> include/config.h echo "#undef THROTTLE_ENABLE" >> include/config.h
libtoolize --force libtoolize --force
aclocal aclocal
@ -149,7 +154,7 @@ jobs:
uses: actions/checkout@v2 uses: actions/checkout@v2
with: with:
path: ircd-hybrid path: ircd-hybrid
ref: 8.2.38 ref: 8.2.39
repository: ircd-hybrid/ircd-hybrid repository: ircd-hybrid/ircd-hybrid
- name: Build Hybrid - name: Build Hybrid
run: | run: |
@ -179,12 +184,12 @@ jobs:
uses: actions/checkout@v2 uses: actions/checkout@v2
with: with:
path: inspircd path: inspircd
ref: v3.10.0 ref: v3.12.0
repository: inspircd/inspircd repository: inspircd/inspircd
- 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 +229,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
@ -256,10 +262,10 @@ jobs:
with: with:
python-version: 3.7 python-version: 3.7
- name: clone - name: clone
run: 'curl https://gitlab.com/rizon/plexus4/-/archive/403a967e3677a2a8420b504f451e7557259e0790/plexus4-403a967e3677a2a8420b504f451e7557259e0790.tar.gz run: 'curl https://gitlab.com/rizon/plexus4/-/archive/20211115_0-611/plexus4-20211115_0-611.tar
| tar -zx | tar -x
mv plexus4* plexus4' mv plexus* plexus4'
- name: build - name: build
run: 'cd $GITHUB_WORKSPACE/plexus4 run: 'cd $GITHUB_WORKSPACE/plexus4
@ -301,7 +307,7 @@ jobs:
uses: actions/checkout@v2 uses: actions/checkout@v2
with: with:
path: solanum path: solanum
ref: e370888264da666a1bd9faac86cd5f2aa06084f4 ref: 492d560ee13e71dc35403fd676e58c2d5bdcf2a9
repository: solanum-ircd/solanum repository: solanum-ircd/solanum
- name: Build Solanum - name: Build Solanum
run: | run: |
@ -337,13 +343,13 @@ jobs:
uses: actions/setup-python@v2 uses: actions/setup-python@v2
with: with:
python-version: 3.7 python-version: 3.7
- name: Checkout UnrealIRCd - name: Checkout UnrealIRCd 6
uses: actions/checkout@v2 uses: actions/checkout@v2
with: with:
path: unrealircd path: unrealircd
ref: 94993a03ca8d3c193c0295c33af39270c3f9d27d ref: cedd23ae9cdd5985ce16e9869cbdb808479c3fc4
repository: unrealircd/unrealircd repository: unrealircd/unrealircd
- name: Build UnrealIRCd - name: Build UnrealIRCd 6
run: | run: |
cd $GITHUB_WORKSPACE/unrealircd/ cd $GITHUB_WORKSPACE/unrealircd/
cp $GITHUB_WORKSPACE/data/unreal/* . cp $GITHUB_WORKSPACE/data/unreal/* .
@ -354,6 +360,8 @@ jobs:
CFLAGS="-O0 -march=x86-64" CXXFLAGS="$CFLAGS" ./Config -quick CFLAGS="-O0 -march=x86-64" CXXFLAGS="$CFLAGS" ./Config -quick
make -j 4 make -j 4
make install make install
# Prevent download of geoIP database on first startup
sed -i 's/loadmodule "geoip_classic";//' ~/.local/unrealircd/conf/modules.default.conf
- name: Make artefact tarball - name: Make artefact tarball
run: cd ~; tar -czf artefacts-unrealircd.tar.gz .local/ go/ run: cd ~; tar -czf artefacts-unrealircd.tar.gz .local/ go/
- name: Upload build artefacts - name: Upload build artefacts
@ -362,9 +370,55 @@ jobs:
name: installed-unrealircd name: installed-unrealircd
path: ~/artefacts-*.tar.gz path: ~/artefacts-*.tar.gz
retention-days: 1 retention-days: 1
build-unrealircd-5:
runs-on: ubuntu-latest
steps:
- name: Create directories
run: cd ~/; mkdir -p .local/ go/
- name: Cache dependencies
uses: actions/cache@v2
with:
key: 3-${{ runner.os }}-unrealircd-5-stable
path: '~/.cache
${ github.workspace }/unrealircd
'
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Checkout UnrealIRCd 5
uses: actions/checkout@v2
with:
path: unrealircd
ref: 6604856973f713a494f83d38992d7d61ce6b9db4
repository: unrealircd/unrealircd
- name: Build UnrealIRCd 5
run: |
cd $GITHUB_WORKSPACE/unrealircd/
cp $GITHUB_WORKSPACE/data/unreal/* .
# Need to use a specific -march, because GitHub has inconsistent
# architectures across workers, which result in random SIGILL with some
# worker combinations
sudo apt install libsodium-dev libargon2-dev
CFLAGS="-O0 -march=x86-64" CXXFLAGS="$CFLAGS" ./Config -quick
make -j 4
make install
# Prevent download of geoIP database on first startup
sed -i 's/loadmodule "geoip_classic";//' ~/.local/unrealircd/conf/modules.default.conf
- name: Make artefact tarball
run: cd ~; tar -czf artefacts-unrealircd-5.tar.gz .local/ go/
- name: Upload build artefacts
uses: actions/upload-artifact@v2
with:
name: installed-unrealircd-5
path: ~/artefacts-*.tar.gz
retention-days: 1
publish-test-results: publish-test-results:
if: success() || failure() if: success() || failure()
name: Publish Unit Tests Results name: Publish Dashboard
needs: needs:
- test-bahamut - test-bahamut
- test-bahamut-anope - test-bahamut-anope
@ -378,6 +432,7 @@ jobs:
- test-irc2 - test-irc2
- test-ircu2 - test-ircu2
- test-limnoria - test-limnoria
- test-nefarious
- test-ngircd - test-ngircd
- test-ngircd-anope - test-ngircd-anope
- test-ngircd-atheme - test-ngircd-atheme
@ -385,8 +440,10 @@ jobs:
- test-solanum - test-solanum
- test-sopel - test-sopel
- test-unrealircd - test-unrealircd
- test-unrealircd-5
- test-unrealircd-anope - test-unrealircd-anope
- test-unrealircd-atheme - test-unrealircd-atheme
- test-unrealircd-dlk
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
@ -394,27 +451,23 @@ jobs:
uses: actions/download-artifact@v2 uses: actions/download-artifact@v2
with: with:
path: artifacts path: artifacts
- if: github.event_name == 'pull_request' - name: Install dashboard dependencies
name: Publish Unit Test Results run: |-
uses: actions/github-script@v4 python -m pip install --upgrade pip
with: pip install defusedxml docutils -r requirements.txt
result-encoding: string - name: Generate dashboard
script: | run: |-
let body = ''; shopt -s globstar
const options = {}; python3 -m irctest.dashboard.format dashboard/ artifacts/**/*.xml
options.listeners = { echo '/ /index.xhtml' > dashboard/_redirects
stdout: (data) => { - name: Install netlify-cli
body += data.toString(); run: npm i -g netlify-cli
} - env:
}; GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
await exec.exec('bash', ['-c', 'shopt -s globstar; python3 report.py artifacts/**/*.xml'], options); NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }}
github.issues.createComment({ NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID }}
issue_number: context.issue.number, name: Deploy to Netlify
owner: context.repo.owner, run: ./.github/deploy_to_netlify.py
repo: context.repo.repo,
body: body,
});
return body;
test-bahamut: test-bahamut:
needs: needs:
- build-bahamut - build-bahamut
@ -432,8 +485,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -441,11 +494,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
bahamut bahamut
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results bahamut (stable) name: pytest-results_bahamut_stable
path: pytest.xml path: pytest.xml
test-bahamut-anope: test-bahamut-anope:
needs: needs:
@ -470,8 +524,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -479,11 +533,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
bahamut-anope bahamut-anope
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results bahamut-anope (stable) name: pytest-results_bahamut-anope_stable
path: pytest.xml path: pytest.xml
test-bahamut-atheme: test-bahamut-atheme:
needs: needs:
@ -502,8 +557,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -511,11 +566,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
bahamut-atheme bahamut-atheme
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results bahamut-atheme (stable) name: pytest-results_bahamut-atheme_stable
path: pytest.xml path: pytest.xml
test-charybdis: test-charybdis:
needs: needs:
@ -534,8 +590,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -543,11 +599,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
charybdis charybdis
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results charybdis (stable) name: pytest-results_charybdis_stable
path: pytest.xml path: pytest.xml
test-ergo: test-ergo:
needs: [] needs: []
@ -566,15 +623,15 @@ jobs:
repository: ergochat/ergo repository: ergochat/ergo
- uses: actions/setup-go@v2 - uses: actions/setup-go@v2
with: with:
go-version: ~1.16 go-version: ^1.19.0
- run: go version - run: go version
- name: Build Ergo - name: Build Ergo
run: | run: |
cd $GITHUB_WORKSPACE/ergo/ cd $GITHUB_WORKSPACE/ergo/
make build make build
make install make install
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -582,15 +639,17 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/go/sbin:~/go/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/go/sbin:~/go/bin:$PATH
make ergo make ergo
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ergo (stable) name: pytest-results_ergo_stable
path: pytest.xml path: pytest.xml
test-hybrid: test-hybrid:
needs: needs:
- build-hybrid - build-hybrid
- build-anope
runs-on: ubuntu-latest runs-on: ubuntu-latest
steps: steps:
- uses: actions/checkout@v2 - uses: actions/checkout@v2
@ -603,22 +662,28 @@ jobs:
with: with:
name: installed-hybrid name: installed-hybrid
path: '~' path: '~'
- name: Download build artefacts
uses: actions/download-artifact@v2
with:
name: installed-anope
path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
pip install pytest pytest-xdist -r requirements.txt pip install pytest pytest-xdist -r requirements.txt
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
hybrid hybrid
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results hybrid (stable) name: pytest-results_hybrid_stable
path: pytest.xml path: pytest.xml
test-inspircd: test-inspircd:
needs: needs:
@ -637,8 +702,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -646,11 +711,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH
make inspircd make inspircd
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results inspircd (stable) name: pytest-results_inspircd_stable
path: pytest.xml path: pytest.xml
test-inspircd-anope: test-inspircd-anope:
needs: needs:
@ -675,8 +741,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -684,11 +750,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH make
inspircd-anope inspircd-anope
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results inspircd-anope (stable) name: pytest-results_inspircd-anope_stable
path: pytest.xml path: pytest.xml
test-inspircd-atheme: test-inspircd-atheme:
needs: needs:
@ -707,8 +774,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -716,11 +783,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/inspircd/sbin:~/.local/inspircd/bin:$PATH
make inspircd-atheme make inspircd-atheme
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results inspircd-atheme (stable) name: pytest-results_inspircd-atheme_stable
path: pytest.xml path: pytest.xml
test-irc2: test-irc2:
needs: [] needs: []
@ -731,38 +799,33 @@ jobs:
uses: actions/setup-python@v2 uses: actions/setup-python@v2
with: with:
python-version: 3.7 python-version: 3.7
- name: Get source code - name: Checkout irc2
run: curl http://ftp.irc.org/ftp/irc/server/irc2.11.2p3.tgz | tar -zx uses: actions/checkout@v2
- name: Configure with:
run: 'cd $GITHUB_WORKSPACE/irc2.11.2p3 path: irc2.11.2p3
ref: 59649f24c3a5c27bad5648b48774f27475bccfd3
repository: irc-archive/irc2-mirror
- name: Build irc2
run: |
# Configure
cd $GITHUB_WORKSPACE/irc2.11.2p3
./configure --prefix=$HOME/.local/ ./configure --prefix=$HOME/.local/
cd x86* cd x86*
echo "#define CMDLINE_CONFIG/" >> config.h echo "#define CMDLINE_CONFIG/" >> config.h
echo "#define DEFAULT_SPLIT_USERS 0" >> config.h echo "#define DEFAULT_SPLIT_USERS 0" >> config.h
echo "#define DEFAULT_SPLIT_SERVERS 0" >> config.h echo "#define DEFAULT_SPLIT_SERVERS 0" >> config.h
#echo "#undef LIST_ALIS_NOTE" >> config.h #echo "#undef LIST_ALIS_NOTE" >> config.h
# TODO: find a better way to make it not fork... # TODO: find a better way to make it not fork...
echo "#define fork() (0)" >> config.h
echo "#define fork() (0)" >> config.h' # Compile and install
- name: Compile and install cd $GITHUB_WORKSPACE/irc2.11.2p3/x86*
run: 'cd $GITHUB_WORKSPACE/irc2.11.2p3/x86*
make -j 4 all make -j 4 all
make install make install
mkdir -p $HOME/.local/bin mkdir -p $HOME/.local/bin
cp $HOME/.local/sbin/ircd $HOME/.local/bin/ircd
cp $HOME/.local/sbin/ircd $HOME/.local/bin/ircd' - name: Install system dependencies
- name: Install Atheme run: sudo apt-get install atheme-services faketime
run: sudo apt-get install atheme-services
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -770,11 +833,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
irc2 irc2
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results irc2 (stable) name: pytest-results_irc2_stable
path: pytest.xml path: pytest.xml
test-ircu2: test-ircu2:
needs: [] needs: []
@ -799,8 +863,8 @@ jobs:
./configure --prefix=$HOME/.local/ --with-maxcon=1024 --enable-debug ./configure --prefix=$HOME/.local/ --with-maxcon=1024 --enable-debug
make -j 4 make -j 4
make install make install
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -808,11 +872,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
ircu2 ircu2
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ircu2 (stable) name: pytest-results_ircu2_stable
path: pytest.xml path: pytest.xml
test-limnoria: test-limnoria:
needs: [] needs: []
@ -824,9 +889,9 @@ jobs:
with: with:
python-version: 3.7 python-version: 3.7
- name: Install dependencies - name: Install dependencies
run: pip install limnoria==2021.06.15 cryptography pyxmpp2-scram run: pip install limnoria==2022.03.17 cryptography pyxmpp2-scram
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -834,11 +899,50 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
limnoria limnoria
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results limnoria (stable) name: pytest-results_limnoria_stable
path: pytest.xml
test-nefarious:
needs: []
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Checkout nefarious
uses: actions/checkout@v2
with:
path: nefarious
ref: 985704168ecada12d9e53b46df6087ef9d9fb40b
repository: evilnet/nefarious2
- name: Build nefarious
run: |
cd $GITHUB_WORKSPACE/nefarious
./configure --prefix=$HOME/.local/ --enable-debug
make -j 4
make install
cp $GITHUB_WORKSPACE/data/nefarious/* $HOME/.local/lib
- name: Install system dependencies
run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies
run: |-
python -m pip install --upgrade pip
pip install pytest pytest-xdist -r requirements.txt
- name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
nefarious
timeout-minutes: 30
- if: always()
name: Publish results
uses: actions/upload-artifact@v2
with:
name: pytest-results_nefarious_stable
path: pytest.xml path: pytest.xml
test-ngircd: test-ngircd:
needs: needs:
@ -857,8 +961,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -866,11 +970,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH
make ngircd make ngircd
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ngircd (stable) name: pytest-results_ngircd_stable
path: pytest.xml path: pytest.xml
test-ngircd-anope: test-ngircd-anope:
needs: needs:
@ -895,8 +1000,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -904,11 +1009,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH make
ngircd-anope ngircd-anope
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ngircd-anope (stable) name: pytest-results_ngircd-anope_stable
path: pytest.xml path: pytest.xml
test-ngircd-atheme: test-ngircd-atheme:
needs: needs:
@ -927,8 +1033,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -936,11 +1042,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local//sbin:~/.local//bin:$PATH
make ngircd-atheme make ngircd-atheme
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results ngircd-atheme (stable) name: pytest-results_ngircd-atheme_stable
path: pytest.xml path: pytest.xml
test-plexus4: test-plexus4:
needs: needs:
@ -965,8 +1072,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -974,11 +1081,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
plexus4 plexus4
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results plexus4 (stable) name: pytest-results_plexus4_stable
path: pytest.xml path: pytest.xml
test-solanum: test-solanum:
needs: needs:
@ -997,8 +1105,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -1006,11 +1114,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
solanum solanum
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results solanum (stable) name: pytest-results_solanum_stable
path: pytest.xml path: pytest.xml
test-sopel: test-sopel:
needs: [] needs: []
@ -1022,9 +1131,9 @@ jobs:
with: with:
python-version: 3.7 python-version: 3.7
- name: Install dependencies - name: Install dependencies
run: pip install sopel==7.1.1 run: pip install sopel==7.1.8
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -1032,11 +1141,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH make
sopel sopel
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results sopel (stable) name: pytest-results_sopel_stable
path: pytest.xml path: pytest.xml
test-unrealircd: test-unrealircd:
needs: needs:
@ -1055,8 +1165,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -1064,11 +1174,45 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH
make unrealircd make unrealircd
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results unrealircd (stable) name: pytest-results_unrealircd_stable
path: pytest.xml
test-unrealircd-5:
needs:
- build-unrealircd-5
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Download build artefacts
uses: actions/download-artifact@v2
with:
name: installed-unrealircd-5
path: '~'
- name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install system dependencies
run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies
run: |-
python -m pip install --upgrade pip
pip install pytest pytest-xdist -r requirements.txt
- name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH
make unrealircd-5
timeout-minutes: 30
- if: always()
name: Publish results
uses: actions/upload-artifact@v2
with:
name: pytest-results_unrealircd-5_stable
path: pytest.xml path: pytest.xml
test-unrealircd-anope: test-unrealircd-anope:
needs: needs:
@ -1093,8 +1237,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -1102,11 +1246,12 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH make run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH make
unrealircd-anope unrealircd-anope
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results unrealircd-anope (stable) name: pytest-results_unrealircd-anope_stable
path: pytest.xml path: pytest.xml
test-unrealircd-atheme: test-unrealircd-atheme:
needs: needs:
@ -1125,8 +1270,8 @@ jobs:
path: '~' path: '~'
- name: Unpack artefacts - name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \; run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Install Atheme - name: Install system dependencies
run: sudo apt-get install atheme-services run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies - name: Install irctest dependencies
run: |- run: |-
python -m pip install --upgrade pip python -m pip install --upgrade pip
@ -1134,11 +1279,58 @@ jobs:
- name: Test with pytest - name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH
make unrealircd-atheme make unrealircd-atheme
timeout-minutes: 30
- if: always() - if: always()
name: Publish results name: Publish results
uses: actions/upload-artifact@v2 uses: actions/upload-artifact@v2
with: with:
name: pytest results unrealircd-atheme (stable) name: pytest-results_unrealircd-atheme_stable
path: pytest.xml
test-unrealircd-dlk:
needs:
- build-unrealircd
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- name: Set up Python 3.7
uses: actions/setup-python@v2
with:
python-version: 3.7
- name: Download build artefacts
uses: actions/download-artifact@v2
with:
name: installed-unrealircd
path: '~'
- name: Unpack artefacts
run: cd ~; find -name 'artefacts-*.tar.gz' -exec tar -xzf '{}' \;
- name: Checkout Dlk
uses: actions/checkout@v2
with:
path: Dlk-Services
ref: effd18652fc1c847d1959089d9cca9ff9837a8c0
repository: DalekIRC/Dalek-Services
- name: Build Dlk
run: |
pip install pifpaf
wget -q https://raw.githubusercontent.com/wp-cli/builds/gh-pages/phar/wp-cli.phar
wget -q https://wordpress.org/latest.zip -O wordpress-latest.zip
- name: Install system dependencies
run: sudo apt-get install atheme-services faketime
- name: Install irctest dependencies
run: |-
python -m pip install --upgrade pip
pip install pytest pytest-xdist -r requirements.txt
- name: Test with pytest
run: PYTEST_ARGS='--junit-xml pytest.xml' PATH=$HOME/.local/bin:$PATH PATH=~/.local/unrealircd/sbin:~/.local/unrealircd/bin:$PATH
IRCTEST_DLK_PATH="${{ github.workspace }}/Dlk-Services" IRCTEST_WP_CLI_PATH="${{
github.workspace }}/wp-cli.phar" IRCTEST_WP_ZIP_PATH="${{ github.workspace
}}/wordpress-latest.zip" make unrealircd-dlk
timeout-minutes: 30
- if: always()
name: Publish results
uses: actions/upload-artifact@v2
with:
name: pytest-results_unrealircd-dlk_stable
path: pytest.xml path: pytest.xml
name: irctest with stable versions name: irctest with stable versions
'on': 'on':

View File

@ -2,7 +2,7 @@ exclude: ^irctest/scram
repos: repos:
- repo: https://github.com/psf/black - repo: https://github.com/psf/black
rev: 20.8b1 rev: 22.3.0
hooks: hooks:
- id: black - id: black
language_version: python3 language_version: python3
@ -13,7 +13,7 @@ repos:
- id: isort - id: isort
- repo: https://gitlab.com/pycqa/flake8 - repo: https://gitlab.com/pycqa/flake8
rev: 3.8.3 rev: 5.0.4
hooks: hooks:
- id: flake8 - id: flake8

130
Makefile
View File

@ -7,74 +7,47 @@ PYTEST_ARGS ?=
# Will be appended at the end of the -k argument to pytest # Will be appended at the end of the -k argument to pytest
EXTRA_SELECTORS ?= EXTRA_SELECTORS ?=
# testPlainLarge fails because it doesn't handle split AUTHENTICATE (reported on IRC)
ANOPE_SELECTORS := \
and not testPlainLarge
# buffering tests cannot pass because of issues with UTF-8 handling: https://github.com/DALnet/bahamut/issues/196
BAHAMUT_SELECTORS := \ BAHAMUT_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \ and not deprecated \
and not strict \ and not strict \
and not IRCv3 \ and not IRCv3 \
and not buffering \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# testQuitErrors is very flaky
# AccountTagTestCase.testInvite fails because https://github.com/solanum-ircd/solanum/issues/166
# testKickDefaultComment fails because it uses the nick of the kickee rather than the kicker.
# testWhoisNumerics[oper] fails because charybdis uses RPL_WHOISSPECIAL instead of RPL_WHOISOPERATOR
CHARYBDIS_SELECTORS := \ CHARYBDIS_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \ and not deprecated \
and not strict \ and not strict \
and not testQuitErrors \
and not testKickDefaultComment \
and not (AccountTagTestCase and testInvite) \
and not (testWhoisNumerics and oper) \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
ERGO_SELECTORS := \ ERGO_SELECTORS := \
not deprecated \ not deprecated \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# testInviteUnoppedModern is the only strict test that Hybrid fails
HYBRID_SELECTORS := \ HYBRID_SELECTORS := \
not Ergo \ not Ergo \
and not testInviteUnoppedModern \
and not deprecated \ and not deprecated \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# 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
# testNamesInvalidChannel and testNamesNonexistingChannel fail because https://github.com/inspircd/inspircd/pull/1922 is not released yet.
INSPIRCD_SELECTORS := \ INSPIRCD_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \ and not deprecated \
and not strict \ and not strict \
and not testNoticeNonexistentChannel \
and not testBotPrivateMessage and not testBotChannelMessage \
and not testNamesInvalidChannel and not testNamesNonexistingChannel \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# buffering tests fail because ircu2 discards the whole buffer on long lines (TODO: refine how we exclude these tests) # HelpTestCase fails because it returns NOTICEs instead of numerics
# testQuit and testQuitErrors fail because ircu2 does not send ERROR or QUIT
# lusers tests fail because they depend on Modern behavior, not just RFC2812 (TODO: update lusers tests to accept RFC2812-compliant implementations)
# statusmsg tests fail because STATUSMSG is present in ISUPPORT, but it not actually supported as PRIVMSG target
# testKeyValidation[empty] fails because ircu2 returns ERR_NEEDMOREPARAMS on empty keys: https://github.com/UndernetIRC/ircu2/issues/13
# 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.
IRCU2_SELECTORS := \ IRCU2_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \ and not deprecated \
and not strict \ and not strict \
and not buffering \ $(EXTRA_SELECTORS)
and not testQuit \
and not lusers \ # same justification as ircu2
and not statusmsg \ # lusers "unregistered" tests fail because
and not (testKeyValidation and empty) \ NEFARIOUS_SELECTORS := \
and not testKickDefaultComment \ not Ergo \
and not testEmptyRealname \ and not deprecated \
and not strict \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# same justification as ircu2 # same justification as ircu2
@ -82,22 +55,12 @@ SNIRCD_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \ and not deprecated \
and not strict \ and not strict \
and not buffering \
and not testQuit \
and not lusers \
and not statusmsg \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# testListEmpty and testListOne fails because irc2 deprecated LIST
# testKickDefaultComment fails because it uses the nick of the kickee rather than the kicker.
# testWallopsPrivileges fails because it ignores the command instead of replying ERR_UNKNOWNCOMMAND
IRC2_SELECTORS := \ IRC2_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \ and not deprecated \
and not strict \ and not strict \
and not testListEmpty and not testListOne \
and not testKickDefaultComment \
and not testWallopsPrivileges \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
MAMMON_SELECTORS := \ MAMMON_SELECTORS := \
@ -106,26 +69,14 @@ MAMMON_SELECTORS := \
and not strict \ and not strict \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# testKeyValidation[spaces] and testKeyValidation[empty] fail because ngIRCd does not validate them https://github.com/ngircd/ngircd/issues/290
# testStarNick: wat
# testEmptyRealname fails because it uses a default value instead of ERR_NEEDMOREPARAMS.
# chathistory tests fail because they need nicks longer than 9 chars
NGIRCD_SELECTORS := \ NGIRCD_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \ and not deprecated \
and not strict \ and not strict \
and not (testKeyValidation and (spaces or empty)) \
and not testStarNick \
and not testEmptyRealname \
and not chathistory \
$(EXTRA_SELECTORS) $(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 := \ PLEXUS4_SELECTORS := \
not Ergo \ not Ergo \
and not testInviteUnoppedModern \
and not testInviteInviteOnlyModern \
and not deprecated \ and not deprecated \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
@ -136,47 +87,33 @@ LIMNORIA_SELECTORS := \
(foo or not foo) \ (foo or not foo) \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# testQuitErrors is too flaky for CI
# testKickDefaultComment fails because solanum uses the nick of the kickee rather than the kicker.
SOLANUM_SELECTORS := \ SOLANUM_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \ and not deprecated \
and not strict \ and not strict \
and not testQuitErrors \
and not testKickDefaultComment \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# Same as Limnoria
SOPEL_SELECTORS := \ SOPEL_SELECTORS := \
not testPlainNotAvailable \ (foo or not foo) \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
# testNoticeNonexistentChannel fails: https://bugs.unrealircd.org/view.php?id=5949
# regressions::testTagCap fails: https://bugs.unrealircd.org/view.php?id=5948
# messages::testLineTooLong fails: https://bugs.unrealircd.org/view.php?id=5947
# testCapRemovalByClient and testNakWhole fail pending https://github.com/unrealircd/unrealircd/pull/148
# Tests marked with arbitrary_client_tags can't pass because Unreal whitelists which tags it relays # Tests marked with arbitrary_client_tags can't pass because Unreal whitelists which tags it relays
# Tests marked with react_tag can't pass because Unreal blocks +draft/react https://github.com/unrealircd/unrealircd/pull/149 # Tests marked with react_tag can't pass because Unreal blocks +draft/react https://github.com/unrealircd/unrealircd/pull/149
# Tests marked with private_chathistory can't pass because Unreal does not implement CHATHISTORY for DMs # Tests marked with private_chathistory can't pass because Unreal does not implement CHATHISTORY for DMs
# testChathistory[BETWEEN] fails: https://bugs.unrealircd.org/view.php?id=5952
# testChathistory[AROUND] fails: https://bugs.unrealircd.org/view.php?id=5953
UNREALIRCD_SELECTORS := \ UNREALIRCD_SELECTORS := \
not Ergo \ not Ergo \
and not deprecated \ and not deprecated \
and not strict \ and not strict \
and not testNoticeNonexistentChannel \
and not (regressions.py and testTagCap) \
and not (messages.py and testLineTooLong) \
and not (cap.py and (testCapRemovalByClient or testNakWhole)) \
and not (account_tag.py and testInvite) \
and not arbitrary_client_tags \ and not arbitrary_client_tags \
and not react_tag \ and not react_tag \
and not private_chathistory \ and not private_chathistory \
and not (testChathistory and (between or around)) \
$(EXTRA_SELECTORS) $(EXTRA_SELECTORS)
.PHONY: all flakes bahamut charybdis ergo inspircd ircu2 snircd irc2 mammon limnoria sopel solanum unrealircd .PHONY: all flakes bahamut charybdis ergo inspircd ircu2 snircd irc2 mammon nefarious limnoria sopel solanum unrealircd
all: flakes bahamut charybdis ergo inspircd ircu2 snircd irc2 mammon limnoria sopel solanum unrealircd all: flakes bahamut charybdis ergo inspircd ircu2 snircd irc2 mammon nefarious limnoria sopel solanum unrealircd
flakes: flakes:
find irctest/ -name "*.py" -not -path "irctest/scram/*" -print0 | xargs -0 pyflakes3 find irctest/ -name "*.py" -not -path "irctest/scram/*" -print0 | xargs -0 pyflakes3
@ -185,7 +122,8 @@ bahamut:
$(PYTEST) $(PYTEST_ARGS) \ $(PYTEST) $(PYTEST_ARGS) \
--controller=irctest.controllers.bahamut \ --controller=irctest.controllers.bahamut \
-m 'not services' \ -m 'not services' \
-n 10 \ -n 4 \
-vv -s \
-k '$(BAHAMUT_SELECTORS)' -k '$(BAHAMUT_SELECTORS)'
bahamut-atheme: bahamut-atheme:
@ -193,7 +131,6 @@ bahamut-atheme:
--controller=irctest.controllers.bahamut \ --controller=irctest.controllers.bahamut \
--services-controller=irctest.controllers.atheme_services \ --services-controller=irctest.controllers.atheme_services \
-m 'services' \ -m 'services' \
-n 10 \
-k '$(BAHAMUT_SELECTORS)' -k '$(BAHAMUT_SELECTORS)'
bahamut-anope: bahamut-anope:
@ -201,8 +138,7 @@ bahamut-anope:
--controller=irctest.controllers.bahamut \ --controller=irctest.controllers.bahamut \
--services-controller=irctest.controllers.anope_services \ --services-controller=irctest.controllers.anope_services \
-m 'services' \ -m 'services' \
-n 10 \ -k '$(BAHAMUT_SELECTORS)'
-k '$(BAHAMUT_SELECTORS) $(ANOPE_SELECTORS)'
charybdis: charybdis:
$(PYTEST) $(PYTEST_ARGS) \ $(PYTEST) $(PYTEST_ARGS) \
@ -218,7 +154,7 @@ ergo:
hybrid: hybrid:
$(PYTEST) $(PYTEST_ARGS) \ $(PYTEST) $(PYTEST_ARGS) \
--controller irctest.controllers.hybrid \ --controller irctest.controllers.hybrid \
-m 'not services' \ --services-controller=irctest.controllers.anope_services \
-k "$(HYBRID_SELECTORS)" -k "$(HYBRID_SELECTORS)"
inspircd: inspircd:
@ -239,27 +175,34 @@ inspircd-anope:
--controller=irctest.controllers.inspircd \ --controller=irctest.controllers.inspircd \
--services-controller=irctest.controllers.anope_services \ --services-controller=irctest.controllers.anope_services \
-m 'services' \ -m 'services' \
-k '$(INSPIRCD_SELECTORS) $(ANOPE_SELECTORS)' -k '$(INSPIRCD_SELECTORS)'
ircu2: ircu2:
$(PYTEST) $(PYTEST_ARGS) \ $(PYTEST) $(PYTEST_ARGS) \
--controller=irctest.controllers.ircu2 \ --controller=irctest.controllers.ircu2 \
-m 'not services and not IRCv3' \ -m 'not services and not IRCv3' \
-n 10 \ -n 4 \
-k '$(IRCU2_SELECTORS)' -k '$(IRCU2_SELECTORS)'
nefarious:
$(PYTEST) $(PYTEST_ARGS) \
--controller=irctest.controllers.nefarious \
-m 'not services' \
-n 4 \
-k '$(NEFARIOUS_SELECTORS)'
snircd: snircd:
$(PYTEST) $(PYTEST_ARGS) \ $(PYTEST) $(PYTEST_ARGS) \
--controller=irctest.controllers.snircd \ --controller=irctest.controllers.snircd \
-m 'not services and not IRCv3' \ -m 'not services and not IRCv3' \
-n 10 \ -n 4 \
-k '$(SNIRCD_SELECTORS)' -k '$(SNIRCD_SELECTORS)'
irc2: irc2:
$(PYTEST) $(PYTEST_ARGS) \ $(PYTEST) $(PYTEST_ARGS) \
--controller=irctest.controllers.irc2 \ --controller=irctest.controllers.irc2 \
-m 'not services and not IRCv3' \ -m 'not services and not IRCv3' \
-n 10 \ -n 4 \
-k '$(IRC2_SELECTORS)' -k '$(IRC2_SELECTORS)'
limnoria: limnoria:
@ -275,14 +218,14 @@ mammon:
plexus4: plexus4:
$(PYTEST) $(PYTEST_ARGS) \ $(PYTEST) $(PYTEST_ARGS) \
--controller irctest.controllers.plexus4 \ --controller irctest.controllers.plexus4 \
-m 'not services' \ --services-controller=irctest.controllers.anope_services \
-k "$(PLEXUS4_SELECTORS)" -k "$(PLEXUS4_SELECTORS)"
ngircd: ngircd:
$(PYTEST) $(PYTEST_ARGS) \ $(PYTEST) $(PYTEST_ARGS) \
--controller irctest.controllers.ngircd \ --controller irctest.controllers.ngircd \
-m 'not services' \ -m 'not services' \
-n 10 \ -n 4 \
-k "$(NGIRCD_SELECTORS)" -k "$(NGIRCD_SELECTORS)"
ngircd-anope: ngircd-anope:
@ -316,6 +259,8 @@ unrealircd:
-m 'not services' \ -m 'not services' \
-k '$(UNREALIRCD_SELECTORS)' -k '$(UNREALIRCD_SELECTORS)'
unrealircd-5: unrealircd
unrealircd-atheme: unrealircd-atheme:
$(PYTEST) $(PYTEST_ARGS) \ $(PYTEST) $(PYTEST_ARGS) \
--controller=irctest.controllers.unrealircd \ --controller=irctest.controllers.unrealircd \
@ -328,4 +273,11 @@ unrealircd-anope:
--controller=irctest.controllers.unrealircd \ --controller=irctest.controllers.unrealircd \
--services-controller=irctest.controllers.anope_services \ --services-controller=irctest.controllers.anope_services \
-m 'services' \ -m 'services' \
-k '$(UNREALIRCD_SELECTORS) $(ANOPE_SELECTORS)' -k '$(UNREALIRCD_SELECTORS)'
unrealircd-dlk:
pifpaf run mysql -- $(PYTEST) $(PYTEST_ARGS) \
--controller=irctest.controllers.unrealircd \
--services-controller=irctest.controllers.dlk_services \
-m 'services' \
-k '$(UNREALIRCD_SELECTORS)'

View File

@ -18,11 +18,11 @@ have no side effect.
Install irctest and dependencies: Install irctest and dependencies:
``` ```
sudo apt install faketime # Optional, but greatly speeds up irctest/server_tests/list.py
cd ~ cd ~
git clone https://github.com/ProgVal/irctest.git git clone https://github.com/ProgVal/irctest.git
cd irctest cd irctest
pip3 install --user -r requirements.txt pip3 install --user -r requirements.txt
python3 setup.py install --user
``` ```
Add `~/.local/bin/` (and/or `~/go/bin/` for Ergo) Add `~/.local/bin/` (and/or `~/go/bin/` for Ergo)
@ -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

View File

@ -106,7 +106,13 @@ def pytest_collection_modifyitems(session, config, items):
assert isinstance(item, _pytest.python.Function) assert isinstance(item, _pytest.python.Function)
# unittest-style test functions have the node of UnitTest class as parent # unittest-style test functions have the node of UnitTest class as parent
assert isinstance(item.parent, _pytest.python.Instance) assert isinstance(
item.parent,
(
_pytest.python.Class, # pytest >= 7.0.0
_pytest.python.Instance, # pytest < 7.0.0
),
)
# and that node references the UnitTest class # and that node references the UnitTest class
assert issubclass(item.parent.cls, _IrcTestCase) assert issubclass(item.parent.cls, _IrcTestCase)

83
data/nefarious/ircd.pem Normal file
View File

@ -0,0 +1,83 @@
-----BEGIN PRIVATE KEY-----
MIIJRAIBADANBgkqhkiG9w0BAQEFAASCCS4wggkqAgEAAoICAQDT0URxi7/l7ZGe
tkPv9Yh8h2s9BpbAR4Wq8sakgqETWg/nE/JQM5dPxroVbtZWWQXuJEFsgBKbASLa
/eg5cyJv4Uu5WIZpG1LxdPEEIOSMWjzoAGwoLxbTRGrS7qNXsknB9RwDuq8lPQiK
kiAahg1Cn1vRrQ4cRrG+AkQWpRHJEDoLjCSo8IcAsKAZlw/eGtAcmeNvkr5AujEw
XjIwx2FoDyKaNGRH5Z7gLWvCKBNxQuJuMTzh8guLqdGbE4hH3rqyICbW5DGPaOZL
LErWuJ7kEhLZG2HDW5JaXOr0QfFYAA8pl9/qCuFMdoxRUKRcYBoxoMmz6dlsmipN
7vIj+TT6TemwcAT25pwMJIVS4WC4+BZilNH2eWKD9hZA8Kq7FDPu+1rxOJaLbE/b
vpK8jZeRdqFzE1eBCgPkw8D8V0r7J18d+DsmgOe2kRycaia/t9M4rhqe0FXjX1X1
lzQ52grxgc28Ejd1fGQXIJmdTh4BqKqTzxup0izS7dgFP1Ezm6Z4O+wklpL5uQF2
Ex4X6QEj76iCxH+J/01/cvbxMe3iuGXECbO/y1FIrg7FKzZSrQo4aP63lS7Y7aq0
t2t6kOS83ebhnpgHClgFs8/C3ayzYBBtbK63PYthwO8Rt6WamCIZFF5tA3XoI4Ak
fZcWD18loZai+QznVzbLNINf++rTwwIDAQABAoICAQCs1tT3piZHU2yAyo9zLbJa
kxGxcT/v1CzBSmtG8ATJyrKxRzhxszdj9HABbzjcqrXJFbKA+5yy+OFdOtSUlFtk
Wb21lwPOnmo29sp4KPL1h+itEzMuMwZ4DBry1aFZvPSsnPpoHJwwUbY3hHdHzVzi
oTCGTqT188Wzmxu+MqHppCEJLSj45ZPzvyxU1UwwW0a4H+ZTM7WlEYlzw1lHLlpQ
VBFTLS8q77aNjOKiQptiz0X+zpS0dhJvu3l7BhwtMRS8prmqnffG4r0QWCsVPP8C
cbEJkWtbwswQikN6Xpi1yw6UTQZ8brZa810aOSh07EJTfrU35rjxAndEspbJPd+4
Zz6fKNnRA7A4fFap2hF4TSP/UF12b4ulQ8FfcMMTFszWko5M6fWFuCeWfNbyXML5
fmn+NmSOboj7LkDqbpxtyaIVXXb2Y3F6A2fNllm/mxaGrRoEGNaH3o0qBgWRzJJB
TDSvIQtJddzL+iMaqxz4ufXAREJernZmPa3vlkVGLINNQUC9JLrB5eFjLzPiZN2U
8RgQ9YX5tjoJ+DtPWuMFDiCS1ZE20/UBOEYTeqIVuKdK3AjJDMFSjg8fRvsWRqZe
zsHv6tCtIFZFxYRxtrRGTUPQF+1QD6zBjYxZZk1B4n3uYBGVQFM/LnNHUxRnJBx1
PunD4ICOY97xd2hcPmGiCQKCAQEA8NCXYaHzhv6fg95H/iMuJVcOCKrJ5rVr4poG
SD0KZtS7SLzUYat8WcuoSubh5Eb2hHtrsnLrSOTnwQUO61f4gCRm2sEqHYsOAd7+
mNe1jfil0UBVqqL9GBcGYJkc5+DHgUlJQaxMV+4YLt8fD0KfZEnHaDAYX3kUdz+p
be//YAKv+JmxWcUdBF60AUWPjbCJT/1pfJeY8nEBFiYtlYKKN24+4OiRdJ2yRGzt
ZtNHaWy5EFF70yVgPX5MGQ7Z2JpejzK+lt+9nG4h1uJ4M2X4YrGVrRCn1W8jwqm/
bXest3E6wkkLoWDm9EaeYj00DUgMOviPyP4ckyxilG+Fny4JbwKCAQEA4SyUV03X
KEoL5sOD69sLu3TpnIQz73u9an9W/f2f7rsGwmCcR9RdYGV29ltwfBvOm0FnPQio
GliN+3PTWAL6bb8VYo2IR53VKpVHKSQUlzDOD9PGObXw1CT/+0zoMP7FBA4dTJDf
xQ63AQNpSCGdwbxZygPWzLV5O1WxMeXhnQRL1EBvMyJ52od0+HbajDXg5mNiBKNQ
AtVhB9pEu47Blu/KBqWjfh/GeBLPZB7MHmGNBYbBGGskKRLG4sIbwShs9cx8UM0/
9dxXkX2d8hjnSD/0ZBh54HHUaEvKAKfpz1L8AC0ll7biCAy0CZK23AoZu/KT8zJ+
qvz3AuJcW3lo7QKCAQEAzfGltNJawPUamBzNttKBUV+s2c6tkkdO92C/xKGnNp/x
dtg+TTTpyKV5zGy9fIsPoecnCFptS06vwAvCYZQ/Kd93stcFXHSiSwlY9H9tfffK
XzTEzoRLLIHsa0omRUufcrqpEqf2NjChr9wS5OsWAx9xkHGpNmUHEqB4FlPsM0C5
G0LdQCdplGYlTP0fMo5qL+VJhErle1kXE8kcrMMRzyvSTGe4lWGTph79vDUt2kQn
1IPLAJzzPEO5cqiXtzz1Z0N/aOn5b0FkYTAWmeY30LeMiJA46Df+/ihLVKPHKq6E
EMmFT8LeYMPQCbXLwRv/kaMm3D4tU9PejpD9Vk95swKCAQAtULBlxXeIVyaAAVba
L1H0Hroo0n41MtzSwt+568G05JSep5yr4/QKw0CmoY5Im7v/iLEDGmviKXIhaZTd
wHOvhGYEWGFVsFDG6hXRFL7EEoFVtBPPZ2sY9n1BkJ+lxI/XmhORZhJycNypaotU
hddets4HFrCyr86+/ybS2OWHmOa9x13Zl5WYQexrWFfxIaKqGtQOBOPEPjbxwp5U
dI1HF+i7X7hAWJqzbW2pQ31mm9EqjIztoho73diCp/e37q/G46kdBcFadEZ3NCWG
JDbfVmeTgU19usq5Vo9HhINMQvIOAwfuuVJRtmTBDHKaY7n8FfxqU/4j4RbA0Ncv
XYadAoIBAQC7yh4/UZCGhklUhhk/667OfchWvWGriCSaYGmQPdMzxjnIjAvvIUe9
riOTBSZXYYLmZHsmY/jK7KMGB3AsLTypSa9+ddAWqWn2dvOYyxNiAaSJK/RZfA9A
ocVfvvkhOfNAYIF+A+fyJ2pznsDkBf9tPkhN7kovl+mr/e25qZb1d09377770Pi7
thzEi+JLrRgYVLrCrPi2j4l7/Va/UaAPz+Dtu2GCT9vXgnhZtpb8R1kTViZFryTv
k+qbNYJzVm61Vit9mVAGe+WuzhlclJnN6LIZGG3zYHIulRAJrH1XDauHZfHzCKgi
FnMesy4thDMH/MhUfRtbylZTq45gtvCA
-----END PRIVATE KEY-----
-----BEGIN CERTIFICATE-----
MIIFazCCA1OgAwIBAgIUYHD08+9S32VTD9IEsr2Oe1dH3VEwDQYJKoZIhvcNAQEL
BQAwRTELMAkGA1UEBhMCQVUxEzARBgNVBAgMClNvbWUtU3RhdGUxITAfBgNVBAoM
GEludGVybmV0IFdpZGdpdHMgUHR5IEx0ZDAeFw0yMjA0MDQxODE2NTZaFw0yMzA0
MDQxODE2NTZaMEUxCzAJBgNVBAYTAkFVMRMwEQYDVQQIDApTb21lLVN0YXRlMSEw
HwYDVQQKDBhJbnRlcm5ldCBXaWRnaXRzIFB0eSBMdGQwggIiMA0GCSqGSIb3DQEB
AQUAA4ICDwAwggIKAoICAQDT0URxi7/l7ZGetkPv9Yh8h2s9BpbAR4Wq8sakgqET
Wg/nE/JQM5dPxroVbtZWWQXuJEFsgBKbASLa/eg5cyJv4Uu5WIZpG1LxdPEEIOSM
WjzoAGwoLxbTRGrS7qNXsknB9RwDuq8lPQiKkiAahg1Cn1vRrQ4cRrG+AkQWpRHJ
EDoLjCSo8IcAsKAZlw/eGtAcmeNvkr5AujEwXjIwx2FoDyKaNGRH5Z7gLWvCKBNx
QuJuMTzh8guLqdGbE4hH3rqyICbW5DGPaOZLLErWuJ7kEhLZG2HDW5JaXOr0QfFY
AA8pl9/qCuFMdoxRUKRcYBoxoMmz6dlsmipN7vIj+TT6TemwcAT25pwMJIVS4WC4
+BZilNH2eWKD9hZA8Kq7FDPu+1rxOJaLbE/bvpK8jZeRdqFzE1eBCgPkw8D8V0r7
J18d+DsmgOe2kRycaia/t9M4rhqe0FXjX1X1lzQ52grxgc28Ejd1fGQXIJmdTh4B
qKqTzxup0izS7dgFP1Ezm6Z4O+wklpL5uQF2Ex4X6QEj76iCxH+J/01/cvbxMe3i
uGXECbO/y1FIrg7FKzZSrQo4aP63lS7Y7aq0t2t6kOS83ebhnpgHClgFs8/C3ayz
YBBtbK63PYthwO8Rt6WamCIZFF5tA3XoI4AkfZcWD18loZai+QznVzbLNINf++rT
wwIDAQABo1MwUTAdBgNVHQ4EFgQU+9eHi2eqy0f3fDS0GjqkijGDDocwHwYDVR0j
BBgwFoAU+9eHi2eqy0f3fDS0GjqkijGDDocwDwYDVR0TAQH/BAUwAwEB/zANBgkq
hkiG9w0BAQsFAAOCAgEAAJXO3qUc/PW75pI2dt1cKv20VqozkfEf7P0eeVisCDxn
1p3QhVgI2lEe9kzdHp7t42g5xLkUhQEVmBcKm9xbl+k2D1X0+T8px1x6ZiWfbhXL
ptc/qCIXjPCgVN3s+Kasii3hHkZxKGZz/ySmBmfDJZjQZtbZzQWpvvX6SD4s7sjo
gWbZW3qvQ0bFTGdD1IjKYGaxK6aSrNkAIutiAX4RczJ1QSwb9Z2EIen+ABAvOZS9
xv3LiiidWcuOT7WzXEa4QvOslCEkAF+jj6mGYB7NWtly0kj4AEPvI4IoYTi9dohS
CA0zd1DTfjRwpAnT5P4sj4mpjLyRBumeeVGpCZhUxfKpFjIB2AnlgxrU+LPq5c9R
ZZ9Q5oeLxjRPjpqBeWwgnbjXstQCL9g0U7SsEemsv+zmvG5COhAmG5Wce/65ILlg
450H4bcn1ul0xvxz9hat6tqEZry3HcNE/CGDT+tXuhHpqOXkY1/c78C0QbWjWodR
tCvlXW00a+7TlEhNr4XBNdqtIQfYS9K9yiVVNfZLPEsN/SA3BGXmrr+du1/E4Ria
CkVpmBdJsVu5eMaUj1arsCqI4fwHzljtojJe/pCzZBVkOaSWQEQ+LL4iVnMas68m
qyshtNf4KNiM55OQmyTiFHMTIxCtdEcHaR3mUxR7GrIhc/bxyxUUBtMAuUX0Kjs=
-----END CERTIFICATE-----

View File

@ -2,6 +2,7 @@ from __future__ import annotations
import dataclasses import dataclasses
import os import os
from pathlib import Path
import shutil import shutil
import socket import socket
import subprocess import subprocess
@ -87,7 +88,7 @@ class DirectoryBasedController(_BaseController):
"""Helper for controllers whose software configuration is based on an """Helper for controllers whose software configuration is based on an
arbitrary directory.""" arbitrary directory."""
directory: Optional[str] directory: Optional[Path]
def __init__(self, test_config: TestCaseControllerConfig): def __init__(self, test_config: TestCaseControllerConfig):
super().__init__(test_config) super().__init__(test_config)
@ -110,22 +111,21 @@ class DirectoryBasedController(_BaseController):
"""Open a file in the configuration directory.""" """Open a file in the configuration directory."""
assert self.directory assert self.directory
if os.sep in name: if os.sep in name:
dir_ = os.path.join(self.directory, os.path.dirname(name)) dir_ = self.directory / os.path.dirname(name)
if not os.path.isdir(dir_): dir_.mkdir(parents=True, exist_ok=True)
os.makedirs(dir_) assert dir_.is_dir()
assert os.path.isdir(dir_) return (self.directory / name).open(mode)
return open(os.path.join(self.directory, name), mode)
def create_config(self) -> None: def create_config(self) -> None:
if not self.directory: if not self.directory:
self.directory = tempfile.mkdtemp() self.directory = Path(tempfile.mkdtemp())
def gen_ssl(self) -> None: def gen_ssl(self) -> None:
assert self.directory assert self.directory
self.csr_path = os.path.join(self.directory, "ssl.csr") self.csr_path = self.directory / "ssl.csr"
self.key_path = os.path.join(self.directory, "ssl.key") self.key_path = self.directory / "ssl.key"
self.pem_path = os.path.join(self.directory, "ssl.pem") self.pem_path = self.directory / "ssl.pem"
self.dh_path = os.path.join(self.directory, "dh.pem") self.dh_path = self.directory / "dh.pem"
subprocess.check_output( subprocess.check_output(
[ [
self.openssl_bin, self.openssl_bin,
@ -189,6 +189,10 @@ class BaseServerController(_BaseController):
"""Character used for the 'mute' extban""" """Character used for the 'mute' extban"""
nickserv = "NickServ" nickserv = "NickServ"
def __init__(self, *args: Any, **kwargs: Any):
super().__init__(*args, **kwargs)
self.faketime_enabled = False
def get_hostname_and_port(self) -> Tuple[str, int]: def get_hostname_and_port(self) -> Tuple[str, int]:
return find_hostname_and_port() return find_hostname_and_port()
@ -202,6 +206,7 @@ class BaseServerController(_BaseController):
run_services: bool, run_services: bool,
valid_metadata_keys: Optional[Set[str]], valid_metadata_keys: Optional[Set[str]],
invalid_metadata_keys: Optional[Set[str]], invalid_metadata_keys: Optional[Set[str]],
faketime: Optional[str],
) -> None: ) -> None:
raise NotImplementedError() raise NotImplementedError()
@ -217,6 +222,7 @@ class BaseServerController(_BaseController):
raise NotImplementedByController("account registration") raise NotImplementedByController("account registration")
def wait_for_port(self) -> None: def wait_for_port(self) -> None:
started_at = time.time()
while not self.port_open: while not self.port_open:
self.check_is_alive() self.check_is_alive()
time.sleep(self._port_wait_interval) time.sleep(self._port_wait_interval)
@ -239,11 +245,16 @@ class BaseServerController(_BaseController):
# ircu2 cuts the connection without a message if registration # ircu2 cuts the connection without a message if registration
# is not complete. # is not complete.
pass pass
except socket.timeout:
# irc2 just keeps it open
pass
c.close() c.close()
self.port_open = True self.port_open = True
except Exception: except ConnectionRefusedError:
continue if time.time() - started_at >= 60:
# waited for 60 seconds, giving up
raise
def wait_for_services(self) -> None: def wait_for_services(self) -> None:
assert self.services_controller assert self.services_controller
@ -290,10 +301,11 @@ class BaseServicesController(_BaseController):
c.sendLine("PONG :" + msg.params[0]) c.sendLine("PONG :" + msg.params[0])
c.getMessages() c.getMessages()
timeout = time.time() + 5 timeout = time.time() + 3
while True: while True:
c.sendLine(f"PRIVMSG {self.server_controller.nickserv} :HELP") c.sendLine(f"PRIVMSG {self.server_controller.nickserv} :help")
msgs = self.getNickServResponse(c)
msgs = self.getNickServResponse(c, timeout=1)
for msg in msgs: for msg in msgs:
if msg.command == "401": if msg.command == "401":
# NickServ not available yet # NickServ not available yet
@ -319,11 +331,12 @@ class BaseServicesController(_BaseController):
c.disconnect() c.disconnect()
self.services_up = True self.services_up = True
def getNickServResponse(self, client: Any) -> List[Message]: def getNickServResponse(self, client: Any, timeout: int = 0) -> List[Message]:
"""Wrapper aroung getMessages() that waits longer, because NickServ """Wrapper aroung getMessages() that waits longer, because NickServ
is queried asynchronously.""" is queried asynchronously."""
msgs: List[Message] = [] msgs: List[Message] = []
while not msgs: start_time = time.time()
while not msgs and (not timeout or start_time + timeout > time.time()):
time.sleep(0.05) time.sleep(0.05)
msgs = client.getMessages() msgs = client.getMessages()
return msgs return msgs

View File

@ -69,6 +69,30 @@ TController = TypeVar("TController", bound=basecontrollers._BaseController)
T = TypeVar("T") T = TypeVar("T")
def retry(f: TCallable) -> TCallable:
"""Retry the function if it raises ConnectionClosed; as a workaround for flaky
connection, such as::
1: connects to server.
1 -> S: NICK foo
1 -> S: USER username * * :Realname
S -> 1: :My.Little.Server NOTICE * :*** Found your hostname (cached)
S -> 1: :My.Little.Server NOTICE * :*** Checking Ident
S -> 1: :My.Little.Server NOTICE * :*** No Ident response
S -> 1: ERROR :Closing Link: cpu-pool.com (Use a different port)
"""
@functools.wraps(f)
def newf(*args, **kwargs): # type: ignore
try:
return f(*args, **kwargs)
except ConnectionClosed:
time.sleep(1)
return f(*args, **kwargs)
return newf # type: ignore
class ChannelJoinException(Exception): class ChannelJoinException(Exception):
def __init__(self, code: str, params: List[str]): def __init__(self, code: str, params: List[str]):
super().__init__(f"Failed to join channel ({code}): {params}") super().__init__(f"Failed to join channel ({code}): {params}")
@ -162,7 +186,7 @@ class _IrcTestCase(Generic[TController]):
msg=msg, msg=msg,
) )
if prefix and not patma.match_string(msg.prefix, prefix): if prefix is not None and not patma.match_string(msg.prefix, prefix):
fail_msg = ( fail_msg = (
fail_msg or "expected prefix to match {expects}, got {got}: {msg}" fail_msg or "expected prefix to match {expects}, got {got}: {msg}"
) )
@ -170,7 +194,7 @@ class _IrcTestCase(Generic[TController]):
*extra_format, got=msg.prefix, expects=prefix, msg=msg *extra_format, got=msg.prefix, expects=prefix, msg=msg
) )
if params and not patma.match_list(list(msg.params), params): if params is not None and not patma.match_list(list(msg.params), params):
fail_msg = ( fail_msg = (
fail_msg or "expected params to match {expects}, got {got}: {msg}" fail_msg or "expected params to match {expects}, got {got}: {msg}"
) )
@ -178,11 +202,11 @@ class _IrcTestCase(Generic[TController]):
*extra_format, got=msg.params, expects=params, msg=msg *extra_format, got=msg.params, expects=params, msg=msg
) )
if tags and not patma.match_dict(msg.tags, tags): if tags is not None and not patma.match_dict(msg.tags, tags):
fail_msg = fail_msg or "expected tags to match {expects}, got {got}: {msg}" fail_msg = fail_msg or "expected tags to match {expects}, got {got}: {msg}"
return fail_msg.format(*extra_format, got=msg.tags, expects=tags, msg=msg) return fail_msg.format(*extra_format, got=msg.tags, expects=tags, msg=msg)
if nick: if nick is not None:
got_nick = msg.prefix.split("!")[0] if msg.prefix else None got_nick = msg.prefix.split("!")[0] if msg.prefix else None
if nick != got_nick: if nick != got_nick:
fail_msg = ( fail_msg = (
@ -508,6 +532,12 @@ class BaseServerTestCase(
server_support: Optional[Dict[str, Optional[str]]] server_support: Optional[Dict[str, Optional[str]]]
run_services = False run_services = False
faketime: Optional[str] = None
"""If not None and the controller supports it and libfaketime is available,
runs the server using faketime and this value set as the $FAKETIME env variable.
Tests must check ``self.controller.faketime_enabled`` is True before
relying on this."""
__new__ = object.__new__ # pytest won't collect Generic[] subclasses otherwise __new__ = object.__new__ # pytest won't collect Generic[] subclasses otherwise
def setUp(self) -> None: def setUp(self) -> None:
@ -522,6 +552,7 @@ class BaseServerTestCase(
invalid_metadata_keys=self.invalid_metadata_keys, invalid_metadata_keys=self.invalid_metadata_keys,
ssl=self.ssl, ssl=self.ssl,
run_services=self.run_services, run_services=self.run_services,
faketime=self.faketime,
) )
self.clients: Dict[TClientName, client_mock.ClientMock] = {} self.clients: Dict[TClientName, client_mock.ClientMock] = {}
@ -539,13 +570,10 @@ class BaseServerTestCase(
if self.run_services: if self.run_services:
self.controller.wait_for_services() self.controller.wait_for_services()
if not name: if not name:
new_name: int = ( used_ids: List[int] = [
max( int(name) for name in self.clients if isinstance(name, (int, str))
[int(name) for name in self.clients if isinstance(name, (int, str))] ]
+ [0] new_name = max(used_ids + [0]) + 1
)
+ 1
)
name = cast(TClientName, new_name) name = cast(TClientName, new_name)
show_io = show_io if show_io is not None else self.show_io show_io = show_io if show_io is not None else self.show_io
self.clients[name] = client_mock.ClientMock(name=name, show_io=show_io) self.clients[name] = client_mock.ClientMock(name=name, show_io=show_io)
@ -647,6 +675,17 @@ class BaseServerTestCase(
else: else:
raise raise
def authenticateClient(
self, client: TClientName, account: str, password: str
) -> None:
self.sendLine(client, "AUTHENTICATE PLAIN")
m = self.getRegistrationMessage(client)
self.assertMessageMatch(m, command="AUTHENTICATE", params=["+"])
self.sendLine(client, sasl_plain_blob(account, password))
m = self.getRegistrationMessage(client)
self.assertIn(m.command, ["900", "903"], str(m))
@retry
def connectClient( def connectClient(
self, self,
nick: str, nick: str,
@ -665,17 +704,12 @@ class BaseServerTestCase(
client = self.addClient(name, show_io=show_io) client = self.addClient(name, show_io=show_io)
if capabilities: if capabilities:
self.sendLine(client, "CAP LS 302") self.sendLine(client, "CAP LS 302")
m = self.getRegistrationMessage(client) self.getCapLs(client)
self.requestCapabilities(client, capabilities, skip_if_cap_nak) self.requestCapabilities(client, capabilities, skip_if_cap_nak)
if password is not None: if password is not None:
if "sasl" not in (capabilities or ()): if "sasl" not in (capabilities or ()):
raise ValueError("Used 'password' option without sasl capbilitiy") raise ValueError("Used 'password' option without sasl capbilitiy")
self.sendLine(client, "AUTHENTICATE PLAIN") self.authenticateClient(client, account or nick, password)
m = self.getRegistrationMessage(client)
self.assertMessageMatch(m, command="AUTHENTICATE", params=["+"])
self.sendLine(client, sasl_plain_blob(account or nick, password))
m = self.getRegistrationMessage(client)
self.assertIn(m.command, ["900", "903"], str(m))
self.sendLine(client, "NICK {}".format(nick)) self.sendLine(client, "NICK {}".format(nick))
self.sendLine(client, "USER %s * * :Realname" % (ident,)) self.sendLine(client, "USER %s * * :Realname" % (ident,))
@ -700,6 +734,12 @@ class BaseServerTestCase(
self.server_support[param] = None self.server_support[param] = None
welcome.append(m) welcome.append(m)
self.targmax: Dict[str, Optional[str]] = dict(
item.split(":", 1) # type: ignore
for item in (self.server_support.get("TARGMAX") or "").split(",")
if item
)
return welcome return welcome
def joinClient(self, client: TClientName, channel: str) -> None: def joinClient(self, client: TClientName, channel: str) -> None:
@ -730,50 +770,55 @@ class BaseServerTestCase(
raise ChannelJoinException(msg.command, msg.params) raise ChannelJoinException(msg.command, msg.params)
_TSelf = TypeVar("_TSelf", bound="OptionalityHelper") _TSelf = TypeVar("_TSelf", bound="_IrcTestCase")
_TReturn = TypeVar("_TReturn") _TReturn = TypeVar("_TReturn")
class OptionalityHelper(Generic[TController]): def skipUnlessHasMechanism(
controller: TController mech: str,
) -> Callable[[Callable[..., _TReturn]], Callable[..., _TReturn]]:
def checkSaslSupport(self) -> None: # Just a function returning a function that takes functions and
if self.controller.supported_sasl_mechanisms: # returns functions, nothing to see here.
return # If Python didn't have such an awful syntax for callables, it would be:
raise runner.NotImplementedByController("SASL") # str -> ((TSelf -> TReturn) -> (TSelf -> TReturn))
def decorator(f: Callable[..., _TReturn]) -> Callable[..., _TReturn]:
def checkMechanismSupport(self, mechanism: str) -> None:
if mechanism in self.controller.supported_sasl_mechanisms:
return
raise runner.OptionalSaslMechanismNotSupported(mechanism)
@staticmethod
def skipUnlessHasMechanism(
mech: str,
) -> Callable[[Callable[..., _TReturn]], Callable[..., _TReturn]]:
# Just a function returning a function that takes functions and
# returns functions, nothing to see here.
# If Python didn't have such an awful syntax for callables, it would be:
# str -> ((TSelf -> TReturn) -> (TSelf -> TReturn))
def decorator(f: Callable[..., _TReturn]) -> Callable[..., _TReturn]:
@functools.wraps(f)
def newf(self: _TSelf, *args: Any, **kwargs: Any) -> _TReturn:
self.checkMechanismSupport(mech)
return f(self, *args, **kwargs)
return newf
return decorator
@staticmethod
def skipUnlessHasSasl(f: Callable[..., _TReturn]) -> Callable[..., _TReturn]:
@functools.wraps(f) @functools.wraps(f)
def newf(self: _TSelf, *args: Any, **kwargs: Any) -> _TReturn: def newf(self: _TSelf, *args: Any, **kwargs: Any) -> _TReturn:
self.checkSaslSupport() if mech not in self.controller.supported_sasl_mechanisms:
raise runner.OptionalSaslMechanismNotSupported(mech)
return f(self, *args, **kwargs) return f(self, *args, **kwargs)
return newf return newf
return decorator
def xfailIf(
condition: Callable[..., bool], reason: str
) -> Callable[[Callable[..., _TReturn]], Callable[..., _TReturn]]:
# Works about the same as skipUnlessHasMechanism
def decorator(f: Callable[..., _TReturn]) -> Callable[..., _TReturn]:
@functools.wraps(f)
def newf(self: _TSelf, *args: Any, **kwargs: Any) -> _TReturn:
if condition(self):
try:
return f(self, *args, **kwargs)
except Exception:
pytest.xfail(reason)
assert False # make mypy happy
else:
return f(self, *args, **kwargs)
return newf
return decorator
def xfailIfSoftware(
names: List[str], reason: str
) -> Callable[[Callable[..., _TReturn]], Callable[..., _TReturn]]:
return xfailIf(lambda testcase: testcase.controller.software_name in names, reason)
def mark_services(cls: TClass) -> TClass: def mark_services(cls: TClass) -> TClass:
cls.run_services = True cls.run_services = True

View File

@ -1,3 +1,5 @@
"""Format of ``CAP LS`` sent by IRCv3 clients."""
from irctest import cases from irctest import cases
from irctest.irc_utils.message_parser import Message from irctest.irc_utils.message_parser import Message

View File

@ -1,3 +1,8 @@
"""SASL authentication from clients, for all known mechanisms.
For now, only `SASLv3.1 <https://ircv3.net/specs/extensions/sasl-3.1>`_
is tested, not `SASLv3.2 <https://ircv3.net/specs/extensions/sasl-3.2>`_."""
import base64 import base64
import pytest import pytest
@ -34,8 +39,8 @@ class IdentityHash:
return self._data return self._data
class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper): class SaslTestCase(cases.BaseClientTestCase):
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
def testPlain(self): def testPlain(self):
"""Test PLAIN authentication with correct username/password.""" """Test PLAIN authentication with correct username/password."""
auth = authentication.Authentication( auth = authentication.Authentication(
@ -55,7 +60,8 @@ class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
m = self.negotiateCapabilities(["sasl"], False) m = self.negotiateCapabilities(["sasl"], False)
self.assertEqual(m, Message({}, None, "CAP", ["END"])) self.assertEqual(m, Message({}, None, "CAP", ["END"]))
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
@cases.xfailIfSoftware(["Sopel"], "Sopel requests SASL PLAIN even if not available")
def testPlainNotAvailable(self): def testPlainNotAvailable(self):
"""`sasl=EXTERNAL` is advertized, whereas the client is configured """`sasl=EXTERNAL` is advertized, whereas the client is configured
to use PLAIN. to use PLAIN.
@ -84,8 +90,9 @@ class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
m = self.getMessage() m = self.getMessage()
self.assertMessageMatch(m, command="CAP") self.assertMessageMatch(m, command="CAP")
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @pytest.mark.parametrize("pattern", ["barbaz", "éèà"])
def testPlainLarge(self): @cases.skipUnlessHasMechanism("PLAIN")
def testPlainLarge(self, pattern):
"""Test the client splits large AUTHENTICATE messages whose payload """Test the client splits large AUTHENTICATE messages whose payload
is not a multiple of 400. is not a multiple of 400.
<http://ircv3.net/specs/extensions/sasl-3.1.html#the-authenticate-command> <http://ircv3.net/specs/extensions/sasl-3.1.html#the-authenticate-command>
@ -94,10 +101,10 @@ class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
auth = authentication.Authentication( auth = authentication.Authentication(
mechanisms=[authentication.Mechanisms.plain], mechanisms=[authentication.Mechanisms.plain],
username="foo", username="foo",
password="bar" * 200, password=pattern * 100,
) )
authstring = base64.b64encode( authstring = base64.b64encode(
b"\x00".join([b"foo", b"foo", b"bar" * 200]) b"\x00".join([b"foo", b"foo", pattern.encode() * 100])
).decode() ).decode()
m = self.negotiateCapabilities(["sasl"], auth=auth) m = self.negotiateCapabilities(["sasl"], auth=auth)
self.assertEqual(m, Message({}, None, "AUTHENTICATE", ["PLAIN"])) self.assertEqual(m, Message({}, None, "AUTHENTICATE", ["PLAIN"]))
@ -113,8 +120,9 @@ class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
m = self.negotiateCapabilities(["sasl"], False) m = self.negotiateCapabilities(["sasl"], False)
self.assertEqual(m, Message({}, None, "CAP", ["END"])) self.assertEqual(m, Message({}, None, "CAP", ["END"]))
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
def testPlainLargeMultiple(self): @pytest.mark.parametrize("pattern", ["quux", "éè"])
def testPlainLargeMultiple(self, pattern):
"""Test the client splits large AUTHENTICATE messages whose payload """Test the client splits large AUTHENTICATE messages whose payload
is a multiple of 400. is a multiple of 400.
<http://ircv3.net/specs/extensions/sasl-3.1.html#the-authenticate-command> <http://ircv3.net/specs/extensions/sasl-3.1.html#the-authenticate-command>
@ -123,10 +131,10 @@ class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
auth = authentication.Authentication( auth = authentication.Authentication(
mechanisms=[authentication.Mechanisms.plain], mechanisms=[authentication.Mechanisms.plain],
username="foo", username="foo",
password="quux" * 148, password=pattern * 148,
) )
authstring = base64.b64encode( authstring = base64.b64encode(
b"\x00".join([b"foo", b"foo", b"quux" * 148]) b"\x00".join([b"foo", b"foo", pattern.encode() * 148])
).decode() ).decode()
m = self.negotiateCapabilities(["sasl"], auth=auth) m = self.negotiateCapabilities(["sasl"], auth=auth)
self.assertEqual(m, Message({}, None, "AUTHENTICATE", ["PLAIN"])) self.assertEqual(m, Message({}, None, "AUTHENTICATE", ["PLAIN"]))
@ -143,7 +151,7 @@ class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
self.assertEqual(m, Message({}, None, "CAP", ["END"])) self.assertEqual(m, Message({}, None, "CAP", ["END"]))
@pytest.mark.skipif(ecdsa is None, reason="python3-ecdsa is not available") @pytest.mark.skipif(ecdsa is None, reason="python3-ecdsa is not available")
@cases.OptionalityHelper.skipUnlessHasMechanism("ECDSA-NIST256P-CHALLENGE") @cases.skipUnlessHasMechanism("ECDSA-NIST256P-CHALLENGE")
def testEcdsa(self): def testEcdsa(self):
"""Test ECDSA authentication.""" """Test ECDSA authentication."""
auth = authentication.Authentication( auth = authentication.Authentication(
@ -177,7 +185,7 @@ class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
m = self.negotiateCapabilities(["sasl"], False) m = self.negotiateCapabilities(["sasl"], False)
self.assertEqual(m, Message({}, None, "CAP", ["END"])) self.assertEqual(m, Message({}, None, "CAP", ["END"]))
@cases.OptionalityHelper.skipUnlessHasMechanism("SCRAM-SHA-256") @cases.skipUnlessHasMechanism("SCRAM-SHA-256")
def testScram(self): def testScram(self):
"""Test SCRAM-SHA-256 authentication.""" """Test SCRAM-SHA-256 authentication."""
auth = authentication.Authentication( auth = authentication.Authentication(
@ -219,7 +227,7 @@ class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
self.assertEqual(m.command, "AUTHENTICATE", m) self.assertEqual(m.command, "AUTHENTICATE", m)
self.assertEqual(m.params, ["+"], m) self.assertEqual(m.params, ["+"], m)
@cases.OptionalityHelper.skipUnlessHasMechanism("SCRAM-SHA-256") @cases.skipUnlessHasMechanism("SCRAM-SHA-256")
def testScramBadPassword(self): def testScramBadPassword(self):
"""Test SCRAM-SHA-256 authentication with a bad password.""" """Test SCRAM-SHA-256 authentication with a bad password."""
auth = authentication.Authentication( auth = authentication.Authentication(
@ -254,8 +262,8 @@ class SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
authenticator.response(msg) authenticator.response(msg)
class Irc302SaslTestCase(cases.BaseClientTestCase, cases.OptionalityHelper): class Irc302SaslTestCase(cases.BaseClientTestCase):
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
def testPlainNotAvailable(self): def testPlainNotAvailable(self):
"""Test the client does not try to authenticate using a mechanism the """Test the client does not try to authenticate using a mechanism the
server does not advertise. server does not advertise.

View File

@ -1,6 +1,10 @@
"""Clients should validate certificates; either with a CA or fingerprints."""
import socket import socket
import ssl import ssl
import pytest
from irctest import cases, runner, tls from irctest import cases, runner, tls
from irctest.exceptions import ConnectionClosed from irctest.exceptions import ConnectionClosed
from irctest.patma import ANYSTR from irctest.patma import ANYSTR
@ -136,7 +140,7 @@ class TlsTestCase(cases.BaseClientTestCase):
self.getMessage() self.getMessage()
class StsTestCase(cases.BaseClientTestCase, cases.OptionalityHelper): class StsTestCase(cases.BaseClientTestCase):
def setUp(self): def setUp(self):
super().setUp() super().setUp()
self.insecure_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM) self.insecure_server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
@ -148,7 +152,8 @@ class StsTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
super().tearDown() super().tearDown()
@cases.mark_capabilities("sts") @cases.mark_capabilities("sts")
def testSts(self): @pytest.mark.parametrize("portOnSecure", [False, True])
def testSts(self, portOnSecure):
if not self.controller.supports_sts: if not self.controller.supports_sts:
raise runner.CapabilityNotSupported("sts") raise runner.CapabilityNotSupported("sts")
tls_config = tls.TlsConfig( tls_config = tls.TlsConfig(
@ -176,10 +181,12 @@ class StsTestCase(cases.BaseClientTestCase, cases.OptionalityHelper):
# and reconnect securely on the stated port." # and reconnect securely on the stated port."
self.acceptClient(tls_cert=GOOD_CERT, tls_key=GOOD_KEY) self.acceptClient(tls_cert=GOOD_CERT, tls_key=GOOD_KEY)
# Send the STS policy, over secure connection this time # Send the STS policy, over secure connection this time.
self.sendLine( if portOnSecure:
"CAP * LS :sts=duration=10,port={}".format(self.server.getsockname()[1]) # Should be ignored
) self.sendLine("CAP * LS :sts=duration=10,port=12345")
else:
self.sendLine("CAP * LS :sts=duration=10")
# Make the client reconnect. It should reconnect to the secure server. # Make the client reconnect. It should reconnect to the secure server.
self.sendLine("ERROR :closing link") self.sendLine("ERROR :closing link")

View File

@ -1,4 +1,4 @@
import os from pathlib import Path
import shutil import shutil
import subprocess import subprocess
from typing import Type from typing import Type
@ -73,6 +73,8 @@ module {{ name = "ns_cert" }}
class AnopeController(BaseServicesController, DirectoryBasedController): class AnopeController(BaseServicesController, DirectoryBasedController):
"""Collaborator for server controllers that rely on Anope""" """Collaborator for server controllers that rely on Anope"""
software_name = "Anope"
def run(self, protocol: str, server_hostname: str, server_port: int) -> None: def run(self, protocol: str, server_hostname: str, server_port: int) -> None:
self.create_config() self.create_config()
@ -99,14 +101,11 @@ class AnopeController(BaseServicesController, DirectoryBasedController):
pass pass
assert self.directory assert self.directory
services_path = shutil.which("services")
assert services_path
# Config and code need to be in the same directory, *obviously* # Config and code need to be in the same directory, *obviously*
os.symlink( (self.directory / "lib").symlink_to(Path(services_path).parent.parent / "lib")
os.path.join(
os.path.dirname(shutil.which("services")), "..", "lib" # type: ignore
),
os.path.join(self.directory, "lib"),
)
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
[ [

View File

@ -1,4 +1,3 @@
import os
import subprocess import subprocess
from typing import Optional, Type from typing import Optional, Type
@ -56,6 +55,8 @@ saslserv {{
class AthemeController(BaseServicesController, DirectoryBasedController): class AthemeController(BaseServicesController, DirectoryBasedController):
"""Mixin for server controllers that rely on Atheme""" """Mixin for server controllers that rely on Atheme"""
software_name = "Atheme"
def run(self, protocol: str, server_hostname: str, server_port: int) -> None: def run(self, protocol: str, server_hostname: str, server_port: int) -> None:
self.create_config() self.create_config()
@ -79,11 +80,11 @@ class AthemeController(BaseServicesController, DirectoryBasedController):
"atheme-services", "atheme-services",
"-n", # don't fork "-n", # don't fork
"-c", "-c",
os.path.join(self.directory, "services.conf"), self.directory / "services.conf",
"-l", "-l",
f"/tmp/services-{server_port}.log", f"/tmp/services-{server_port}.log",
"-p", "-p",
os.path.join(self.directory, "services.pid"), self.directory / "services.pid",
"-D", "-D",
self.directory, self.directory,
], ],

View File

@ -1,4 +1,4 @@
import os from pathlib import Path
import shutil import shutil
import subprocess import subprocess
from typing import Optional, Set, Type from typing import Optional, Set, Type
@ -80,6 +80,19 @@ oper {{
""" """
def initialize_entropy(directory: Path) -> None:
# https://github.com/DALnet/bahamut/blob/7fc039d403f66a954225c5dc4ad1fe683aedd794/include/dh.h#L35-L38
nb_rand_bytes = 512 // 8
# https://github.com/DALnet/bahamut/blob/7fc039d403f66a954225c5dc4ad1fe683aedd794/src/dh.c#L186
entropy_file_size = nb_rand_bytes * 4
# Not actually random; but we don't care.
entropy = b"\x00" * entropy_file_size
with (directory / ".ircd.entropy").open("wb") as fd:
fd.write(entropy)
class BahamutController(BaseServerController, DirectoryBasedController): class BahamutController(BaseServerController, DirectoryBasedController):
software_name = "Bahamut" software_name = "Bahamut"
supported_sasl_mechanisms: Set[str] = set() supported_sasl_mechanisms: Set[str] = set()
@ -102,6 +115,7 @@ class BahamutController(BaseServerController, DirectoryBasedController):
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
restricted_metadata_keys: Optional[Set[str]] = None, restricted_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
) -> None: ) -> None:
if valid_metadata_keys or invalid_metadata_keys: if valid_metadata_keys or invalid_metadata_keys:
raise NotImplementedByController( raise NotImplementedByController(
@ -120,9 +134,14 @@ class BahamutController(BaseServerController, DirectoryBasedController):
assert self.directory assert self.directory
# Bahamut reads some bytes from /dev/urandom on startup, which causes
# GitHub Actions to sometimes freeze and timeout.
# This initializes the entropy file so Bahamut does not need to do it itself.
initialize_entropy(self.directory)
# they are hardcoded... thankfully Bahamut reads them from the CWD. # they are hardcoded... thankfully Bahamut reads them from the CWD.
shutil.copy(self.pem_path, os.path.join(self.directory, "ircd.crt")) shutil.copy(self.pem_path, self.directory / "ircd.crt")
shutil.copy(self.key_path, os.path.join(self.directory, "ircd.key")) shutil.copy(self.key_path, self.directory / "ircd.key")
with self.open_file("server.conf") as fd: with self.open_file("server.conf") as fd:
fd.write( fd.write(
@ -136,15 +155,21 @@ class BahamutController(BaseServerController, DirectoryBasedController):
# pem_path=self.pem_path, # pem_path=self.pem_path,
) )
) )
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
[ [
# "strace", "-f", "-e", "file", *faketime_cmd,
"ircd", "ircd",
"-t", # don't fork "-t", # don't fork
"-f", "-f",
os.path.join(self.directory, "server.conf"), self.directory / "server.conf",
], ],
# stdout=subprocess.DEVNULL,
) )
if run_services: if run_services:

View File

@ -1,4 +1,4 @@
import os import shutil
import subprocess import subprocess
from typing import Optional, Set from typing import Optional, Set
@ -43,6 +43,7 @@ class BaseHybridController(BaseServerController, DirectoryBasedController):
run_services: bool, run_services: bool,
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
) -> None: ) -> None:
if valid_metadata_keys or invalid_metadata_keys: if valid_metadata_keys or invalid_metadata_keys:
raise NotImplementedByController( raise NotImplementedByController(
@ -73,14 +74,22 @@ class BaseHybridController(BaseServerController, DirectoryBasedController):
) )
) )
assert self.directory assert self.directory
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
[ [
*faketime_cmd,
self.binary_name, self.binary_name,
"-foreground", "-foreground",
"-configfile", "-configfile",
os.path.join(self.directory, "server.conf"), self.directory / "server.conf",
"-pidfile", "-pidfile",
os.path.join(self.directory, "server.pid"), self.directory / "server.pid",
], ],
# stderr=subprocess.DEVNULL, # stderr=subprocess.DEVNULL,
) )

View File

@ -12,6 +12,9 @@ serverinfo {{
general {{ general {{
throttle_count = 100; # We need to connect lots of clients quickly throttle_count = 100; # We need to connect lots of clients quickly
# disable throttling for LIST and similar:
pace_wait_simple = 0 second;
pace_wait = 0 second;
sasl_service = "SaslServ"; sasl_service = "SaslServ";
}}; }};

View File

@ -0,0 +1,245 @@
import os
from pathlib import Path
import secrets
import subprocess
from typing import Optional, Type
import irctest
from irctest.basecontrollers import BaseServicesController, DirectoryBasedController
import irctest.cases
import irctest.runner
TEMPLATE_DLK_CONFIG = """\
info {{
SID "00A";
network-name "testnetwork";
services-name "services.example.org";
admin-email "admin@example.org";
}}
link {{
hostname "{server_hostname}";
port "{server_port}";
password "password";
}}
log {{
debug "yes";
}}
sql {{
port "3306";
username "pifpaf";
password "pifpaf";
database "pifpaf";
sockfile "{mysql_socket}";
prefix "{dlk_prefix}";
}}
wordpress {{
prefix "{wp_prefix}";
}}
"""
TEMPLATE_DLK_WP_CONFIG = """
<?php
global $wpconfig;
$wpconfig = [
"dbprefix" => "{wp_prefix}",
"default_avatar" => "https://valware.uk/wp-content/plugins/ultimate-member/assets/img/default_avatar.jpg",
"forumschan" => "#DLK-Support",
];
"""
TEMPLATE_WP_CONFIG = """
define( 'DB_NAME', 'pifpaf' );
define( 'DB_USER', 'pifpaf' );
define( 'DB_PASSWORD', 'pifpaf' );
define( 'DB_HOST', 'localhost:{mysql_socket}' );
define( 'DB_CHARSET', 'utf8' );
define( 'DB_COLLATE', '' );
define( 'AUTH_KEY', 'put your unique phrase here' );
define( 'SECURE_AUTH_KEY', 'put your unique phrase here' );
define( 'LOGGED_IN_KEY', 'put your unique phrase here' );
define( 'NONCE_KEY', 'put your unique phrase here' );
define( 'AUTH_SALT', 'put your unique phrase here' );
define( 'SECURE_AUTH_SALT', 'put your unique phrase here' );
define( 'LOGGED_IN_SALT', 'put your unique phrase here' );
define( 'NONCE_SALT', 'put your unique phrase here' );
$table_prefix = '{wp_prefix}';
define( 'WP_DEBUG', false );
if (!defined('ABSPATH')) {{
define( 'ABSPATH', '{wp_path}' );
}}
/* That's all, stop editing! Happy publishing. */
/** Absolute path to the WordPress directory. */
/** Sets up WordPress vars and included files. */
require_once ABSPATH . 'wp-settings.php';
"""
class DlkController(BaseServicesController, DirectoryBasedController):
"""Mixin for server controllers that rely on DLK"""
software_name = "Dlk-Services"
def run_sql(self, sql: str) -> None:
mysql_socket = os.environ["PIFPAF_MYSQL_SOCKET"]
subprocess.run(
["mysql", "-S", mysql_socket, "pifpaf"],
input=sql.encode(),
check=True,
)
def run(self, protocol: str, server_hostname: str, server_port: int) -> None:
self.create_config()
if protocol == "unreal4":
protocol = "unreal5"
assert protocol in ("unreal5",), protocol
mysql_socket = os.environ["PIFPAF_MYSQL_SOCKET"]
assert self.directory
try:
self.wp_cli_path = Path(os.environ["IRCTEST_WP_CLI_PATH"])
if not self.wp_cli_path.is_file():
raise KeyError()
except KeyError:
raise RuntimeError(
"$IRCTEST_WP_CLI_PATH must be set to a WP-CLI executable (eg. "
"downloaded from <https://raw.githubusercontent.com/wp-cli/builds/"
"gh-pages/phar/wp-cli.phar>)"
) from None
try:
self.dlk_path = Path(os.environ["IRCTEST_DLK_PATH"])
if not self.dlk_path.is_dir():
raise KeyError()
except KeyError:
raise RuntimeError("$IRCTEST_DLK_PATH is not set") from None
self.dlk_path = self.dlk_path.resolve()
# Unpack a fresh Wordpress install in the temporary directory.
# In theory we could have a common Wordpress install and only wp-config.php
# in the temporary directory; but wp-cli assumes wp-config.php must be
# in a Wordpress directory, and fails in various places if it isn't.
# Rather than symlinking everything to make it work, let's just copy
# the whole code, it's not that big.
try:
wp_zip_path = Path(os.environ["IRCTEST_WP_ZIP_PATH"])
if not wp_zip_path.is_file():
raise KeyError()
except KeyError:
raise RuntimeError(
"$IRCTEST_WP_ZIP_PATH must be set to a Wordpress source zipball "
"(eg. downloaded from <https://wordpress.org/latest.zip>)"
) from None
subprocess.run(
["unzip", wp_zip_path, "-d", self.directory], stdout=subprocess.DEVNULL
)
self.wp_path = self.directory / "wordpress"
rand_hex = secrets.token_hex(6)
self.wp_prefix = f"wp{rand_hex}_"
self.dlk_prefix = f"dlk{rand_hex}_"
template_vars = dict(
protocol=protocol,
server_hostname=server_hostname,
server_port=server_port,
mysql_socket=mysql_socket,
wp_path=self.wp_path,
wp_prefix=self.wp_prefix,
dlk_prefix=self.dlk_prefix,
)
# Configure Wordpress
wp_config_path = self.directory / "wp-config.php"
with open(wp_config_path, "w") as fd:
fd.write(TEMPLATE_WP_CONFIG.format(**template_vars))
subprocess.run(
[
"php",
self.wp_cli_path,
"core",
"install",
"--url=http://localhost/",
"--title=irctest site",
"--admin_user=adminuser",
"--admin_email=adminuser@example.org",
f"--path={self.wp_path}",
],
check=True,
)
# Configure Dlk
dlk_log_dir = self.directory / "logs"
dlk_conf_dir = self.directory / "conf"
dlk_conf_path = dlk_conf_dir / "dalek.conf"
os.mkdir(dlk_conf_dir)
with open(dlk_conf_path, "w") as fd:
fd.write(TEMPLATE_DLK_CONFIG.format(**template_vars))
dlk_wp_config_path = dlk_conf_dir / "wordpress.conf"
with open(dlk_wp_config_path, "w") as fd:
fd.write(TEMPLATE_DLK_WP_CONFIG.format(**template_vars))
(dlk_conf_dir / "modules.conf").symlink_to(self.dlk_path / "conf/modules.conf")
self.proc = subprocess.Popen(
[
"php",
"src/dalek",
],
cwd=self.dlk_path,
env={
**os.environ,
"DALEK_CONF_DIR": str(dlk_conf_dir),
"DALEK_LOG_DIR": str(dlk_log_dir),
},
)
def terminate(self) -> None:
super().terminate()
def kill(self) -> None:
super().kill()
def registerUser(
self,
case: irctest.cases.BaseServerTestCase,
username: str,
password: Optional[str] = None,
) -> None:
assert password
subprocess.run(
[
"php",
self.wp_cli_path,
"user",
"create",
username,
f"{username}@example.org",
f"--user_pass={password}",
f"--path={self.wp_path}",
],
check=True,
)
def get_irctest_controller_class() -> Type[DlkController]:
return DlkController

View File

@ -1,6 +1,7 @@
import copy import copy
import json import json
import os import os
import shutil
import subprocess import subprocess
from typing import Any, Dict, Optional, Set, Type, Union from typing import Any, Dict, Optional, Set, Type, Union
@ -155,6 +156,7 @@ class ErgoController(BaseServerController, DirectoryBasedController):
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
restricted_metadata_keys: Optional[Set[str]] = None, restricted_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
config: Optional[Any] = None, config: Optional[Any] = None,
) -> None: ) -> None:
if valid_metadata_keys or invalid_metadata_keys: if valid_metadata_keys or invalid_metadata_keys:
@ -183,27 +185,32 @@ class ErgoController(BaseServerController, DirectoryBasedController):
bind_address = "127.0.0.1:%s" % (port,) bind_address = "127.0.0.1:%s" % (port,)
listener_conf = None # plaintext listener_conf = None # plaintext
if ssl: if ssl:
self.key_path = os.path.join(self.directory, "ssl.key") self.key_path = self.directory / "ssl.key"
self.pem_path = os.path.join(self.directory, "ssl.pem") self.pem_path = self.directory / "ssl.pem"
listener_conf = {"tls": {"cert": self.pem_path, "key": self.key_path}} listener_conf = {"tls": {"cert": self.pem_path, "key": self.key_path}}
config["server"]["listeners"][bind_address] = listener_conf # type: ignore config["server"]["listeners"][bind_address] = listener_conf # type: ignore
config["datastore"]["path"] = os.path.join( # type: ignore config["datastore"]["path"] = str(self.directory / "ircd.db") # type: ignore
self.directory, "ircd.db"
)
if password is not None: if password is not None:
config["server"]["password"] = hash_password(password) # type: ignore config["server"]["password"] = hash_password(password) # type: ignore
assert self.proc is None assert self.proc is None
self._config_path = os.path.join(self.directory, "server.yml") self._config_path = self.directory / "server.yml"
self._config = config self._config = config
self._write_config() self._write_config()
subprocess.call(["ergo", "initdb", "--conf", self._config_path, "--quiet"]) subprocess.call(["ergo", "initdb", "--conf", self._config_path, "--quiet"])
subprocess.call(["ergo", "mkcerts", "--conf", self._config_path, "--quiet"]) subprocess.call(["ergo", "mkcerts", "--conf", self._config_path, "--quiet"])
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
["ergo", "run", "--conf", self._config_path, "--quiet"] [*faketime_cmd, "ergo", "run", "--conf", self._config_path, "--quiet"]
) )
def wait_for_services(self) -> None: def wait_for_services(self) -> None:

View File

@ -42,6 +42,7 @@ class ExternalServerController(BaseServerController):
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
restricted_metadata_keys: Optional[Set[str]] = None, restricted_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
) -> None: ) -> None:
pass pass

View File

@ -40,7 +40,7 @@ class {{
}}; }};
connect {{ connect {{
name = "services.example.org"; name = "services.example.org";
host = "localhost"; # Used to validate incoming connection host = "127.0.0.1"; # Used to validate incoming connection
port = 0; # We don't need servers to connect to services port = 0; # We don't need servers to connect to services
send_password = "password"; send_password = "password";
accept_password = "password"; accept_password = "password";

View File

@ -1,4 +1,4 @@
import os import shutil
import subprocess import subprocess
from typing import Optional, Set, Type from typing import Optional, Set, Type
@ -17,6 +17,8 @@ TEMPLATE_CONFIG = """
resolvehostnames="no" # Faster resolvehostnames="no" # Faster
recvq="40960" # Needs to be larger than a valid message with tags recvq="40960" # Needs to be larger than a valid message with tags
timeout="10" # So tests don't hang too long timeout="10" # So tests don't hang too long
localmax="1000"
globalmax="1000"
{password_field}> {password_field}>
<class <class
@ -57,8 +59,10 @@ TEMPLATE_CONFIG = """
target="services.example.org"> target="services.example.org">
# Protocol: # Protocol:
<module name="banexception">
<module name="botmode"> <module name="botmode">
<module name="cap"> <module name="cap">
<module name="inviteexception">
<module name="ircv3"> <module name="ircv3">
<module name="ircv3_accounttag"> <module name="ircv3_accounttag">
<module name="ircv3_batch"> <module name="ircv3_batch">
@ -74,9 +78,14 @@ TEMPLATE_CONFIG = """
<module name="namesx"> # For multi-prefix <module name="namesx"> # For multi-prefix
<module name="sasl"> <module name="sasl">
# HELP/HELPOP
<module name="alias"> # for the HELP alias
<module name="helpop">
<include file="examples/helpop.conf.example">
# Misc: # Misc:
<log method="file" type="*" level="debug" target="/tmp/ircd-{port}.log"> <log method="file" type="*" level="debug" target="/tmp/ircd-{port}.log">
<server name="My.Little.Server" description="testnet" id="000" network="testnet"> <server name="My.Little.Server" description="test server" id="000" network="testnet">
""" """
TEMPLATE_SSL_CONFIG = """ TEMPLATE_SSL_CONFIG = """
@ -107,6 +116,7 @@ class InspircdController(BaseServerController, DirectoryBasedController):
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
restricted_metadata_keys: Optional[Set[str]] = None, restricted_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str] = None,
) -> None: ) -> None:
if valid_metadata_keys or invalid_metadata_keys: if valid_metadata_keys or invalid_metadata_keys:
raise NotImplementedByController( raise NotImplementedByController(
@ -140,12 +150,20 @@ class InspircdController(BaseServerController, DirectoryBasedController):
) )
) )
assert self.directory assert self.directory
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
[ [
*faketime_cmd,
"inspircd", "inspircd",
"--nofork", "--nofork",
"--config", "--config",
os.path.join(self.directory, "server.conf"), self.directory / "server.conf",
], ],
stdout=subprocess.DEVNULL, stdout=subprocess.DEVNULL,
) )

View File

@ -1,4 +1,4 @@
import os import shutil
import subprocess import subprocess
from typing import Optional, Set, Type from typing import Optional, Set, Type
@ -10,7 +10,7 @@ from irctest.basecontrollers import (
TEMPLATE_CONFIG = """ TEMPLATE_CONFIG = """
# M:<Server NAME>:<YOUR Internet IP#>:<Geographic Location>:<Port>:<SID>: # M:<Server NAME>:<YOUR Internet IP#>:<Geographic Location>:<Port>:<SID>:
M:My.Little.Server:{hostname}:Somewhere:{port}:0042: M:My.Little.Server:{hostname}:test server:{port}:0042:
# A:<Your Name/Location>:<Your E-Mail Addr>:<other info>::<network name>: # A:<Your Name/Location>:<Your E-Mail Addr>:<other info>::<network name>:
A:Organization, IRC dept.:Daemon <ircd@example.irc.org>:Client Server::IRCnet: A:Organization, IRC dept.:Daemon <ircd@example.irc.org>:Client Server::IRCnet:
@ -29,8 +29,8 @@ O:*:operpassword:operuser::::
""" """
class Ircu2Controller(BaseServerController, DirectoryBasedController): class Irc2Controller(BaseServerController, DirectoryBasedController):
binary_name: str software_name = "irc2"
services_protocol: str services_protocol: str
supports_sts = False supports_sts = False
@ -51,6 +51,7 @@ class Ircu2Controller(BaseServerController, DirectoryBasedController):
run_services: bool, run_services: bool,
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
) -> None: ) -> None:
if valid_metadata_keys or invalid_metadata_keys: if valid_metadata_keys or invalid_metadata_keys:
raise NotImplementedByController( raise NotImplementedByController(
@ -66,7 +67,7 @@ class Ircu2Controller(BaseServerController, DirectoryBasedController):
self.create_config() self.create_config()
password_field = password if password else "" password_field = password if password else ""
assert self.directory assert self.directory
pidfile = os.path.join(self.directory, "ircd.pid") pidfile = self.directory / "ircd.pid"
with self.open_file("server.conf") as fd: with self.open_file("server.conf") as fd:
fd.write( fd.write(
TEMPLATE_CONFIG.format( TEMPLATE_CONFIG.format(
@ -76,18 +77,26 @@ class Ircu2Controller(BaseServerController, DirectoryBasedController):
pidfile=pidfile, pidfile=pidfile,
) )
) )
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
[ [
*faketime_cmd,
"ircd", "ircd",
"-s", # no iauth "-s", # no iauth
"-p", "-p",
"on", "on",
"-f", "-f",
os.path.join(self.directory, "server.conf"), self.directory / "server.conf",
], ],
# stderr=subprocess.DEVNULL, # stderr=subprocess.DEVNULL,
) )
def get_irctest_controller_class() -> Type[Ircu2Controller]: def get_irctest_controller_class() -> Type[Irc2Controller]:
return Ircu2Controller return Irc2Controller

View File

@ -1,4 +1,4 @@
import os import shutil
import subprocess import subprocess
from typing import Optional, Set, Type from typing import Optional, Set, Type
@ -51,6 +51,7 @@ features {{
class Ircu2Controller(BaseServerController, DirectoryBasedController): class Ircu2Controller(BaseServerController, DirectoryBasedController):
software_name = "ircu2"
supports_sts = False supports_sts = False
extban_mute_char = None extban_mute_char = None
@ -69,6 +70,7 @@ class Ircu2Controller(BaseServerController, DirectoryBasedController):
run_services: bool, run_services: bool,
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
) -> None: ) -> None:
if valid_metadata_keys or invalid_metadata_keys: if valid_metadata_keys or invalid_metadata_keys:
raise NotImplementedByController( raise NotImplementedByController(
@ -84,7 +86,7 @@ class Ircu2Controller(BaseServerController, DirectoryBasedController):
self.create_config() self.create_config()
password_field = 'password = "{}";'.format(password) if password else "" password_field = 'password = "{}";'.format(password) if password else ""
assert self.directory assert self.directory
pidfile = os.path.join(self.directory, "ircd.pid") pidfile = self.directory / "ircd.pid"
with self.open_file("server.conf") as fd: with self.open_file("server.conf") as fd:
fd.write( fd.write(
TEMPLATE_CONFIG.format( TEMPLATE_CONFIG.format(
@ -94,12 +96,20 @@ class Ircu2Controller(BaseServerController, DirectoryBasedController):
pidfile=pidfile, pidfile=pidfile,
) )
) )
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
[ [
*faketime_cmd,
"ircd", "ircd",
"-n", # don't detach "-n", # don't detach
"-f", "-f",
os.path.join(self.directory, "server.conf"), self.directory / "server.conf",
"-x", "-x",
"DEBUG", "DEBUG",
], ],

View File

@ -1,4 +1,3 @@
import os
import subprocess import subprocess
from typing import Optional, Type from typing import Optional, Type
@ -55,13 +54,19 @@ class LimnoriaController(BaseClientController, DirectoryBasedController):
# Runs a client with the config given as arguments # Runs a client with the config given as arguments
assert self.proc is None assert self.proc is None
self.create_config() self.create_config()
username = password = ""
mechanisms = ""
if auth: if auth:
mechanisms = " ".join(mech.to_string() for mech in auth.mechanisms) mechanisms = " ".join(mech.to_string() for mech in auth.mechanisms)
if auth.ecdsa_key: if auth.ecdsa_key:
with self.open_file("ecdsa_key.pem") as fd: with self.open_file("ecdsa_key.pem") as fd:
fd.write(auth.ecdsa_key) fd.write(auth.ecdsa_key)
else:
mechanisms = "" if auth.username:
username = auth.username.encode("unicode_escape").decode()
if auth.password:
password = auth.password.encode("unicode_escape").decode()
with self.open_file("bot.conf") as fd: with self.open_file("bot.conf") as fd:
fd.write( fd.write(
TEMPLATE_CONFIG.format( TEMPLATE_CONFIG.format(
@ -69,8 +74,8 @@ class LimnoriaController(BaseClientController, DirectoryBasedController):
loglevel="CRITICAL", loglevel="CRITICAL",
hostname=hostname, hostname=hostname,
port=port, port=port,
username=auth.username if auth else "", username=username,
password=auth.password if auth else "", password=password,
mechanisms=mechanisms.lower(), mechanisms=mechanisms.lower(),
enable_tls=tls_config.enable if tls_config else "False", enable_tls=tls_config.enable if tls_config else "False",
trusted_fingerprints=" ".join(tls_config.trusted_fingerprints) trusted_fingerprints=" ".join(tls_config.trusted_fingerprints)
@ -79,9 +84,7 @@ class LimnoriaController(BaseClientController, DirectoryBasedController):
) )
) )
assert self.directory assert self.directory
self.proc = subprocess.Popen( self.proc = subprocess.Popen(["supybot", self.directory / "bot.conf"])
["supybot", os.path.join(self.directory, "bot.conf")]
)
def get_irctest_controller_class() -> Type[LimnoriaController]: def get_irctest_controller_class() -> Type[LimnoriaController]:

View File

@ -1,4 +1,4 @@
import os import shutil
import subprocess import subprocess
from typing import Optional, Set, Type from typing import Optional, Set, Type
@ -92,6 +92,7 @@ class MammonController(BaseServerController, DirectoryBasedController):
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
restricted_metadata_keys: Optional[Set[str]] = None, restricted_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
) -> None: ) -> None:
if password is not None: if password is not None:
raise NotImplementedByController("PASS command") raise NotImplementedByController("PASS command")
@ -113,12 +114,20 @@ class MammonController(BaseServerController, DirectoryBasedController):
# with self.open_file('server.yml', 'r') as fd: # with self.open_file('server.yml', 'r') as fd:
# print(fd.read()) # print(fd.read())
assert self.directory assert self.directory
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
[ [
*faketime_cmd,
"mammond", "mammond",
"--nofork", # '--debug', "--nofork", # '--debug',
"--config", "--config",
os.path.join(self.directory, "server.yml"), self.directory / "server.yml",
] ]
) )

View File

@ -0,0 +1,11 @@
from typing import Type
from .ircu2 import Ircu2Controller
class NefariousController(Ircu2Controller):
software_name = "Nefarious"
def get_irctest_controller_class() -> Type[NefariousController]:
return NefariousController

View File

@ -1,4 +1,4 @@
import os import shutil
import subprocess import subprocess
from typing import Optional, Set, Type from typing import Optional, Set, Type
@ -12,7 +12,7 @@ from irctest.irc_utils.junkdrawer import find_hostname_and_port
TEMPLATE_CONFIG = """ TEMPLATE_CONFIG = """
[Global] [Global]
Name = My.Little.Server Name = My.Little.Server
Info = ExampleNET Server Info = test server
Bind = {hostname} Bind = {hostname}
Ports = {port} Ports = {port}
AdminInfo1 = Bob Smith AdminInfo1 = Bob Smith
@ -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
@ -53,6 +56,7 @@ class NgircdController(BaseServerController, DirectoryBasedController):
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
restricted_metadata_keys: Optional[Set[str]] = None, restricted_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
) -> None: ) -> None:
if valid_metadata_keys or invalid_metadata_keys: if valid_metadata_keys or invalid_metadata_keys:
raise NotImplementedByController( raise NotImplementedByController(
@ -78,6 +82,7 @@ class NgircdController(BaseServerController, DirectoryBasedController):
fd.write("\n") fd.write("\n")
assert self.directory assert self.directory
with self.open_file("server.conf") as fd: with self.open_file("server.conf") as fd:
fd.write( fd.write(
TEMPLATE_CONFIG.format( TEMPLATE_CONFIG.format(
@ -88,15 +93,23 @@ class NgircdController(BaseServerController, DirectoryBasedController):
password_field=password_field, password_field=password_field,
key_path=self.key_path, key_path=self.key_path,
pem_path=self.pem_path, pem_path=self.pem_path,
empty_file=os.path.join(self.directory, "empty.txt"), empty_file=self.directory / "empty.txt",
) )
) )
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
[ [
*faketime_cmd,
"ngircd", "ngircd",
"--nodaemon", "--nodaemon",
"--config", "--config",
os.path.join(self.directory, "server.conf"), self.directory / "server.conf",
], ],
# stdout=subprocess.DEVNULL, # stdout=subprocess.DEVNULL,
) )

View File

@ -45,7 +45,7 @@ class {{
}}; }};
connect {{ connect {{
name = "services.example.org"; name = "services.example.org";
host = "localhost"; # Used to validate incoming connection host = "127.0.0.1"; # Used to validate incoming connection
port = 0; # We don't need servers to connect to services port = 0; # We don't need servers to connect to services
send_password = "password"; send_password = "password";
accept_password = "password"; accept_password = "password";
@ -74,7 +74,7 @@ operator {{
class Plexus4Controller(BaseHybridController): class Plexus4Controller(BaseHybridController):
software_name = "Hybrid" software_name = "Plexus4"
binary_name = "ircd" binary_name = "ircd"
services_protocol = "plexus" services_protocol = "plexus"

View File

@ -1,4 +1,4 @@
import os import shutil
import subprocess import subprocess
from typing import Optional, Set, Type from typing import Optional, Set, Type
@ -69,6 +69,7 @@ class SnircdController(BaseServerController, DirectoryBasedController):
run_services: bool, run_services: bool,
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
) -> None: ) -> None:
if valid_metadata_keys or invalid_metadata_keys: if valid_metadata_keys or invalid_metadata_keys:
raise NotImplementedByController( raise NotImplementedByController(
@ -84,7 +85,7 @@ class SnircdController(BaseServerController, DirectoryBasedController):
self.create_config() self.create_config()
password_field = 'password = "{}";'.format(password) if password else "" password_field = 'password = "{}";'.format(password) if password else ""
assert self.directory assert self.directory
pidfile = os.path.join(self.directory, "ircd.pid") pidfile = self.directory / "ircd.pid"
with self.open_file("server.conf") as fd: with self.open_file("server.conf") as fd:
fd.write( fd.write(
TEMPLATE_CONFIG.format( TEMPLATE_CONFIG.format(
@ -94,12 +95,20 @@ class SnircdController(BaseServerController, DirectoryBasedController):
pidfile=pidfile, pidfile=pidfile,
) )
) )
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen( self.proc = subprocess.Popen(
[ [
*faketime_cmd,
"ircd", "ircd",
"-n", # don't detach "-n", # don't detach
"-f", "-f",
os.path.join(self.directory, "server.conf"), self.directory / "server.conf",
"-x", "-x",
"DEBUG", "DEBUG",
], ],

View File

@ -1,4 +1,4 @@
import os from pathlib import Path
import subprocess import subprocess
import tempfile import tempfile
from typing import Optional, TextIO, Type, cast from typing import Optional, TextIO, Type, cast
@ -38,14 +38,14 @@ class SopelController(BaseClientController):
super().kill() super().kill()
if self.filename: if self.filename:
try: try:
os.unlink(os.path.join(os.path.expanduser("~/.sopel/"), self.filename)) (Path("~/.sopel/").expanduser() / self.filename).unlink()
except OSError: #  File does not exist except OSError: # File does not exist
pass pass
def open_file(self, filename: str, mode: str = "a") -> TextIO: def open_file(self, filename: str, mode: str = "a") -> TextIO:
dir_path = os.path.expanduser("~/.sopel/") dir_path = Path("~/.sopel/").expanduser()
os.makedirs(dir_path, exist_ok=True) dir_path.mkdir(parents=True, exist_ok=True)
return cast(TextIO, open(os.path.join(dir_path, filename), mode)) return cast(TextIO, (dir_path / filename).open(mode))
def create_config(self) -> None: def create_config(self) -> None:
with self.open_file(self.filename): with self.open_file(self.filename):

View File

@ -1,6 +1,11 @@
import os import contextlib
import fcntl
import functools
from pathlib import Path
import shutil
import subprocess import subprocess
from typing import Optional, Set, Type import textwrap
from typing import Callable, ContextManager, Iterator, Optional, Set, Type
from irctest.basecontrollers import ( from irctest.basecontrollers import (
BaseServerController, BaseServerController,
@ -12,10 +17,12 @@ from irctest.irc_utils.junkdrawer import find_hostname_and_port
TEMPLATE_CONFIG = """ TEMPLATE_CONFIG = """
include "modules.default.conf"; include "modules.default.conf";
include "operclass.default.conf"; include "operclass.default.conf";
{extras}
include "help/help.conf";
me {{ me {{
name "My.Little.Server"; name "My.Little.Server";
info "ExampleNET Server"; info "test server";
sid "001"; sid "001";
}} }}
admin {{ admin {{
@ -87,11 +94,15 @@ set {{
// Prevent throttling, especially test_buffering.py which // Prevent throttling, especially test_buffering.py which
// triggers anti-flood with its very long lines // triggers anti-flood with its very long lines
unknown-users {{ unknown-users {{
nick-flood 255:10;
lag-penalty 1; lag-penalty 1;
lag-penalty-bytes 10000; lag-penalty-bytes 10000;
}} }}
}} }}
modes-on-join "+H 100:1d"; // Enables CHATHISTORY modes-on-join "+H 100:1d"; // Enables CHATHISTORY
{set_extras}
}} }}
tld {{ tld {{
@ -101,6 +112,10 @@ tld {{
rules "{empty_file}"; rules "{empty_file}";
}} }}
files {{
tunefile "{empty_file}";
}}
oper "operuser" {{ oper "operuser" {{
password = "operpassword"; password = "operpassword";
mask *; mask *;
@ -110,12 +125,53 @@ oper "operuser" {{
""" """
def _filelock(path: Path) -> Callable[[], ContextManager]:
"""Alternative to :cls:`multiprocessing.Lock` that works with pytest-xdist"""
@contextlib.contextmanager
def f() -> Iterator[None]:
with open(path, "a") as fd:
fcntl.flock(fd, fcntl.LOCK_EX)
yield
return f
_UNREALIRCD_BIN = shutil.which("unrealircd")
if _UNREALIRCD_BIN:
_UNREALIRCD_PREFIX = Path(_UNREALIRCD_BIN).parent.parent
# Try to keep that lock file specific to this Unrealircd instance
_LOCK_PATH = _UNREALIRCD_PREFIX / "irctest-unrealircd-startstop.lock"
else:
# unrealircd not found; we are probably going to crash later anyway...
_LOCK_PATH = Path("/tmp/irctest-unrealircd-startstop.lock")
_STARTSTOP_LOCK = _filelock(_LOCK_PATH)
"""
Unreal cleans its tmp/ directory after each run, which prevents
multiple processes from starting/stopping at the same time.
"""
@functools.lru_cache()
def installed_version() -> int:
output = subprocess.check_output(["unrealircd", "-v"], universal_newlines=True)
if output.startswith("UnrealIRCd-5."):
return 5
elif output.startswith("UnrealIRCd-6."):
return 6
else:
assert False, f"unexpected version: {output}"
class UnrealircdController(BaseServerController, DirectoryBasedController): class UnrealircdController(BaseServerController, DirectoryBasedController):
software_name = "UnrealIRCd" software_name = "UnrealIRCd"
supported_sasl_mechanisms = {"PLAIN"} supported_sasl_mechanisms = {"PLAIN"}
supports_sts = False supports_sts = False
extban_mute_char = "q" extban_mute_char = "quiet" if installed_version() >= 6 else "q"
software_version = installed_version()
def create_config(self) -> None: def create_config(self) -> None:
super().create_config() super().create_config()
@ -133,6 +189,7 @@ class UnrealircdController(BaseServerController, DirectoryBasedController):
valid_metadata_keys: Optional[Set[str]] = None, valid_metadata_keys: Optional[Set[str]] = None,
invalid_metadata_keys: Optional[Set[str]] = None, invalid_metadata_keys: Optional[Set[str]] = None,
restricted_metadata_keys: Optional[Set[str]] = None, restricted_metadata_keys: Optional[Set[str]] = None,
faketime: Optional[str],
) -> None: ) -> None:
if valid_metadata_keys or invalid_metadata_keys: if valid_metadata_keys or invalid_metadata_keys:
raise NotImplementedByController( raise NotImplementedByController(
@ -142,51 +199,86 @@ class UnrealircdController(BaseServerController, DirectoryBasedController):
self.port = port self.port = port
self.hostname = hostname self.hostname = hostname
self.create_config() self.create_config()
(unused_hostname, unused_port) = find_hostname_and_port()
(services_hostname, services_port) = find_hostname_and_port()
password_field = 'password "{}";'.format(password) if password else "" if installed_version() >= 6:
extras = textwrap.dedent(
self.gen_ssl() """
if ssl: include "snomasks.default.conf";
(tls_hostname, tls_port) = (hostname, port) loadmodule "cloak_md5";
(hostname, port) = (unused_hostname, unused_port) """
)
set_extras = textwrap.indent(
textwrap.dedent(
"""
// Remove RPL_WHOISSPECIAL used to advertise security groups
whois-details {
security-groups { everyone none; self none; oper none; }
}
"""
),
" ",
)
else: else:
# Unreal refuses to start without TLS enabled extras = ""
(tls_hostname, tls_port) = (unused_hostname, unused_port) set_extras = ""
with self.open_file("empty.txt") as fd: with self.open_file("empty.txt") as fd:
fd.write("\n") fd.write("\n")
assert self.directory password_field = 'password "{}";'.format(password) if password else ""
with self.open_file("unrealircd.conf") as fd:
fd.write( with _STARTSTOP_LOCK():
TEMPLATE_CONFIG.format( (services_hostname, services_port) = find_hostname_and_port()
hostname=hostname, (unused_hostname, unused_port) = find_hostname_and_port()
port=port,
services_hostname=services_hostname, self.gen_ssl()
services_port=services_port, if ssl:
tls_hostname=tls_hostname, (tls_hostname, tls_port) = (hostname, port)
tls_port=tls_port, (hostname, port) = (unused_hostname, unused_port)
password_field=password_field, else:
key_path=self.key_path, # Unreal refuses to start without TLS enabled
pem_path=self.pem_path, (tls_hostname, tls_port) = (unused_hostname, unused_port)
empty_file=os.path.join(self.directory, "empty.txt"),
assert self.directory
with self.open_file("unrealircd.conf") as fd:
fd.write(
TEMPLATE_CONFIG.format(
hostname=hostname,
port=port,
services_hostname=services_hostname,
services_port=services_port,
tls_hostname=tls_hostname,
tls_port=tls_port,
password_field=password_field,
key_path=self.key_path,
pem_path=self.pem_path,
empty_file=self.directory / "empty.txt",
extras=extras,
set_extras=set_extras,
)
) )
if faketime and shutil.which("faketime"):
faketime_cmd = ["faketime", "-f", faketime]
self.faketime_enabled = True
else:
faketime_cmd = []
self.proc = subprocess.Popen(
[
*faketime_cmd,
"unrealircd",
"-t",
"-F", # BOOT_NOFORK
"-f",
self.directory / "unrealircd.conf",
],
# stdout=subprocess.DEVNULL,
) )
self.proc = subprocess.Popen( self.wait_for_port()
[
"unrealircd",
"-t",
"-F", # BOOT_NOFORK
"-f",
os.path.join(self.directory, "unrealircd.conf"),
],
# stdout=subprocess.DEVNULL,
)
if run_services: if run_services:
self.wait_for_port()
self.services_controller = self.services_controller_class( self.services_controller = self.services_controller_class(
self.test_config, self self.test_config, self
) )
@ -196,6 +288,14 @@ class UnrealircdController(BaseServerController, DirectoryBasedController):
server_port=services_port, server_port=services_port,
) )
def kill_proc(self) -> None:
assert self.proc
with _STARTSTOP_LOCK():
self.proc.kill()
self.proc.wait(5) # wait for it to actually die
self.proc = None
def get_irctest_controller_class() -> Type[UnrealircdController]: def get_irctest_controller_class() -> Type[UnrealircdController]:
return UnrealircdController return UnrealircdController

449
irctest/dashboard/format.py Normal file
View File

@ -0,0 +1,449 @@
import base64
import dataclasses
import gzip
import hashlib
import importlib
from pathlib import Path
import re
import sys
from typing import (
IO,
Callable,
Dict,
Iterable,
Iterator,
List,
Optional,
Tuple,
TypeVar,
)
import xml.etree.ElementTree as ET
from defusedxml.ElementTree import parse as parse_xml
import docutils.core
NETLIFY_CHAR_BLACKLIST = frozenset('":<>|*?\r\n#')
"""Characters not allowed in output filenames"""
@dataclasses.dataclass
class CaseResult:
module_name: str
class_name: str
test_name: str
job: str
success: bool
skipped: bool
system_out: Optional[str]
details: Optional[str] = None
type: Optional[str] = None
message: Optional[str] = None
def output_filename(self):
test_name = self.test_name
if len(test_name) > 50 or set(test_name) & NETLIFY_CHAR_BLACKLIST:
# File name too long or otherwise invalid. This should be good enough:
m = re.match(r"(?P<function_name>\w+?)\[(?P<params>.+)\]", test_name)
assert m, "File name is too long but has no parameter."
test_name = f'{m.group("function_name")}[{md5sum(m.group("params"))}]'
return f"{self.job}_{self.module_name}.{self.class_name}.{test_name}.txt"
TK = TypeVar("TK")
TV = TypeVar("TV")
def md5sum(text: str) -> str:
return base64.urlsafe_b64encode(hashlib.md5(text.encode()).digest()).decode()
def group_by(list_: Iterable[TV], key: Callable[[TV], TK]) -> Dict[TK, List[TV]]:
groups: Dict[TK, List[TV]] = {}
for value in list_:
groups.setdefault(key(value), []).append(value)
return groups
def iter_job_results(job_file_name: Path, job: ET.ElementTree) -> Iterator[CaseResult]:
(suite,) = job.getroot()
for case in suite:
if "name" not in case.attrib:
continue
success = True
skipped = False
details = None
system_out = None
extra = {}
for child in case:
if child.tag == "skipped":
success = True
skipped = True
details = None
extra = child.attrib
elif child.tag in ("failure", "error"):
success = False
skipped = False
details = child.text
extra = child.attrib
elif child.tag == "system-out":
assert (
system_out is None
# for some reason, skipped tests have two system-out;
# and the second one contains test teardown
or child.text.startswith(system_out.rstrip())
), ("Duplicate system-out tag", repr(system_out), repr(child.text))
system_out = child.text
else:
assert False, child
(module_name, class_name) = case.attrib["classname"].rsplit(".", 1)
m = re.match(
r"(.*/)?pytest[ -]results[ _](?P<name>.*)"
r"[ _][(]?(stable|release|devel|devel_release)[)]?/pytest.xml(.gz)?",
str(job_file_name),
)
assert m, job_file_name
yield CaseResult(
module_name=module_name,
class_name=class_name,
test_name=case.attrib["name"],
job=m.group("name"),
success=success,
skipped=skipped,
details=details,
system_out=system_out,
**extra,
)
def rst_to_element(s: str) -> ET.Element:
html = docutils.core.publish_parts(s, writer_name="xhtml")["html_body"]
htmltree = ET.fromstring(html)
return htmltree
def append_docstring(element: ET.Element, obj: object) -> None:
if obj.__doc__ is None:
return
element.append(rst_to_element(obj.__doc__))
def build_job_html(job: str, results: List[CaseResult]) -> ET.Element:
jobs = sorted({result.job for result in results})
root = ET.Element("html")
head = ET.SubElement(root, "head")
ET.SubElement(head, "title").text = job
ET.SubElement(head, "link", rel="stylesheet", type="text/css", href="./style.css")
body = ET.SubElement(root, "body")
ET.SubElement(body, "h1").text = job
table = build_test_table(jobs, results)
table.set("class", "job-results test-matrix")
body.append(table)
return root
def build_module_html(
jobs: List[str], results: List[CaseResult], module_name: str
) -> ET.Element:
module = importlib.import_module(module_name)
root = ET.Element("html")
head = ET.SubElement(root, "head")
ET.SubElement(head, "title").text = module_name
ET.SubElement(head, "link", rel="stylesheet", type="text/css", href="./style.css")
body = ET.SubElement(root, "body")
ET.SubElement(body, "h1").text = module_name
append_docstring(body, module)
table = build_test_table(jobs, results)
table.set("class", "module-results test-matrix")
body.append(table)
return root
def build_test_table(jobs: List[str], results: List[CaseResult]) -> ET.Element:
multiple_modules = len({r.module_name for r in results}) > 1
results_by_module_and_class = group_by(
results, lambda r: (r.module_name, r.class_name)
)
table = ET.Element("table")
job_row = ET.Element("tr")
ET.SubElement(job_row, "th") # column of case name
for job in jobs:
cell = ET.SubElement(job_row, "th")
ET.SubElement(ET.SubElement(cell, "div"), "span").text = job
cell.set("class", "job-name")
for ((module_name, class_name), class_results) in sorted(
results_by_module_and_class.items()
):
if multiple_modules:
# if the page shows classes from various modules, use the fully-qualified
# name in order to disambiguate and be clearer (eg. show
# "irctest.server_tests.extended_join.MetadataTestCase" instead of just
# "MetadataTestCase" which looks like it's about IRCv3's METADATA spec.
qualified_class_name = f"{module_name}.{class_name}"
else:
# otherwise, it's not needed, so let's not display it
qualified_class_name = class_name
module = importlib.import_module(module_name)
# Header row: class name
header_row = ET.SubElement(table, "tr")
th = ET.SubElement(header_row, "th", colspan=str(len(jobs) + 1))
row_anchor = f"{qualified_class_name}"
section_header = ET.SubElement(
ET.SubElement(th, "h2"),
"a",
href=f"#{row_anchor}",
id=row_anchor,
)
section_header.text = qualified_class_name
append_docstring(th, getattr(module, class_name))
# Header row: one column for each implementation
table.append(job_row)
# One row for each test:
results_by_test = group_by(class_results, key=lambda r: r.test_name)
for (test_name, test_results) in sorted(results_by_test.items()):
row_anchor = f"{qualified_class_name}.{test_name}"
if len(row_anchor) >= 50:
# Too long; give up on generating readable URL
# TODO: only hash test parameter
row_anchor = md5sum(row_anchor)
row = ET.SubElement(table, "tr", id=row_anchor)
cell = ET.SubElement(row, "th")
cell.set("class", "test-name")
cell_link = ET.SubElement(cell, "a", href=f"#{row_anchor}")
cell_link.text = test_name
results_by_job = group_by(test_results, key=lambda r: r.job)
for job_name in jobs:
cell = ET.SubElement(row, "td")
try:
(result,) = results_by_job[job_name]
except KeyError:
cell.set("class", "deselected")
cell.text = "d"
continue
text: Optional[str]
if result.skipped:
cell.set("class", "skipped")
if result.type == "pytest.skip":
text = "s"
elif result.type == "pytest.xfail":
text = "X"
cell.set("class", "expected-failure")
else:
text = result.type
elif result.success:
cell.set("class", "success")
if result.type:
# dead code?
text = result.type
else:
text = "."
else:
cell.set("class", "failure")
if result.type:
# dead code?
text = result.type
else:
text = "f"
if result.system_out:
# There is a log file; link to it.
a = ET.SubElement(cell, "a", href=f"./{result.output_filename()}")
a.text = text or "?"
else:
cell.text = text or "?"
if result.message:
cell.set("title", result.message)
return table
def write_html_pages(
output_dir: Path, results: List[CaseResult]
) -> List[Tuple[str, str, str]]:
"""Returns the list of (module_name, file_name)."""
output_dir.mkdir(parents=True, exist_ok=True)
results_by_module = group_by(results, lambda r: r.module_name)
# used as columns
jobs = list(sorted({r.job for r in results}))
job_categories = {}
for job in jobs:
is_client = any(
"client_tests" in result.module_name and result.job == job
for result in results
)
is_server = any(
"server_tests" in result.module_name and result.job == job
for result in results
)
assert is_client != is_server, (job, is_client, is_server)
if job.endswith(("-atheme", "-anope", "-dlk")):
assert is_server
job_categories[job] = "server-with-services"
elif is_server:
job_categories[job] = "server" # with or without services
else:
assert is_client
job_categories[job] = "client"
pages = []
for (module_name, module_results) in sorted(results_by_module.items()):
# Filter out client jobs if this is a server test module, and vice versa
module_categories = {
job_categories[result.job]
for result in results
if result.module_name == module_name and not result.skipped
}
module_jobs = [job for job in jobs if job_categories[job] in module_categories]
root = build_module_html(module_jobs, module_results, module_name)
file_name = f"{module_name}.xhtml"
write_xml_file(output_dir / file_name, root)
pages.append(("module", module_name, file_name))
for category in ("server", "client"):
for job in [job for job in job_categories if job_categories[job] == category]:
job_results = [
result
for result in results
if result.job == job or result.job.startswith(job + "-")
]
root = build_job_html(job, job_results)
file_name = f"{job}.xhtml"
write_xml_file(output_dir / file_name, root)
pages.append(("job", job, file_name))
return pages
def write_test_outputs(output_dir: Path, results: List[CaseResult]) -> None:
"""Writes stdout files of each test."""
for result in results:
if result.system_out is None:
continue
output_file = output_dir / result.output_filename()
with output_file.open("wt") as fd:
fd.write(result.system_out)
def write_html_index(output_dir: Path, pages: List[Tuple[str, str, str]]) -> None:
root = ET.Element("html")
head = ET.SubElement(root, "head")
ET.SubElement(head, "title").text = "irctest dashboard"
ET.SubElement(head, "link", rel="stylesheet", type="text/css", href="./style.css")
body = ET.SubElement(root, "body")
ET.SubElement(body, "h1").text = "irctest dashboard"
module_pages = []
job_pages = []
for (page_type, title, file_name) in sorted(pages):
if page_type == "module":
module_pages.append((title, file_name))
elif page_type == "job":
job_pages.append((title, file_name))
else:
assert False, page_type
ET.SubElement(body, "h2").text = "Tests by command/specification"
dl = ET.SubElement(body, "dl")
dl.set("class", "module-index")
for (module_name, file_name) in sorted(module_pages):
module = importlib.import_module(module_name)
link = ET.SubElement(ET.SubElement(dl, "dt"), "a", href=f"./{file_name}")
link.text = module_name
append_docstring(ET.SubElement(dl, "dd"), module)
ET.SubElement(body, "h2").text = "Tests by implementation"
ul = ET.SubElement(body, "ul")
ul.set("class", "job-index")
for (job, file_name) in sorted(job_pages):
link = ET.SubElement(ET.SubElement(ul, "li"), "a", href=f"./{file_name}")
link.text = job
write_xml_file(output_dir / "index.xhtml", root)
def write_assets(output_dir: Path) -> None:
css_path = output_dir / "style.css"
source_css_path = Path(__file__).parent / "style.css"
with css_path.open("wt") as fd:
with source_css_path.open() as source_fd:
fd.write(source_fd.read())
def write_xml_file(filename: Path, root: ET.Element) -> None:
# Hacky: ET expects the namespace to be present in every tag we create instead;
# but it would be excessively verbose.
root.set("xmlns", "http://www.w3.org/1999/xhtml")
# Serialize
s = ET.tostring(root)
with filename.open("wb") as fd:
fd.write(s)
def parse_xml_file(filename: Path) -> ET.ElementTree:
fd: IO
if filename.suffix == ".gz":
with gzip.open(filename, "rb") as fd: # type: ignore
return parse_xml(fd) # type: ignore
else:
with open(filename) as fd:
return parse_xml(fd) # type: ignore
def main(output_path: Path, filenames: List[Path]) -> int:
results = [
result
for filename in filenames
for result in iter_job_results(filename, parse_xml_file(filename))
]
pages = write_html_pages(output_path, results)
write_html_index(output_path, pages)
write_test_outputs(output_path, results)
write_assets(output_path)
return 0
if __name__ == "__main__":
(_, output_path, *filenames) = sys.argv
exit(main(Path(output_path), list(map(Path, filenames))))

View File

@ -0,0 +1,87 @@
import dataclasses
import gzip
import io
import json
from pathlib import Path
import sys
from typing import Iterator
import urllib.parse
import urllib.request
import zipfile
@dataclasses.dataclass
class Artifact:
repo: str
run_id: int
name: str
download_url: str
@property
def public_download_url(self):
# GitHub API is not available publicly for artifacts, we need to use
# a third-party proxy to access it...
name = urllib.parse.quote(self.name)
return f"https://nightly.link/{repo}/actions/runs/{self.run_id}/{name}.zip"
def iter_run_artifacts(repo: str, run_id: int) -> Iterator[Artifact]:
request = urllib.request.Request(
f"https://api.github.com/repos/{repo}/actions/runs/{run_id}/artifacts"
"?per_page=100",
headers={"Accept": "application/vnd.github.v3+json"},
)
response = urllib.request.urlopen(request)
for artifact in json.load(response)["artifacts"]:
if not artifact["name"].startswith(("pytest-results_", "pytest results ")):
continue
if artifact["expired"]:
continue
yield Artifact(
repo=repo,
run_id=run_id,
name=artifact["name"],
download_url=artifact["archive_download_url"],
)
def download_artifact(output_name: Path, url: str) -> None:
if output_name.exists():
return
response = urllib.request.urlopen(url)
archive_bytes = response.read() # Can't stream it, it's a ZIP
with zipfile.ZipFile(io.BytesIO(archive_bytes)) as archive:
with archive.open("pytest.xml") as input_fd:
pytest_xml = input_fd.read()
tmp_output_path = output_name.with_suffix(".tmp")
with gzip.open(tmp_output_path, "wb") as output_fd:
output_fd.write(pytest_xml)
# Atomically write to the output path, so that we don't write partial files in case
# the download process is interrupted
tmp_output_path.rename(output_name)
def main(output_dir: Path, repo: str, run_id: int) -> int:
output_dir.mkdir(parents=True, exist_ok=True)
run_path = output_dir / str(run_id)
run_path.mkdir(exist_ok=True)
for artifact in iter_run_artifacts(repo, run_id):
artifact_path = run_path / artifact.name / "pytest.xml.gz"
artifact_path.parent.mkdir(exist_ok=True)
try:
download_artifact(artifact_path, artifact.download_url)
except Exception:
download_artifact(artifact_path, artifact.public_download_url)
print("downloaded", artifact.name)
return 0
if __name__ == "__main__":
(_, output_path, repo, run_id) = sys.argv
exit(main(Path(output_path), repo, int(run_id)))

View File

@ -0,0 +1,67 @@
@media (prefers-color-scheme: dark) {
body {
background-color: #121212;
color: rgba(255, 255, 255, 0.87);
}
a {
filter: invert(0.85) hue-rotate(180deg);
}
}
dl.module-index {
column-width: 40em; /* Magic constant for 2 columns on average laptop/desktop */
}
/* Only 1px solid border between cells */
table.test-matrix {
border-spacing: 0;
border-collapse: collapse;
}
table.test-matrix td {
text-align: center;
border: 1px solid grey;
}
/* Make link take the whole cell */
table.test-matrix td a {
display: block;
margin: 0;
padding: 0;
width: 100%;
height: 100%;
color: black;
text-decoration: none;
}
/* Test matrix colors */
table.test-matrix .deselected {
background-color: grey;
}
table.test-matrix .success {
background-color: green;
}
table.test-matrix .skipped {
background-color: yellow;
}
table.test-matrix .failure {
background-color: red;
}
table.test-matrix .expected-failure {
background-color: orange;
}
/* Rotate headers, thanks to https://css-tricks.com/rotated-table-column-headers/ */
table.module-results th.job-name {
height: 140px;
white-space: nowrap;
}
table.module-results th.job-name > div {
transform:
translate(28px, 50px)
rotate(315deg);
width: 40px;
}
table.module-results th.job-name > div > span {
border-bottom: 1px solid grey;
padding-left: 0px;
}

View File

@ -66,6 +66,7 @@ RPL_WHOISIDLE = "317"
RPL_ENDOFWHOIS = "318" RPL_ENDOFWHOIS = "318"
RPL_WHOISCHANNELS = "319" RPL_WHOISCHANNELS = "319"
RPL_WHOISSPECIAL = "320" RPL_WHOISSPECIAL = "320"
RPL_LISTSTART = "321"
RPL_LIST = "322" RPL_LIST = "322"
RPL_LISTEND = "323" RPL_LISTEND = "323"
RPL_CHANNELMODEIS = "324" RPL_CHANNELMODEIS = "324"
@ -86,6 +87,7 @@ RPL_ENDOFEXCEPTLIST = "349"
RPL_VERSION = "351" RPL_VERSION = "351"
RPL_WHOREPLY = "352" RPL_WHOREPLY = "352"
RPL_NAMREPLY = "353" RPL_NAMREPLY = "353"
RPL_WHOSPCRPL = "354"
RPL_LINKS = "364" RPL_LINKS = "364"
RPL_ENDOFLINKS = "365" RPL_ENDOFLINKS = "365"
RPL_ENDOFNAMES = "366" RPL_ENDOFNAMES = "366"
@ -140,6 +142,7 @@ ERR_USERONCHANNEL = "443"
ERR_NOLOGIN = "444" ERR_NOLOGIN = "444"
ERR_SUMMONDISABLED = "445" ERR_SUMMONDISABLED = "445"
ERR_USERSDISABLED = "446" ERR_USERSDISABLED = "446"
ERR_FORBIDDENCHANNEL = "448"
ERR_NOTREGISTERED = "451" ERR_NOTREGISTERED = "451"
ERR_NEEDMOREPARAMS = "461" ERR_NEEDMOREPARAMS = "461"
ERR_ALREADYREGISTRED = "462" ERR_ALREADYREGISTRED = "462"

View File

@ -13,18 +13,18 @@ class Operator:
pass pass
class AnyStr(Operator): class _AnyStr(Operator):
"""Wildcard matching any string""" """Wildcard matching any string"""
def __repr__(self) -> str: def __repr__(self) -> str:
return "AnyStr" return "ANYSTR"
class AnyOptStr(Operator): class _AnyOptStr(Operator):
"""Wildcard matching any string as well as None""" """Wildcard matching any string as well as None"""
def __repr__(self) -> str: def __repr__(self) -> str:
return "AnyOptStr" return "ANYOPTSTR"
@dataclasses.dataclass(frozen=True) @dataclasses.dataclass(frozen=True)
@ -43,6 +43,14 @@ class NotStrRe(Operator):
return f"NotStrRe(r'{self.regexp}')" return f"NotStrRe(r'{self.regexp}')"
@dataclasses.dataclass(frozen=True)
class InsensitiveStr(Operator):
string: str
def __repr__(self) -> str:
return f"InsensitiveStr({self.string!r})"
@dataclasses.dataclass(frozen=True) @dataclasses.dataclass(frozen=True)
class RemainingKeys(Operator): class RemainingKeys(Operator):
"""Used in a dict pattern to match all remaining keys. """Used in a dict pattern to match all remaining keys.
@ -51,30 +59,42 @@ class RemainingKeys(Operator):
key: Operator key: Operator
def __repr__(self) -> str: def __repr__(self) -> str:
return f"Keys({self.key!r})" return f"RemainingKeys({self.key!r})"
ANYSTR = AnyStr() ANYSTR = _AnyStr()
"""Singleton, spares two characters""" """Singleton, spares two characters"""
ANYDICT = {RemainingKeys(ANYSTR): AnyOptStr()} ANYOPTSTR = _AnyOptStr()
"""Singleton, spares two characters"""
ANYDICT = {RemainingKeys(ANYSTR): ANYOPTSTR}
"""Matches any dictionary; useful to compare tags dict, eg. """Matches any dictionary; useful to compare tags dict, eg.
`match_dict(got_tags, {"label": "foo", **ANYDICT})`""" `match_dict(got_tags, {"label": "foo", **ANYDICT})`"""
class _AnyListRemainder: @dataclasses.dataclass(frozen=True)
class ListRemainder:
item: Operator
min_length: int = 0
def __repr__(self) -> str: def __repr__(self) -> str:
return "*ANYLIST" if self.min_length:
return f"ListRemainder({self.item!r}, min_length={self.min_length})"
elif self.item is ANYSTR:
return "*ANYLIST"
else:
return f"ListRemainder({self.item!r})"
ANYLIST = [_AnyListRemainder()] ANYLIST = [ListRemainder(ANYSTR)]
"""Matches any list remainder""" """Matches any list remainder"""
def match_string(got: Optional[str], expected: Union[str, Operator, None]) -> bool: def match_string(got: Optional[str], expected: Union[str, Operator, None]) -> bool:
if isinstance(expected, AnyOptStr): if isinstance(expected, _AnyOptStr):
return True return True
elif isinstance(expected, AnyStr) and got is not None: elif isinstance(expected, _AnyStr) and got is not None:
return True return True
elif isinstance(expected, StrRe): elif isinstance(expected, StrRe):
if got is None or not re.match(expected.regexp, got): if got is None or not re.match(expected.regexp, got):
@ -82,6 +102,9 @@ def match_string(got: Optional[str], expected: Union[str, Operator, None]) -> bo
elif isinstance(expected, NotStrRe): elif isinstance(expected, NotStrRe):
if got is None or re.match(expected.regexp, got): if got is None or re.match(expected.regexp, got):
return False return False
elif isinstance(expected, InsensitiveStr):
if got is None or got.lower() != expected.string.lower():
return False
elif isinstance(expected, Operator): elif isinstance(expected, Operator):
raise NotImplementedError(f"Unsupported operator: {expected}") raise NotImplementedError(f"Unsupported operator: {expected}")
elif got != expected: elif got != expected:
@ -98,9 +121,13 @@ def match_list(
The ANYSTR operator can be used on the 'expected' side as a wildcard, The ANYSTR operator can be used on the 'expected' side as a wildcard,
matching any *single* value; and StrRe("<regexp>") can be used to match regular matching any *single* value; and StrRe("<regexp>") can be used to match regular
expressions""" expressions"""
if expected[-1] is ANYLIST[0]: if expected and isinstance(expected[-1], ListRemainder):
expected = expected[0:-1] # Expand the 'expected' list to have as many items as the 'got' list
got = got[0 : len(expected)] # Ignore remaining expected = list(expected) # copy
remainder = expected.pop()
nb_remaining_items = len(got) - len(expected)
expected += [remainder.item] * max(nb_remaining_items, remainder.min_length)
if len(got) != len(expected): if len(got) != len(expected):
return False return False
return all( return all(
@ -128,13 +155,15 @@ def match_dict(
for (expected_key, expected_value) in expected.items(): for (expected_key, expected_value) in expected.items():
if isinstance(expected_key, RemainingKeys): if isinstance(expected_key, RemainingKeys):
remaining_keys_wildcard = (expected_key.key, expected_value) remaining_keys_wildcard = (expected_key.key, expected_value)
elif isinstance(expected_key, Operator):
raise NotImplementedError(f"Unsupported operator: {expected_key}")
else: else:
if expected_key not in got: for key in got:
return False if match_string(key, expected_key) and match_string(
got_value = got.pop(expected_key) got[key], expected_value
if not match_string(got_value, expected_value): ):
got.pop(key)
break
else:
# Found no (key, value) pair matching the request
return False return False
if remaining_keys_wildcard: if remaining_keys_wildcard:

View File

@ -14,6 +14,11 @@ class ImplementationChoice(unittest.SkipTest):
) )
class OptionalCommandNotSupported(unittest.SkipTest):
def __str__(self) -> str:
return "Unsupported command: {}".format(self.args[0])
class OptionalExtensionNotSupported(unittest.SkipTest): class OptionalExtensionNotSupported(unittest.SkipTest):
def __str__(self) -> str: def __str__(self) -> str:
return "Unsupported extension: {}".format(self.args[0]) return "Unsupported extension: {}".format(self.args[0])

View File

@ -1,13 +1,24 @@
"""Internal checks of assertion implementations."""
from typing import Dict, List, Tuple from typing import Dict, List, Tuple
import pytest import pytest
from irctest import cases from irctest import cases
from irctest.irc_utils.message_parser import parse_message from irctest.irc_utils.message_parser import parse_message
from irctest.patma import ANYDICT, ANYSTR, AnyOptStr, NotStrRe, RemainingKeys, StrRe from irctest.patma import (
ANYDICT,
ANYLIST,
ANYOPTSTR,
ANYSTR,
ListRemainder,
NotStrRe,
RemainingKeys,
StrRe,
)
# fmt: off # fmt: off
MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str]]] = [ MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str], List[str]]] = [
( (
# the specification: # the specification:
dict( dict(
@ -27,6 +38,11 @@ MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str]]] = [
[ [
"PRIVMSG #chan hello2", "PRIVMSG #chan hello2",
"PRIVMSG #chan2 hello", "PRIVMSG #chan2 hello",
],
# and they each error with:
[
"expected params to match ['#chan', 'hello'], got ['#chan', 'hello2']",
"expected params to match ['#chan', 'hello'], got ['#chan2', 'hello']",
] ]
), ),
( (
@ -49,6 +65,11 @@ MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str]]] = [
[ [
"PRIVMSG #chan :hi", "PRIVMSG #chan :hi",
"PRIVMSG #chan2 hello", "PRIVMSG #chan2 hello",
],
# and they each error with:
[
"expected params to match ['#chan', StrRe(r'hello.*')], got ['#chan', 'hi']",
"expected params to match ['#chan', StrRe(r'hello.*')], got ['#chan2', 'hello']",
] ]
), ),
( (
@ -67,6 +88,12 @@ MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str]]] = [
"PRIVMSG #chan :hi", "PRIVMSG #chan :hi",
":foo2!baz@qux PRIVMSG #chan hello", ":foo2!baz@qux PRIVMSG #chan hello",
"@tag1=bar :foo2!baz@qux PRIVMSG #chan :hello", "@tag1=bar :foo2!baz@qux PRIVMSG #chan :hello",
],
# and they each error with:
[
"expected nick to be foo, got None instead",
"expected nick to be foo, got foo2 instead",
"expected nick to be foo, got foo2 instead",
] ]
), ),
( (
@ -87,6 +114,13 @@ MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str]]] = [
"@tag1=value1 PRIVMSG #chan :hello", "@tag1=value1 PRIVMSG #chan :hello",
"PRIVMSG #chan hello", "PRIVMSG #chan hello",
":foo!baz@qux PRIVMSG #chan hello", ":foo!baz@qux PRIVMSG #chan hello",
],
# and they each error with:
[
"expected tags to match {'tag1': 'bar'}, got {'tag1': 'bar', 'tag2': ''}",
"expected tags to match {'tag1': 'bar'}, got {'tag1': 'value1'}",
"expected tags to match {'tag1': 'bar'}, got {}",
"expected tags to match {'tag1': 'bar'}, got {}",
] ]
), ),
( (
@ -107,6 +141,12 @@ MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str]]] = [
"@tag1=bar;tag2= PRIVMSG #chan :hello", "@tag1=bar;tag2= PRIVMSG #chan :hello",
"PRIVMSG #chan hello", "PRIVMSG #chan hello",
":foo!baz@qux PRIVMSG #chan hello", ":foo!baz@qux PRIVMSG #chan hello",
],
# and they each error with:
[
"expected tags to match {'tag1': ANYSTR}, got {'tag1': 'bar', 'tag2': ''}",
"expected tags to match {'tag1': ANYSTR}, got {}",
"expected tags to match {'tag1': ANYSTR}, got {}",
] ]
), ),
( (
@ -129,12 +169,53 @@ MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str]]] = [
"PRIVMSG #chan hello2", "PRIVMSG #chan hello2",
"PRIVMSG #chan2 hello", "PRIVMSG #chan2 hello",
":foo!baz@qux PRIVMSG #chan hello", ":foo!baz@qux PRIVMSG #chan hello",
],
# and they each error with:
[
"expected command to be PRIVMSG, got PRIVMG",
"expected tags to match {'tag1': 'bar', RemainingKeys(ANYSTR): ANYOPTSTR}, got {'tag1': 'value1'}",
"expected params to match ['#chan', 'hello'], got ['#chan', 'hello2']",
"expected params to match ['#chan', 'hello'], got ['#chan2', 'hello']",
"expected tags to match {'tag1': 'bar', RemainingKeys(ANYSTR): ANYOPTSTR}, got {}",
] ]
), ),
( (
# the specification: # the specification:
dict( dict(
tags={"tag1": "bar", RemainingKeys(NotStrRe("tag2")): AnyOptStr()}, tags={StrRe("tag[12]"): "bar", **ANYDICT},
command="PRIVMSG",
params=["#chan", "hello"],
),
# matches:
[
"@tag1=bar PRIVMSG #chan :hello",
"@tag1=bar;tag2= PRIVMSG #chan :hello",
"@tag1=bar :foo!baz@qux PRIVMSG #chan :hello",
"@tag2=bar PRIVMSG #chan :hello",
"@tag1=bar;tag2= PRIVMSG #chan :hello",
"@tag1=;tag2=bar PRIVMSG #chan :hello",
],
# and does not match:
[
"PRIVMG #chan :hello",
"@tag1=value1 PRIVMSG #chan :hello",
"PRIVMSG #chan hello2",
"PRIVMSG #chan2 hello",
":foo!baz@qux PRIVMSG #chan hello",
],
# and they each error with:
[
"expected command to be PRIVMSG, got PRIVMG",
"expected tags to match {StrRe(r'tag[12]'): 'bar', RemainingKeys(ANYSTR): ANYOPTSTR}, got {'tag1': 'value1'}",
"expected params to match ['#chan', 'hello'], got ['#chan', 'hello2']",
"expected params to match ['#chan', 'hello'], got ['#chan2', 'hello']",
"expected tags to match {StrRe(r'tag[12]'): 'bar', RemainingKeys(ANYSTR): ANYOPTSTR}, got {}",
]
),
(
# the specification:
dict(
tags={"tag1": "bar", RemainingKeys(NotStrRe("tag2")): ANYOPTSTR},
command="PRIVMSG", command="PRIVMSG",
params=["#chan", "hello"], params=["#chan", "hello"],
), ),
@ -150,6 +231,98 @@ MESSAGE_SPECS: List[Tuple[Dict, List[str], List[str]]] = [
"@tag1=value1 PRIVMSG #chan :hello", "@tag1=value1 PRIVMSG #chan :hello",
"@tag1=bar;tag2= PRIVMSG #chan :hello", "@tag1=bar;tag2= PRIVMSG #chan :hello",
"@tag1=bar;tag2=baz PRIVMSG #chan :hello", "@tag1=bar;tag2=baz PRIVMSG #chan :hello",
],
# and they each error with:
[
"expected command to be PRIVMSG, got PRIVMG",
"expected tags to match {'tag1': 'bar', RemainingKeys(NotStrRe(r'tag2')): ANYOPTSTR}, got {'tag1': 'value1'}",
"expected tags to match {'tag1': 'bar', RemainingKeys(NotStrRe(r'tag2')): ANYOPTSTR}, got {'tag1': 'bar', 'tag2': ''}",
"expected tags to match {'tag1': 'bar', RemainingKeys(NotStrRe(r'tag2')): ANYOPTSTR}, got {'tag1': 'bar', 'tag2': 'baz'}",
]
),
(
# the specification:
dict(
command="005",
params=["nick", "FOO=1", *ANYLIST],
),
# matches:
[
"005 nick FOO=1",
"005 nick FOO=1 BAR=2",
],
# and does not match:
[
"005 nick",
"005 nick BAR=2",
],
# and they each error with:
[
"expected params to match ['nick', 'FOO=1', *ANYLIST], got ['nick']",
"expected params to match ['nick', 'FOO=1', *ANYLIST], got ['nick', 'BAR=2']",
]
),
(
# the specification:
dict(
command="005",
params=["nick", ListRemainder(ANYSTR, min_length=1)],
),
# matches:
[
"005 nick FOO=1",
"005 nick FOO=1 BAR=2",
"005 nick BAR=2",
],
# and does not match:
[
"005 nick",
],
# and they each error with:
[
"expected params to match ['nick', ListRemainder(ANYSTR, min_length=1)], got ['nick']",
]
),
(
# the specification:
dict(
command="005",
params=["nick", ListRemainder(StrRe("[A-Z]+=.*"), min_length=1)],
),
# matches:
[
"005 nick FOO=1",
"005 nick FOO=1 BAR=2",
"005 nick BAR=2",
],
# and does not match:
[
"005 nick",
"005 nick foo=1",
],
# and they each error with:
[
"expected params to match ['nick', ListRemainder(StrRe(r'[A-Z]+=.*'), min_length=1)], got ['nick']",
"expected params to match ['nick', ListRemainder(StrRe(r'[A-Z]+=.*'), min_length=1)], got ['nick', 'foo=1']",
]
),
(
# the specification:
dict(
command="PING",
params=["abc"]
),
# matches:
[
"PING abc",
],
# and does not match:
[
"PONG def"
],
# and they each error with:
[
"expected command to be PING, got PONG"
] ]
), ),
] ]
@ -161,7 +334,7 @@ class IrcTestCaseTestCase(cases._IrcTestCase):
"spec,msg", "spec,msg",
[ [
pytest.param(spec, msg, id=f"{spec}-{msg}") pytest.param(spec, msg, id=f"{spec}-{msg}")
for (spec, positive_matches, _) in MESSAGE_SPECS for (spec, positive_matches, _, _) in MESSAGE_SPECS
for msg in positive_matches for msg in positive_matches
], ],
) )
@ -174,7 +347,7 @@ class IrcTestCaseTestCase(cases._IrcTestCase):
"spec,msg", "spec,msg",
[ [
pytest.param(spec, msg, id=f"{spec}-{msg}") pytest.param(spec, msg, id=f"{spec}-{msg}")
for (spec, _, negative_matches) in MESSAGE_SPECS for (spec, _, negative_matches, _) in MESSAGE_SPECS
for msg in negative_matches for msg in negative_matches
], ],
) )
@ -183,3 +356,14 @@ class IrcTestCaseTestCase(cases._IrcTestCase):
assert not self.messageEqual(parse_message(msg), **spec), msg assert not self.messageEqual(parse_message(msg), **spec), msg
with pytest.raises(AssertionError): with pytest.raises(AssertionError):
self.assertMessageMatch(parse_message(msg), **spec), msg self.assertMessageMatch(parse_message(msg), **spec), msg
@pytest.mark.parametrize(
"spec,msg,error_string",
[
pytest.param(spec, msg, error_string, id=error_string)
for (spec, _, negative_matches, error_stringgexps) in MESSAGE_SPECS
for (msg, error_string) in zip(negative_matches, error_stringgexps)
],
)
def test_message_matching_negative_message(self, spec, msg, error_string):
self.assertIn(error_string, self.messageDiffers(parse_message(msg), **spec))

View File

@ -1,3 +1,8 @@
"""
`Draft IRCv3 account-registration
<https://ircv3.net/specs/extensions/account-registration>`_
"""
from irctest import cases from irctest import cases
from irctest.patma import ANYSTR from irctest.patma import ANYSTR

View File

@ -1,12 +1,12 @@
""" """
<http://ircv3.net/specs/extensions/account-tag-3.2.html> `IRCv3 account-tag <https://ircv3.net/specs/extensions/account-tag>`_
""" """
from irctest import cases from irctest import cases
@cases.mark_services @cases.mark_services
class AccountTagTestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class AccountTagTestCase(cases.BaseServerTestCase):
def connectRegisteredClient(self, nick): def connectRegisteredClient(self, nick):
self.addClient() self.addClient()
self.sendLine(2, "CAP LS 302") self.sendLine(2, "CAP LS 302")
@ -40,7 +40,7 @@ class AccountTagTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
self.skipToWelcome(2) self.skipToWelcome(2)
@cases.mark_capabilities("account-tag") @cases.mark_capabilities("account-tag")
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
def testPrivmsg(self): def testPrivmsg(self):
self.connectClient("foo", capabilities=["account-tag"], skip_if_cap_nak=True) self.connectClient("foo", capabilities=["account-tag"], skip_if_cap_nak=True)
self.getMessages(1) self.getMessages(1)
@ -54,7 +54,10 @@ class AccountTagTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
) )
@cases.mark_capabilities("account-tag") @cases.mark_capabilities("account-tag")
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
@cases.xfailIfSoftware(
["Charybdis"], "https://github.com/solanum-ircd/solanum/issues/166"
)
def testInvite(self): def testInvite(self):
self.connectClient("foo", capabilities=["account-tag"], skip_if_cap_nak=True) self.connectClient("foo", capabilities=["account-tag"], skip_if_cap_nak=True)
self.getMessages(1) self.getMessages(1)

View File

@ -1,5 +1,16 @@
"""
AWAY command (`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-4.1>`__,
`Modern <https://modern.ircdocs.horse/#away-message>`__)
"""
from irctest import cases from irctest import cases
from irctest.numerics import RPL_AWAY, RPL_NOWAWAY, RPL_UNAWAY, RPL_USERHOST from irctest.numerics import (
RPL_AWAY,
RPL_NOWAWAY,
RPL_UNAWAY,
RPL_USERHOST,
RPL_WHOISUSER,
)
from irctest.patma import StrRe from irctest.patma import StrRe
@ -32,7 +43,7 @@ class AwayTestCase(cases.BaseServerTestCase):
""" """
"The server acknowledges the change in away status by returning the "The server acknowledges the change in away status by returning the
`RPL_NOWAWAY` and `RPL_UNAWAY` numerics." `RPL_NOWAWAY` and `RPL_UNAWAY` numerics."
-- https://github.com/ircdocs/modern-irc/pull/100 -- https://modern.ircdocs.horse/#away-message
""" """
self.connectClient("bar") self.connectClient("bar")
self.sendLine(1, "AWAY :I'm not here right now") self.sendLine(1, "AWAY :I'm not here right now")
@ -48,7 +59,7 @@ class AwayTestCase(cases.BaseServerTestCase):
""" """
"Servers SHOULD notify clients when a user they're interacting with "Servers SHOULD notify clients when a user they're interacting with
is away when relevant" is away when relevant"
-- https://github.com/ircdocs/modern-irc/pull/100 -- https://modern.ircdocs.horse/#away-message
"<client> <nick> :<message>" "<client> <nick> :<message>"
-- https://modern.ircdocs.horse/#rplaway-301 -- https://modern.ircdocs.horse/#rplaway-301
@ -75,7 +86,7 @@ class AwayTestCase(cases.BaseServerTestCase):
""" """
"Servers SHOULD notify clients when a user they're interacting with "Servers SHOULD notify clients when a user they're interacting with
is away when relevant" is away when relevant"
-- https://github.com/ircdocs/modern-irc/pull/100 -- https://modern.ircdocs.horse/#away-message
"<client> <nick> :<message>" "<client> <nick> :<message>"
-- https://modern.ircdocs.horse/#rplaway-301 -- https://modern.ircdocs.horse/#rplaway-301
@ -113,7 +124,7 @@ class AwayTestCase(cases.BaseServerTestCase):
""" """
"Servers SHOULD notify clients when a user they're interacting with "Servers SHOULD notify clients when a user they're interacting with
is away when relevant" is away when relevant"
-- https://github.com/ircdocs/modern-irc/pull/100 -- https://modern.ircdocs.horse/#away-message
"<client> <nick> :<message>" "<client> <nick> :<message>"
-- https://modern.ircdocs.horse/#rplaway-301 -- https://modern.ircdocs.horse/#rplaway-301
@ -134,3 +145,33 @@ class AwayTestCase(cases.BaseServerTestCase):
self.assertMessageMatch( self.assertMessageMatch(
self.getMessage(2), command=RPL_USERHOST, params=["qux", StrRe(r"bar=-.*")] self.getMessage(2), command=RPL_USERHOST, params=["qux", StrRe(r"bar=-.*")]
) )
@cases.mark_specifications("Modern")
def testAwayEmptyMessage(self):
"""
"If [AWAY] is sent with a nonempty parameter (the 'away message')
then the user is set to be away. If this command is sent with no
parameters, or with the empty string as the parameter, the user is no
longer away."
-- https://modern.ircdocs.horse/#away-message
"""
self.connectClient("bar", name="bar")
self.connectClient("qux", name="qux")
self.sendLine("bar", "AWAY :I'm not here right now")
replies = self.getMessages("bar")
self.assertIn(RPL_NOWAWAY, [msg.command for msg in replies])
self.sendLine("qux", "WHOIS bar")
replies = self.getMessages("qux")
self.assertIn(RPL_WHOISUSER, [msg.command for msg in replies])
self.assertIn(RPL_AWAY, [msg.command for msg in replies])
# empty final parameter to AWAY is treated the same as no parameter,
# i.e., the client is considered to be no longer away
self.sendLine("bar", "AWAY :")
replies = self.getMessages("bar")
self.assertIn(RPL_UNAWAY, [msg.command for msg in replies])
self.sendLine("qux", "WHOIS bar")
replies = self.getMessages("qux")
self.assertIn(RPL_WHOISUSER, [msg.command for msg in replies])
self.assertNotIn(RPL_AWAY, [msg.command for msg in replies])

View File

@ -1,11 +1,11 @@
""" """
<https://ircv3.net/specs/extensions/away-notify-3.1> `IRCv3 away-notify <https://ircv3.net/specs/extensions/away-notify>`_
""" """
from irctest import cases from irctest import cases
class AwayNotifyTestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class AwayNotifyTestCase(cases.BaseServerTestCase):
@cases.mark_capabilities("away-notify") @cases.mark_capabilities("away-notify")
def testAwayNotify(self): def testAwayNotify(self):
"""Basic away-notify test.""" """Basic away-notify test."""

View File

@ -1,6 +1,5 @@
""" """
Draft bot mode specification, as defined in `IRCv3 bot mode <https://ircv3.net/specs/extensions/bot-mode>`_
<https://ircv3.net/specs/extensions/bot-mode>
""" """
from irctest import cases, runner from irctest import cases, runner
@ -68,6 +67,10 @@ class BotModeTestCase(cases.BaseServerTestCase):
message, command=RPL_WHOISBOT, params=["usernick", "botnick", ANYSTR] message, command=RPL_WHOISBOT, params=["usernick", "botnick", ANYSTR]
) )
@cases.xfailIfSoftware(
["InspIRCd"],
"Uses only vendor tags for now: https://github.com/inspircd/inspircd/pull/1910",
)
def testBotPrivateMessage(self): def testBotPrivateMessage(self):
self._initBot() self._initBot()
@ -82,9 +85,13 @@ class BotModeTestCase(cases.BaseServerTestCase):
self.getMessage("user"), self.getMessage("user"),
command="PRIVMSG", command="PRIVMSG",
params=["usernick", "beep boop"], params=["usernick", "beep boop"],
tags={"draft/bot": None, **ANYDICT}, tags={StrRe("(draft/)?bot"): None, **ANYDICT},
) )
@cases.xfailIfSoftware(
["InspIRCd"],
"Uses only vendor tags for now: https://github.com/inspircd/inspircd/pull/1910",
)
def testBotChannelMessage(self): def testBotChannelMessage(self):
self._initBot() self._initBot()
@ -104,7 +111,7 @@ class BotModeTestCase(cases.BaseServerTestCase):
self.getMessage("user"), self.getMessage("user"),
command="PRIVMSG", command="PRIVMSG",
params=["#chan", "beep boop"], params=["#chan", "beep boop"],
tags={"draft/bot": None, **ANYDICT}, tags={StrRe("(draft/)?bot"): None, **ANYDICT},
) )
def testBotWhox(self): def testBotWhox(self):

View File

@ -1,3 +1,9 @@
"""
`Ergo <https://ergo.chat/>`_-specific tests of
`multiclient features
<https://github.com/ergochat/ergo/blob/master/docs/MANUAL.md#multiclient-bouncer>`_
"""
from irctest import cases from irctest import cases
from irctest.irc_utils.sasl import sasl_plain_blob from irctest.irc_utils.sasl import sasl_plain_blob
from irctest.numerics import ERR_NICKNAMEINUSE, RPL_WELCOME from irctest.numerics import ERR_NICKNAMEINUSE, RPL_WELCOME

View File

@ -1,5 +1,7 @@
"""Sends packets with various length to check the server reassembles them """
correctly. Also checks truncation""" Sends packets with various length to check the server reassembles them
correctly. Also checks truncation.
"""
import socket import socket
import time import time
@ -30,6 +32,16 @@ def _sendBytePerByte(self, line):
class BufferingTestCase(cases.BaseServerTestCase): class BufferingTestCase(cases.BaseServerTestCase):
@cases.xfailIfSoftware(
["Bahamut"],
"cannot pass because of issues with UTF-8 handling: "
"https://github.com/DALnet/bahamut/issues/196",
)
@cases.xfailIfSoftware(
["ircu2", "Nefarious", "snircd"],
"ircu2 discards the whole buffer on long lines "
"(TODO: refine how we exclude these tests)",
)
@pytest.mark.parametrize( @pytest.mark.parametrize(
"sender_function,colon", "sender_function,colon",
[ [

View File

@ -1,9 +1,14 @@
"""
`IRCv3 Capability negotiation
<https://ircv3.net/specs/extensions/capability-negotiation>`_
"""
from irctest import cases from irctest import cases
from irctest.patma import ANYSTR from irctest.patma import ANYSTR
from irctest.runner import CapabilityNotSupported, ImplementationChoice from irctest.runner import CapabilityNotSupported, ImplementationChoice
class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class CapTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
def testNoReq(self): def testNoReq(self):
"""Test the server handles gracefully clients which do not send """Test the server handles gracefully clients which do not send
@ -73,6 +78,10 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
) )
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.xfailIfSoftware(
["UnrealIRCd"],
"UnrealIRCd sends a trailing space on CAP NAK: https://github.com/unrealircd/unrealircd/pull/148",
)
def testNakWhole(self): def testNakWhole(self):
"""“The capability identifier set must be accepted as a whole, or """“The capability identifier set must be accepted as a whole, or
rejected entirely.” rejected entirely.”
@ -120,6 +129,10 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
) )
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.xfailIfSoftware(
["UnrealIRCd"],
"UnrealIRCd sends a trailing space on CAP NAK: https://github.com/unrealircd/unrealircd/pull/148",
)
def testCapRemovalByClient(self): def testCapRemovalByClient(self):
"""Test CAP LIST and removal of caps via CAP REQ :-tagname.""" """Test CAP LIST and removal of caps via CAP REQ :-tagname."""
cap1 = "echo-message" cap1 = "echo-message"
@ -172,3 +185,60 @@ class CapTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
enabled_caps.discard("cap-notify") # implicitly added by some impls enabled_caps.discard("cap-notify") # implicitly added by some impls
self.assertEqual(enabled_caps, {cap1}) self.assertEqual(enabled_caps, {cap1})
self.assertNotIn("time", cap_list.tags) self.assertNotIn("time", cap_list.tags)
@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}",
)

View File

@ -1,3 +1,7 @@
"""
Channel casemapping
"""
import pytest import pytest
from irctest import cases, client_mock, runner from irctest import cases, client_mock, runner
@ -18,7 +22,7 @@ class ChannelCaseSensitivityTestCase(cases.BaseServerTestCase):
self.connectClient("foo") self.connectClient("foo")
self.connectClient("bar") self.connectClient("bar")
if self.server_support["CASEMAPPING"] != casemapping: if self.server_support["CASEMAPPING"] != casemapping:
raise runner.NotImplementedByController( raise runner.ImplementationChoice(
"Casemapping {} not implemented".format(casemapping) "Casemapping {} not implemented".format(casemapping)
) )
self.joinClient(1, name1) self.joinClient(1, name1)
@ -43,7 +47,7 @@ class ChannelCaseSensitivityTestCase(cases.BaseServerTestCase):
self.connectClient("foo") self.connectClient("foo")
self.connectClient("bar") self.connectClient("bar")
if self.server_support["CASEMAPPING"] != casemapping: if self.server_support["CASEMAPPING"] != casemapping:
raise runner.NotImplementedByController( raise runner.ImplementationChoice(
"Casemapping {} not implemented".format(casemapping) "Casemapping {} not implemented".format(casemapping)
) )
self.joinClient(1, name1) self.joinClient(1, name1)

View File

@ -1,3 +1,9 @@
"""
`Ergo <https://ergo.chat/>`_-specific tests of channel forwarding
TODO: Should be extended to other servers, once a specification is written.
"""
from irctest import cases from irctest import cases
from irctest.numerics import ERR_CHANOPRIVSNEEDED, ERR_INVALIDMODEPARAM, ERR_LINKCHANNEL from irctest.numerics import ERR_CHANOPRIVSNEEDED, ERR_INVALIDMODEPARAM, ERR_LINKCHANNEL

View File

@ -1,24 +1,22 @@
"""
`Draft IRCv3 channel-rename <https://ircv3.net/specs/extensions/channel-rename>`_
"""
from irctest import cases from irctest import cases
from irctest.numerics import ERR_CHANOPRIVSNEEDED from irctest.numerics import ERR_CHANOPRIVSNEEDED
MODERN_CAPS = [
"server-time",
"message-tags",
"batch",
"labeled-response",
"echo-message",
"account-tag",
]
RENAME_CAP = "draft/channel-rename" RENAME_CAP = "draft/channel-rename"
@cases.mark_specifications("IRCv3")
class ChannelRenameTestCase(cases.BaseServerTestCase): class ChannelRenameTestCase(cases.BaseServerTestCase):
"""Basic tests for channel-rename.""" """Basic tests for channel-rename."""
@cases.mark_specifications("Ergo")
def testChannelRename(self): def testChannelRename(self):
self.connectClient("bar", name="bar", capabilities=MODERN_CAPS + [RENAME_CAP]) self.connectClient(
self.connectClient("baz", name="baz", capabilities=MODERN_CAPS) "bar", name="bar", capabilities=[RENAME_CAP], skip_if_cap_nak=True
)
self.connectClient("baz", name="baz")
self.joinChannel("bar", "#bar") self.joinChannel("bar", "#bar")
self.joinChannel("baz", "#bar") self.joinChannel("baz", "#bar")
self.getMessages("bar") self.getMessages("bar")

View File

@ -1,9 +1,14 @@
"""
`IRCv3 draft chathistory <https://ircv3.net/specs/extensions/chathistory>`_
"""
import functools
import secrets import secrets
import time import time
import pytest import pytest
from irctest import cases from irctest import cases, runner
from irctest.irc_utils.junkdrawer import random_name from irctest.irc_utils.junkdrawer import random_name
from irctest.patma import ANYSTR from irctest.patma import ANYSTR
@ -38,6 +43,16 @@ def validate_chathistory_batch(msgs):
return result return result
def skip_ngircd(f):
@functools.wraps(f)
def newf(self, *args, **kwargs):
if self.controller.software_name == "ngIRCd":
raise runner.OptionalExtensionNotSupported("nicks longer 9 characters")
return f(self, *args, **kwargs)
return newf
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.mark_services @cases.mark_services
class ChathistoryTestCase(cases.BaseServerTestCase): class ChathistoryTestCase(cases.BaseServerTestCase):
@ -45,6 +60,7 @@ class ChathistoryTestCase(cases.BaseServerTestCase):
def config() -> cases.TestCaseControllerConfig: def config() -> cases.TestCaseControllerConfig:
return cases.TestCaseControllerConfig(chathistory=True) return cases.TestCaseControllerConfig(chathistory=True)
@skip_ngircd
def testInvalidTargets(self): def testInvalidTargets(self):
bar, pw = random_name("bar"), random_name("pw") bar, pw = random_name("bar"), random_name("pw")
self.controller.registerUser(self, bar, pw) self.controller.registerUser(self, bar, pw)
@ -90,6 +106,7 @@ class ChathistoryTestCase(cases.BaseServerTestCase):
) )
@pytest.mark.private_chathistory @pytest.mark.private_chathistory
@skip_ngircd
def testMessagesToSelf(self): def testMessagesToSelf(self):
bar, pw = random_name("bar"), random_name("pw") bar, pw = random_name("bar"), random_name("pw")
self.controller.registerUser(self, bar, pw) self.controller.registerUser(self, bar, pw)
@ -162,7 +179,19 @@ class ChathistoryTestCase(cases.BaseServerTestCase):
self.assertEqual(len(set(msg.time for msg in echo_messages)), num_messages) self.assertEqual(len(set(msg.time for msg in echo_messages)), num_messages)
@pytest.mark.parametrize("subcommand", SUBCOMMANDS) @pytest.mark.parametrize("subcommand", SUBCOMMANDS)
@skip_ngircd
def testChathistory(self, subcommand): def testChathistory(self, subcommand):
if subcommand == "BETWEEN" and self.controller.software_name == "UnrealIRCd":
pytest.xfail(
"CHATHISTORY BETWEEN does not apply bounds correct "
"https://bugs.unrealircd.org/view.php?id=5952"
)
if subcommand == "AROUND" and self.controller.software_name == "UnrealIRCd":
pytest.xfail(
"CHATHISTORY AROUND excludes 'central' messages "
"https://bugs.unrealircd.org/view.php?id=5953"
)
self.connectClient( self.connectClient(
"bar", "bar",
capabilities=[ capabilities=[
@ -194,6 +223,7 @@ class ChathistoryTestCase(cases.BaseServerTestCase):
self.validate_chathistory(subcommand, echo_messages, 1, chname) self.validate_chathistory(subcommand, echo_messages, 1, chname)
@pytest.mark.parametrize("subcommand", SUBCOMMANDS) @pytest.mark.parametrize("subcommand", SUBCOMMANDS)
@skip_ngircd
def testChathistoryEventPlayback(self, subcommand): def testChathistoryEventPlayback(self, subcommand):
self.connectClient( self.connectClient(
"bar", "bar",
@ -227,6 +257,7 @@ class ChathistoryTestCase(cases.BaseServerTestCase):
@pytest.mark.parametrize("subcommand", SUBCOMMANDS) @pytest.mark.parametrize("subcommand", SUBCOMMANDS)
@pytest.mark.private_chathistory @pytest.mark.private_chathistory
@skip_ngircd
def testChathistoryDMs(self, subcommand): def testChathistoryDMs(self, subcommand):
c1 = "foo" + secrets.token_hex(12) c1 = "foo" + secrets.token_hex(12)
c2 = "bar" + secrets.token_hex(12) c2 = "bar" + secrets.token_hex(12)
@ -549,6 +580,7 @@ class ChathistoryTestCase(cases.BaseServerTestCase):
self.assertIn(echo_messages[7], result) self.assertIn(echo_messages[7], result)
@pytest.mark.arbitrary_client_tags @pytest.mark.arbitrary_client_tags
@skip_ngircd
def testChathistoryTagmsg(self): def testChathistoryTagmsg(self):
c1 = "foo" + secrets.token_hex(12) c1 = "foo" + secrets.token_hex(12)
c2 = "bar" + secrets.token_hex(12) c2 = "bar" + secrets.token_hex(12)
@ -647,6 +679,7 @@ class ChathistoryTestCase(cases.BaseServerTestCase):
@pytest.mark.arbitrary_client_tags @pytest.mark.arbitrary_client_tags
@pytest.mark.private_chathistory @pytest.mark.private_chathistory
@skip_ngircd
def testChathistoryDMClientOnlyTags(self): def testChathistoryDMClientOnlyTags(self):
# regression test for Ergo #1411 # regression test for Ergo #1411
c1 = "foo" + secrets.token_hex(12) c1 = "foo" + secrets.token_hex(12)

View File

@ -1,3 +1,9 @@
"""
`Ergo <https://ergo.chat/>`_-specific tests of auditorium mode
TODO: Should be extended to other servers, once a specification is written.
"""
import math import math
import time import time

View File

@ -1,11 +1,24 @@
from irctest import cases """
from irctest.numerics import ERR_BANNEDFROMCHAN Channel ban (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.3.1>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.3>`__,
`Modern <https://modern.ircdocs.horse/#ban-channel-mode>`__)
and ban exception (`Modern <https://modern.ircdocs.horse/#exception-channel-mode>`__)
"""
from irctest import cases, runner
from irctest.numerics import (
ERR_BANNEDFROMCHAN,
ERR_CANNOTSENDTOCHAN,
RPL_BANLIST,
RPL_ENDOFBANLIST,
)
from irctest.patma import ANYSTR, StrRe
class BanModeTestCase(cases.BaseServerTestCase): class BanModeTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("RFC1459", "RFC2812") @cases.mark_specifications("RFC1459", "RFC2812", "Modern")
def testBan(self): def testBanJoin(self):
"""Basic ban operation"""
self.connectClient("chanop", name="chanop") self.connectClient("chanop", name="chanop")
self.joinChannel("chanop", "#chan") self.joinChannel("chanop", "#chan")
self.getMessages("chanop") self.getMessages("chanop")
@ -23,6 +36,160 @@ class BanModeTestCase(cases.BaseServerTestCase):
self.sendLine("bar", "JOIN #chan") self.sendLine("bar", "JOIN #chan")
self.assertMessageMatch(self.getMessage("bar"), command="JOIN") self.assertMessageMatch(self.getMessage("bar"), command="JOIN")
@cases.mark_specifications("Modern")
def testBanPrivmsg(self):
"""
TODO: this checks the following quote is false:
"If `<target>` is a channel name and the client is [banned](#ban-channel-mode)
and not covered by a [ban exception](#ban-exception-channel-mode), the
message will not be delivered and the command will silently fail."
-- https://modern.ircdocs.horse/#privmsg-message
to check https://github.com/ircdocs/modern-irc/pull/201
"""
self.connectClient("chanop", name="chanop")
self.joinChannel("chanop", "#chan")
self.getMessages("chanop")
self.connectClient("Bar", name="bar")
self.getMessages("bar")
self.sendLine("bar", "JOIN #chan")
self.getMessages("bar")
self.getMessages("chanop")
self.sendLine("chanop", "MODE #chan +b bar!*@*")
self.assertMessageMatch(self.getMessage("chanop"), command="MODE")
self.getMessages("chanop")
self.getMessages("bar")
self.sendLine("bar", "PRIVMSG #chan :hello world")
self.assertMessageMatch(
self.getMessage("bar"),
command=ERR_CANNOTSENDTOCHAN,
params=["Bar", "#chan", ANYSTR],
)
self.assertEqual(self.getMessages("bar"), [])
self.assertEqual(self.getMessages("chanop"), [])
self.sendLine("chanop", "MODE #chan -b bar!*@*")
self.assertMessageMatch(self.getMessage("chanop"), command="MODE")
self.getMessages("chanop")
self.getMessages("bar")
self.sendLine("bar", "PRIVMSG #chan :hello again")
self.assertEqual(self.getMessages("bar"), [])
self.assertMessageMatch(
self.getMessage("chanop"),
command="PRIVMSG",
params=["#chan", "hello again"],
)
@cases.mark_specifications("Modern")
def testBanList(self):
"""`RPL_BANLIST <https://modern.ircdocs.horse/#rplbanlist-367>`_"""
self.connectClient("chanop")
self.joinChannel(1, "#chan")
self.getMessages(1)
self.sendLine(1, "MODE #chan +b bar!*@*")
self.assertMessageMatch(self.getMessage(1), command="MODE")
self.sendLine(1, "MODE #chan +b")
m = self.getMessage(1)
if len(m.params) == 3:
# Old format
self.assertMessageMatch(
m,
command=RPL_BANLIST,
params=[
"chanop",
"#chan",
"bar!*@*",
],
)
else:
self.assertMessageMatch(
m,
command=RPL_BANLIST,
params=[
"chanop",
"#chan",
"bar!*@*",
StrRe("chanop(!.*@.*)?"),
StrRe("[0-9]+"),
],
)
self.assertMessageMatch(
self.getMessage(1),
command=RPL_ENDOFBANLIST,
params=[
"chanop",
"#chan",
ANYSTR,
],
)
@cases.mark_specifications("Modern")
def testBanException(self):
"""`Exception mode <https://modern.ircdocs.horse/#exception-channel-mode`_,
detected using `ISUPPORT EXCEPTS
<https://modern.ircdocs.horse/#excepts-parameter>`_ and checked against
`ISUPPORT CHANMODES <https://modern.ircdocs.horse/#chanmodes-parameter>`_"""
self.connectClient("chanop", name="chanop")
if "EXCEPTS" in self.server_support:
mode = self.server_support["EXCEPTS"] or "e"
if "CHANMODES" in self.server_support:
self.assertIn(
mode,
self.server_support["CHANMODES"],
fail_msg="ISUPPORT EXCEPTS is present, but '{item}' is missing "
"from 'CHANMODES={list}'",
)
self.assertIn(
mode,
self.server_support["CHANMODES"].split(",")[0],
fail_msg="ISUPPORT EXCEPTS is present, but '{item}' is not "
"in group A",
)
else:
mode = "e"
if "CHANMODES" in self.server_support:
if "e" not in self.server_support["CHANMODES"]:
raise runner.OptionalExtensionNotSupported(
"Ban exception (or mode letter is not +e)"
)
self.assertIn(
mode,
self.server_support["CHANMODES"].split(",")[0],
fail_msg="Mode +e (assumed to be ban exception) is present, "
"but 'e' is not in group A",
)
else:
raise runner.OptionalExtensionNotSupported("ISUPPORT CHANMODES")
self.sendLine("chanop", "JOIN #chan")
self.getMessages("chanop")
self.sendLine("chanop", "MODE #chan +b ba*!*@*")
self.getMessages("chanop")
# banned client cannot join
self.connectClient("Bar", name="bar")
self.sendLine("bar", "JOIN #chan")
self.assertMessageMatch(self.getMessage("bar"), command=ERR_BANNEDFROMCHAN)
# chanop sets exception
self.sendLine("chanop", "MODE #chan +e *ar!*@*")
self.assertMessageMatch(self.getMessage("chanop"), command="MODE")
# client can now join
self.sendLine("bar", "JOIN #chan")
self.assertMessageMatch(self.getMessage("bar"), command="JOIN")
# TODO: Add testBanExceptionList, once the numerics are specified in Modern
@cases.mark_specifications("Ergo") @cases.mark_specifications("Ergo")
def testCaseInsensitive(self): def testCaseInsensitive(self):
"""Some clients allow unsetting modes if their argument matches """Some clients allow unsetting modes if their argument matches

View File

@ -1,3 +1,7 @@
"""
Various Ergo-specific channel modes
"""
from irctest import cases from irctest import cases
from irctest.numerics import ERR_CANNOTSENDTOCHAN, ERR_CHANOPRIVSNEEDED from irctest.numerics import ERR_CANNOTSENDTOCHAN, ERR_CHANOPRIVSNEEDED

View File

@ -1,3 +1,10 @@
"""
Channel key (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.3.1>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.3>`__,
`Modern <https://modern.ircdocs.horse/#key-channel-mode>`__)
"""
import pytest import pytest
from irctest import cases from irctest import cases
@ -20,10 +27,16 @@ class KeyTestCase(cases.BaseServerTestCase):
self.connectClient("qux") self.connectClient("qux")
self.getMessages(2) self.getMessages(2)
# JOIN with a missing key MUST receive ERR_BADCHANNELKEY:
self.sendLine(2, "JOIN #chan") self.sendLine(2, "JOIN #chan")
reply = self.getMessages(2) reply_cmds = {msg.command for msg in self.getMessages(2)}
self.assertNotIn("JOIN", {msg.command for msg in reply}) self.assertNotIn("JOIN", reply_cmds)
self.assertIn(ERR_BADCHANNELKEY, {msg.command for msg in reply}) self.assertIn(ERR_BADCHANNELKEY, reply_cmds)
# similarly for JOIN with an incorrect key:
self.sendLine(2, "JOIN #chan bees")
reply_cmds = {msg.command for msg in self.getMessages(2)}
self.assertNotIn("JOIN", reply_cmds)
self.assertIn(ERR_BADCHANNELKEY, reply_cmds)
self.sendLine(2, "JOIN #chan beer") self.sendLine(2, "JOIN #chan beer")
reply = self.getMessages(2) reply = self.getMessages(2)
@ -57,6 +70,21 @@ class KeyTestCase(cases.BaseServerTestCase):
-- https://modern.ircdocs.horse/#key-channel-mode -- https://modern.ircdocs.horse/#key-channel-mode
-- https://github.com/ircdocs/modern-irc/pull/111 -- https://github.com/ircdocs/modern-irc/pull/111
""" """
if key == "" and self.controller.software_name in (
"ircu2",
"Nefarious",
"snircd",
):
pytest.xfail(
"ircu2 returns ERR_NEEDMOREPARAMS on empty keys: "
"https://github.com/UndernetIRC/ircu2/issues/13"
)
if (key == "" or " " in key) and self.controller.software_name == "ngIRCd":
pytest.xfail(
"ngIRCd does not validate channel keys: "
"https://github.com/ngircd/ngircd/issues/290"
)
self.connectClient("bar") self.connectClient("bar")
self.joinChannel(1, "#chan") self.joinChannel(1, "#chan")
self.sendLine(1, f"MODE #chan +k :{key}") self.sendLine(1, f"MODE #chan +k :{key}")

View File

@ -1,3 +1,9 @@
"""
Channel moderation mode (`RFC 2812
<https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.3>`__,
`Modern <https://modern.ircdocs.horse/#ban-channel-mode>`__)
"""
from irctest import cases from irctest import cases
from irctest.numerics import ERR_CANNOTSENDTOCHAN from irctest.numerics import ERR_CANNOTSENDTOCHAN

View File

@ -1,3 +1,7 @@
"""
Mute extban, currently no specifications or ways to discover it.
"""
from irctest import cases, runner from irctest import cases, runner
from irctest.numerics import ERR_CANNOTSENDTOCHAN, ERR_CHANOPRIVSNEEDED from irctest.numerics import ERR_CANNOTSENDTOCHAN, ERR_CHANOPRIVSNEEDED
from irctest.patma import ANYLIST, StrRe from irctest.patma import ANYLIST, StrRe
@ -16,7 +20,7 @@ class MuteExtbanTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("Ergo") @cases.mark_specifications("Ergo")
def testISupport(self): def testISupport(self):
self.connectClient(1) # Fetches ISUPPORT self.connectClient("chk") # Fetches ISUPPORT
isupport = self.server_support isupport = self.server_support
token = isupport["EXTBAN"] token = isupport["EXTBAN"]
prefix, comma, types = token.partition(",") prefix, comma, types = token.partition(",")
@ -194,7 +198,7 @@ class MuteExtbanTestCase(cases.BaseServerTestCase):
self.getMessages(client) self.getMessages(client)
# +e grants an exemption to +b # +e grants an exemption to +b
self.sendLine("chanop", f"MODE #chan +e {prefix}{self.char()}:*!~evan@*") self.sendLine("chanop", f"MODE #chan +e {prefix}{self.char()}:*!*evan@*")
replies = {msg.command for msg in self.getMessages("chanop")} replies = {msg.command for msg in self.getMessages("chanop")}
self.assertIn("MODE", replies) self.assertIn("MODE", replies)
self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies) self.assertNotIn(ERR_CHANOPRIVSNEEDED, replies)

View File

@ -0,0 +1,62 @@
"""
Channel secrecy mode (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.3.1>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.3>`__,
`Modern <https://modern.ircdocs.horse/#secret-channel-mode>`__)
"""
from irctest import cases
from irctest.numerics import RPL_LIST
class SecretChannelTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("RFC1459", "Modern")
def testSecretChannelListCommand(self):
"""
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.6>
"Likewise, secret channels are not listed
at all unless the client is a member of the channel in question."
<https://modern.ircdocs.horse/#secret-channel-mode>
"A channel that is set to secret will not show up in responses to
the LIST or NAMES command unless the client sending the command is
joined to the channel."
"""
def get_listed_channels(replies):
channels = set()
for reply in replies:
# skip pseudo-channels (&SERVER, &NOTICES) listed by ngircd
# and ircu:
if reply.command == RPL_LIST and reply.params[1].startswith("#"):
channels.add(reply.params[1])
return channels
# test that a silent channel is shown in list if the user is in the channel.
self.connectClient("first", name="first")
self.joinChannel("first", "#gen")
self.getMessages("first")
self.sendLine("first", "MODE #gen +s")
# run command LIST
self.sendLine("first", "LIST")
replies = self.getMessages("first")
self.assertEqual(get_listed_channels(replies), {"#gen"})
# test that another client would not see the secret
# channel.
self.connectClient("second", name="second")
self.getMessages("second")
self.sendLine("second", "LIST")
replies = self.getMessages("second")
# RPL_LIST 322 should NOT be present this time.
self.assertEqual(get_listed_channels(replies), set())
# Second client will join the secret channel
# and call command LIST. The channel SHOULD
# appear this time.
self.joinChannel("second", "#gen")
self.sendLine("second", "LIST")
replies = self.getMessages("second")
# Should be only one line with command RPL_LIST
self.assertEqual(get_listed_channels(replies), {"#gen"})

View File

@ -1,3 +1,8 @@
"""
`Ergo <https://ergo.chat/>`_-specific tests for nick collisions based on Unicode
confusable characters
"""
from irctest import cases from irctest import cases
from irctest.numerics import ERR_NICKNAMEINUSE, RPL_WELCOME from irctest.numerics import ERR_NICKNAMEINUSE, RPL_WELCOME

View File

@ -1,11 +1,13 @@
""" """
Tests section 4.1 of RFC 1459. Tests section 4.1 of RFC 1459.
<https://tools.ietf.org/html/rfc1459#section-4.1> <https://tools.ietf.org/html/rfc1459#section-4.1>
TODO: cross-reference Modern and RFC 2812 too
""" """
from irctest import cases from irctest import cases
from irctest.client_mock import ConnectionClosed from irctest.client_mock import ConnectionClosed
from irctest.numerics import ERR_NEEDMOREPARAMS from irctest.numerics import ERR_NEEDMOREPARAMS, ERR_PASSWDMISMATCH
from irctest.patma import ANYSTR, StrRe from irctest.patma import ANYSTR, StrRe
@ -36,8 +38,14 @@ class PasswordedConnectionRegistrationTestCase(cases.BaseServerTestCase):
m.command, "001", msg="Got 001 after NICK+USER but missing PASS" m.command, "001", msg="Got 001 after NICK+USER but missing PASS"
) )
@cases.mark_specifications("RFC1459", "RFC2812") @cases.mark_specifications("Modern")
def testWrongPassword(self): def testWrongPassword(self):
"""
"If the password supplied does not match the password expected by the server,
then the server SHOULD send ERR_PASSWDMISMATCH and MUST close the connection
with ERROR."
-- https://github.com/ircdocs/modern-irc/pull/172
"""
self.addClient() self.addClient()
self.sendLine(1, "PASS {}".format(self.password + "garbage")) self.sendLine(1, "PASS {}".format(self.password + "garbage"))
self.sendLine(1, "NICK foo") self.sendLine(1, "NICK foo")
@ -46,6 +54,13 @@ class PasswordedConnectionRegistrationTestCase(cases.BaseServerTestCase):
self.assertNotEqual( self.assertNotEqual(
m.command, "001", msg="Got 001 after NICK+USER but incorrect PASS" m.command, "001", msg="Got 001 after NICK+USER but incorrect PASS"
) )
self.assertIn(m.command, {ERR_PASSWDMISMATCH, "ERROR"})
if m.command == "ERR_PASSWDMISMATCH":
m = self.getRegistrationMessage(1)
self.assertEqual(
m.command, "ERROR", msg="ERR_PASSWDMISMATCH not followed by ERROR."
)
@cases.mark_specifications("RFC1459", "RFC2812", strict=True) @cases.mark_specifications("RFC1459", "RFC2812", strict=True)
def testPassAfterNickuser(self): def testPassAfterNickuser(self):
@ -82,6 +97,10 @@ class ConnectionRegistrationTestCase(cases.BaseServerTestCase):
self.getMessages(1) self.getMessages(1)
@cases.mark_specifications("RFC2812") @cases.mark_specifications("RFC2812")
@cases.xfailIfSoftware(["Charybdis", "Solanum"], "very flaky")
@cases.xfailIfSoftware(
["ircu2", "Nefarious", "snircd"], "ircu2 does not send ERROR"
)
def testQuitErrors(self): def testQuitErrors(self):
"""“A client session is terminated with a quit message. The server """“A client session is terminated with a quit message. The server
acknowledges this by sending an ERROR message to the client.” acknowledges this by sending an ERROR message to the client.”
@ -162,6 +181,10 @@ class ConnectionRegistrationTestCase(cases.BaseServerTestCase):
"neither got 001.", "neither got 001.",
) )
@cases.xfailIfSoftware(
["ircu2", "Nefarious", "ngIRCd"],
"uses a default value instead of ERR_NEEDMOREPARAMS",
)
def testEmptyRealname(self): def testEmptyRealname(self):
""" """
Syntax: Syntax:
@ -183,60 +206,3 @@ class ConnectionRegistrationTestCase(cases.BaseServerTestCase):
command=ERR_NEEDMOREPARAMS, command=ERR_NEEDMOREPARAMS,
params=[StrRe(r"(\*|foo)"), "USER", ANYSTR], params=[StrRe(r"(\*|foo)"), "USER", ANYSTR],
) )
@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}",
)

View File

@ -1,11 +1,10 @@
""" """
<http://ircv3.net/specs/extensions/echo-message-3.2.html> `IRCv3 echo-message <https://ircv3.net/specs/extensions/echo-message>`_
""" """
import pytest import pytest
from irctest import cases from irctest import cases
from irctest.basecontrollers import NotImplementedByController
from irctest.irc_utils.junkdrawer import random_name from irctest.irc_utils.junkdrawer import random_name
from irctest.patma import ANYDICT from irctest.patma import ANYDICT
@ -23,31 +22,18 @@ class EchoMessageTestCase(cases.BaseServerTestCase):
@cases.mark_capabilities("echo-message") @cases.mark_capabilities("echo-message")
def testEchoMessage(self, command, solo, server_time): def testEchoMessage(self, command, solo, server_time):
"""<http://ircv3.net/specs/extensions/echo-message-3.2.html>""" """<http://ircv3.net/specs/extensions/echo-message-3.2.html>"""
self.addClient() if server_time:
self.sendLine(1, "CAP LS 302") self.connectClient(
capabilities = self.getCapLs(1) "baz",
if "echo-message" not in capabilities: capabilities=["echo-message", "server-time"],
raise NotImplementedByController("echo-message") skip_if_cap_nak=True,
if server_time and "server-time" not in capabilities: )
raise NotImplementedByController("server-time") else:
self.connectClient(
# TODO: check also without this "baz",
self.sendLine( capabilities=["echo-message", "server-time"],
1, skip_if_cap_nak=True,
"CAP REQ :echo-message{}".format(" server-time" if server_time else ""), )
)
self.getRegistrationMessage(1)
# TODO: Remove this one the trailing space issue is fixed in Charybdis
# and Mammon:
# self.assertMessageMatch(m, command='CAP',
# params=['*', 'ACK', 'echo-message'] +
# (['server-time'] if server_time else []),
# fail_msg='Did not ACK advertised capabilities: {msg}')
self.sendLine(1, "USER f * * :foo")
self.sendLine(1, "NICK baz")
self.sendLine(1, "CAP END")
self.skipToWelcome(1)
self.getMessages(1)
self.sendLine(1, "JOIN #chan") self.sendLine(1, "JOIN #chan")

View File

View File

@ -0,0 +1,29 @@
"""
`Ergo <https://ergo.chat/>`-specific tests of NickServ.
"""
from irctest import cases
from irctest.numerics import RPL_YOUREOPER
class NickservTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("Ergo")
def test_saregister(self):
self.connectClient("root", name="root")
self.sendLine("root", "OPER operuser operpassword")
self.assertIn(RPL_YOUREOPER, {msg.command for msg in self.getMessages("root")})
self.sendLine(
"root",
"PRIVMSG NickServ :SAREGISTER saregister_test saregistertestpassphrase",
)
self.getMessages("root")
# test that the account was registered
self.connectClient(
name="saregister_test",
nick="saregister_test",
capabilities=["sasl"],
account="saregister_test",
password="saregistertestpassphrase",
)

View File

@ -1,12 +1,12 @@
""" """
<http://ircv3.net/specs/extensions/extended-join-3.1.html> `IRCv3 extended-join <https://ircv3.net/specs/extensions/extended-join>`_
""" """
from irctest import cases from irctest import cases
@cases.mark_services @cases.mark_services
class MetadataTestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class MetadataTestCase(cases.BaseServerTestCase):
def connectRegisteredClient(self, nick): def connectRegisteredClient(self, nick):
self.addClient() self.addClient()
self.sendLine(2, "CAP LS 302") self.sendLine(2, "CAP LS 302")
@ -50,7 +50,7 @@ class MetadataTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
) )
@cases.mark_capabilities("extended-join") @cases.mark_capabilities("extended-join")
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
def testLoggedIn(self): def testLoggedIn(self):
self.connectClient("foo", capabilities=["extended-join"], skip_if_cap_nak=True) self.connectClient("foo", capabilities=["extended-join"], skip_if_cap_nak=True)
self.joinChannel(1, "#chan") self.joinChannel(1, "#chan")

View File

@ -0,0 +1,126 @@
"""
The HELP and HELPOP command (`Modern <https://modern.ircdocs.horse/#help-message>`__)
"""
import functools
import re
import pytest
from irctest import cases, runner
from irctest.numerics import (
ERR_HELPNOTFOUND,
ERR_UNKNOWNCOMMAND,
RPL_ENDOFHELP,
RPL_HELPSTART,
RPL_HELPTXT,
)
from irctest.patma import ANYSTR, StrRe
def with_xfails(f):
@functools.wraps(f)
def newf(self, command, *args, **kwargs):
if command == "HELP" and self.controller.software_name == "Bahamut":
raise runner.ImplementationChoice(
"fail because Bahamut forwards /HELP to HelpServ (but not /HELPOP)"
)
if self.controller.software_name in ("irc2", "ircu2", "ngIRCd"):
raise runner.ImplementationChoice(
"numerics in reply to /HELP and /HELPOP (uses NOTICE instead)"
)
if self.controller.software_name == "UnrealIRCd":
raise runner.ImplementationChoice(
"fails because Unreal uses custom numerics "
"https://github.com/unrealircd/unrealircd/pull/184"
)
return f(self, command, *args, **kwargs)
return newf
class HelpTestCase(cases.BaseServerTestCase):
def _assertValidHelp(self, messages, subject):
if subject != ANYSTR:
subject = StrRe("(?i)" + re.escape(subject))
self.assertMessageMatch(
messages[0],
command=RPL_HELPSTART,
params=["nick", subject, ANYSTR],
fail_msg=f"Expected {RPL_HELPSTART} (RPL_HELPSTART), got: {{msg}}",
)
self.assertMessageMatch(
messages[-1],
command=RPL_ENDOFHELP,
params=["nick", subject, ANYSTR],
fail_msg=f"Expected {RPL_ENDOFHELP} (RPL_ENDOFHELP), got: {{msg}}",
)
for i in range(1, len(messages) - 1):
self.assertMessageMatch(
messages[i],
command=RPL_HELPTXT,
params=["nick", subject, ANYSTR],
fail_msg=f"Expected {RPL_HELPTXT} (RPL_HELPTXT), got: {{msg}}",
)
@pytest.mark.parametrize("command", ["HELP", "HELPOP"])
@cases.mark_specifications("Modern")
@with_xfails
def testHelpNoArg(self, command):
self.connectClient("nick")
self.sendLine(1, f"{command}")
messages = self.getMessages(1)
if messages[0].command == ERR_UNKNOWNCOMMAND:
raise runner.OptionalCommandNotSupported(command)
self._assertValidHelp(messages, ANYSTR)
@pytest.mark.parametrize("command", ["HELP", "HELPOP"])
@cases.mark_specifications("Modern")
@with_xfails
def testHelpPrivmsg(self, command):
self.connectClient("nick")
self.sendLine(1, f"{command} PRIVMSG")
messages = self.getMessages(1)
if messages[0].command == ERR_UNKNOWNCOMMAND:
raise runner.OptionalCommandNotSupported(command)
self._assertValidHelp(messages, "PRIVMSG")
@pytest.mark.parametrize("command", ["HELP", "HELPOP"])
@cases.mark_specifications("Modern")
@with_xfails
def testHelpUnknownSubject(self, command):
self.connectClient("nick")
self.sendLine(1, f"{command} THISISNOTACOMMAND")
messages = self.getMessages(1)
if messages[0].command == ERR_UNKNOWNCOMMAND:
raise runner.OptionalCommandNotSupported(command)
if messages[0].command == ERR_HELPNOTFOUND:
# Inspircd, Hybrid et al
self.assertEqual(len(messages), 1)
self.assertMessageMatch(
messages[0],
command=ERR_HELPNOTFOUND,
params=[
"nick",
StrRe(
"(?i)THISISNOTACOMMAND"
), # case-insensitive, for Hybrid and Plexus4 (but not Chary et al)
ANYSTR,
],
)
else:
# Unrealircd
self._assertValidHelp(messages, ANYSTR)

View File

@ -0,0 +1,118 @@
"""
The INFO command (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.3.8>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.4.10>`__,
`Modern <https://modern.ircdocs.horse/#info-message>`__)
"""
import pytest
from irctest import cases
from irctest.numerics import ERR_NOSUCHSERVER, RPL_ENDOFINFO, RPL_INFO, RPL_YOUREOPER
from irctest.patma import ANYSTR
class InfoTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
def testInfo(self):
"""
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.3.8>
<https://datatracker.ietf.org/doc/html/rfc2812#section-3.4.10>
"Upon receiving an INFO command, the given server will respond with zero or
more RPL_INFO replies, followed by one RPL_ENDOFINFO numeric"
-- <https://modern.ircdocs.horse/#info-message>
"""
self.connectClient("nick")
# Remote /INFO is oper-only on Unreal and ircu2
self.sendLine(1, "OPER operuser operpassword")
self.assertIn(
RPL_YOUREOPER,
[m.command for m in self.getMessages(1)],
fail_msg="OPER failed",
)
self.sendLine(1, "INFO")
messages = self.getMessages(1)
last_message = messages.pop()
self.assertMessageMatch(
last_message, command=RPL_ENDOFINFO, params=["nick", ANYSTR]
)
for message in messages:
self.assertMessageMatch(message, command=RPL_INFO, params=["nick", ANYSTR])
@pytest.mark.parametrize(
"target",
["My.Little.Server", "*Little*", "nick"],
ids=["target-server", "target-wildcard", "target-nick"],
)
@cases.mark_specifications("RFC1459", "RFC2812", deprecated=True)
def testInfoTarget(self, target):
"""
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.3.8>
<https://datatracker.ietf.org/doc/html/rfc2812#section-3.4.10>
"Upon receiving an INFO command, the given server will respond with zero or
more RPL_INFO replies, followed by one RPL_ENDOFINFO numeric"
-- <https://modern.ircdocs.horse/#info-message>
"""
self.connectClient("nick")
# Remote /INFO is oper-only on Unreal and ircu2
self.sendLine(1, "OPER operuser operpassword")
self.assertIn(
RPL_YOUREOPER,
[m.command for m in self.getMessages(1)],
fail_msg="OPER failed",
)
if target:
self.sendLine(1, "INFO My.Little.Server")
else:
self.sendLine(1, "INFO")
messages = self.getMessages(1)
last_message = messages.pop()
self.assertMessageMatch(
last_message, command=RPL_ENDOFINFO, params=["nick", ANYSTR]
)
for message in messages:
self.assertMessageMatch(message, command=RPL_INFO, params=["nick", ANYSTR])
@pytest.mark.parametrize("target", ["invalid.server.example", "invalidserver"])
@cases.mark_specifications("RFC1459", "RFC2812", deprecated=True)
@cases.xfailIfSoftware(
["Ergo"], "does not apply to Ergo, which ignores the optional <target> argument"
)
def testInfoNosuchserver(self, target):
"""
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.3.8>
<https://datatracker.ietf.org/doc/html/rfc2812#section-3.4.10>
"Upon receiving an INFO command, the given server will respond with zero or
more RPL_INFO replies, followed by one RPL_ENDOFINFO numeric"
-- <https://modern.ircdocs.horse/#info-message>
"""
self.connectClient("nick")
# Remote /INFO is oper-only on Unreal and ircu2
self.sendLine(1, "OPER operuser operpassword")
self.assertIn(
RPL_YOUREOPER,
[m.command for m in self.getMessages(1)],
fail_msg="OPER failed",
)
self.sendLine(1, f"INFO {target}")
self.assertMessageMatch(
self.getMessage(1),
command=ERR_NOSUCHSERVER,
params=["nick", target, ANYSTR],
)

View File

@ -1,9 +1,18 @@
"""
The INVITE command (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.7>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.7>`__,
`Modern <https://modern.ircdocs.horse/#invite-message>`__)
"""
import pytest import pytest
from irctest import cases from irctest import cases, runner
from irctest.numerics import ( from irctest.numerics import (
ERR_BANNEDFROMCHAN,
ERR_CHANOPRIVSNEEDED, ERR_CHANOPRIVSNEEDED,
ERR_INVITEONLYCHAN, ERR_INVITEONLYCHAN,
ERR_NEEDMOREPARAMS,
ERR_NOSUCHNICK, ERR_NOSUCHNICK,
ERR_NOTONCHANNEL, ERR_NOTONCHANNEL,
ERR_USERONCHANNEL, ERR_USERONCHANNEL,
@ -109,7 +118,7 @@ class InviteTestCase(cases.BaseServerTestCase):
"got this instead: {msg}", "got this instead: {msg}",
) )
def _testInvite(self, opped, invite_only, modern): def _testInvite(self, opped, invite_only):
""" """
"Only the user inviting and the user being invited will receive "Only the user inviting and the user being invited will receive
notification of the invitation." notification of the invitation."
@ -162,23 +171,14 @@ class InviteTestCase(cases.BaseServerTestCase):
) )
self.sendLine(1, "INVITE bar #chan") self.sendLine(1, "INVITE bar #chan")
if modern: self.assertMessageMatch(
self.assertMessageMatch( self.getMessage(1),
self.getMessage(1), command=RPL_INVITING,
command=RPL_INVITING, params=["foo", "bar", "#chan"],
params=["foo", "bar", "#chan"], fail_msg=f"After “foo” invited “bar” to a channel, “foo” should have "
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"received “{RPL_INVITING} foo #chan bar” but got this instead: " f"{{msg}}",
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) messages = self.getMessages(2)
self.assertNotEqual( self.assertNotEqual(
@ -196,24 +196,17 @@ class InviteTestCase(cases.BaseServerTestCase):
) )
@pytest.mark.parametrize("invite_only", [True, False]) @pytest.mark.parametrize("invite_only", [True, False])
@cases.mark_specifications("Modern") @cases.mark_specifications("RFC1459", "RFC2812", "Modern")
def testInviteModern(self, invite_only): def testInvite(self, invite_only):
self._testInvite(opped=True, invite_only=invite_only, modern=True) self._testInvite(opped=True, invite_only=invite_only)
@pytest.mark.parametrize("invite_only", [True, False]) @cases.mark_specifications("RFC1459", "RFC2812", "Modern", strict=True)
@cases.mark_specifications("RFC1459", "RFC2812", deprecated=True) @cases.xfailIfSoftware(
def testInviteRfc(self, invite_only): ["Hybrid", "Plexus4"], "the only strict test that Hybrid fails"
self._testInvite(opped=True, invite_only=invite_only, modern=False) )
def testInviteUnopped(self):
@cases.mark_specifications("Modern", strict=True)
def testInviteUnoppedModern(self):
"""Tests invites from unopped users on not-invite-only chans.""" """Tests invites from unopped users on not-invite-only chans."""
self._testInvite(opped=False, invite_only=False, modern=True) self._testInvite(opped=False, invite_only=False)
@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") @cases.mark_specifications("RFC2812", "Modern")
def testInviteNoNotificationForOtherMembers(self): def testInviteNoNotificationForOtherMembers(self):
@ -247,7 +240,13 @@ class InviteTestCase(cases.BaseServerTestCase):
"were notified: {got}", "were notified: {got}",
) )
def _testInviteInviteOnly(self, modern): @cases.mark_specifications("RFC1459", "RFC2812", "Modern")
@cases.xfailIfSoftware(
["Plexus4"],
"Plexus4 allows non-op to invite if (and only if) the channel is not "
"invite-only",
)
def testInviteInviteOnly(self):
""" """
"To invite a user to a channel which is invite only (MODE "To invite a user to a channel which is invite only (MODE
+i), the client sending the invite must be recognised as being a +i), the client sending the invite must be recognised as being a
@ -287,35 +286,17 @@ class InviteTestCase(cases.BaseServerTestCase):
) )
self.sendLine(1, "INVITE bar #chan") self.sendLine(1, "INVITE bar #chan")
if modern: self.assertMessageMatch(
self.assertMessageMatch( self.getMessage(1),
self.getMessage(1), command=ERR_CHANOPRIVSNEEDED,
command=ERR_CHANOPRIVSNEEDED, params=["foo", "#chan", ANYSTR],
params=["foo", "#chan", ANYSTR], fail_msg=f"After “foo” invited “bar” to a channel to an invite-only "
fail_msg=f"After “foo” invited “bar” to a channel to an invite-only " f"channel without being opped,foo” should have received "
f"channel without being opped, “foo” should have received " f"{ERR_CHANOPRIVSNEEDED} foo #chan :*” but got this instead: {{msg}}",
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") @cases.mark_specifications("RFC2812", "Modern")
def _testInviteOnlyFromUsersInChannel(self, modern): def testInviteOnlyFromUsersInChannel(self):
""" """
"if the channel exists, only members of the channel are allowed "if the channel exists, only members of the channel are allowed
to invite other users" to invite other users"
@ -348,26 +329,15 @@ class InviteTestCase(cases.BaseServerTestCase):
self.getMessages(3) self.getMessages(3)
self.sendLine(1, "INVITE bar #chan") self.sendLine(1, "INVITE bar #chan")
if modern: self.assertMessageMatch(
self.assertMessageMatch( self.getMessage(1),
self.getMessage(1), command=ERR_NOTONCHANNEL,
command=ERR_NOTONCHANNEL, params=["foo", "#chan", ANYSTR],
params=["foo", "#chan", ANYSTR], fail_msg=f"After “foo” invited “bar” to a channel it is not on "
fail_msg=f"After “foo” invited “bar” to a channel it is not on " f"#chan, “foo” should have received "
f"#chan, “foo” should have received " f"“ERR_NOTONCHANNEL ({ERR_NOTONCHANNEL}) foo #chan :*” but "
f"“ERR_NOTONCHANNEL ({ERR_NOTONCHANNEL}) foo #chan :*” but " f"got this instead: {{msg}}",
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) messages = self.getMessages(2)
self.assertEqual( self.assertEqual(
@ -377,14 +347,6 @@ class InviteTestCase(cases.BaseServerTestCase):
"not in #chan, “bar” received something.", "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") @cases.mark_specifications("Modern")
def testInviteAlreadyInChannel(self): def testInviteAlreadyInChannel(self):
""" """
@ -410,3 +372,124 @@ class InviteTestCase(cases.BaseServerTestCase):
command=ERR_USERONCHANNEL, command=ERR_USERONCHANNEL,
params=["foo", "bar", "#chan", ANYSTR], params=["foo", "bar", "#chan", ANYSTR],
) )
@cases.mark_specifications("RFC2812", "Modern")
@cases.xfailIfSoftware(
["ircu2"],
"Uses 346/347 instead of 336/337 to reply to INVITE "
"https://github.com/UndernetIRC/ircu2/pull/20",
)
def testInviteList(self):
self.connectClient("foo")
self.connectClient("bar")
self.getMessages(1)
self.getMessages(2)
self.sendLine(1, "JOIN #chan")
self.getMessages(1)
self.sendLine(1, "INVITE bar #chan")
self.getMessages(1)
self.getMessages(2)
self.sendLine(2, "INVITE")
m = self.getMessage(2)
if m.command == ERR_NEEDMOREPARAMS:
raise runner.OptionalExtensionNotSupported("INVITE with no parameter")
if m.command != "337":
# Hybrid always sends an empty list; so skip this.
self.assertMessageMatch(
m,
command="336",
params=["bar", "#chan"],
)
m = self.getMessage(2)
self.assertMessageMatch(
m,
command="337",
params=["bar", ANYSTR],
)
@cases.mark_isupport("INVEX")
@cases.mark_specifications("Modern")
def testInvexList(self):
self.connectClient("foo")
self.getMessages(1)
if "INVEX" in self.server_support:
invex = self.server_support.get("INVEX") or "I"
else:
raise runner.IsupportTokenNotSupported("INVEX")
self.sendLine(1, "JOIN #chan")
self.getMessages(1)
self.sendLine(1, f"MODE #chan +{invex} bar!*@*")
self.getMessages(1)
self.sendLine(1, f"MODE #chan +{invex}")
m = self.getMessage(1)
if len(m.params) == 3:
# Old format
self.assertMessageMatch(
m,
command="346",
params=["foo", "#chan", "bar!*@*"],
)
else:
self.assertMessageMatch(
m,
command="346",
params=[
"foo",
"#chan",
"bar!*@*",
StrRe("foo(!.*@.*)?"),
StrRe("[0-9]+"),
],
)
self.assertMessageMatch(
self.getMessage(1),
command="347",
params=["foo", "#chan", ANYSTR],
)
@cases.mark_specifications("Ergo")
def testInviteExemptsFromBan(self):
# regression test for ergochat/ergo#1876;
# INVITE should override a +b ban
self.connectClient("alice", name="alice")
self.joinChannel("alice", "#alice")
self.sendLine("alice", "MODE #alice +b bob!*@*")
result = {msg.command for msg in self.getMessages("alice")}
self.assertIn("MODE", result)
self.connectClient("bob", name="bob")
self.sendLine("bob", "JOIN #alice")
result = {msg.command for msg in self.getMessages("bob")}
self.assertIn(ERR_BANNEDFROMCHAN, result)
self.assertNotIn("JOIN", result)
self.sendLine("alice", "INVITE bob #alice")
result = {msg.command for msg in self.getMessages("alice")}
self.assertIn(RPL_INVITING, result)
self.assertNotIn(ERR_USERONCHANNEL, result)
result = {msg.command for msg in self.getMessages("bob")}
self.assertIn("INVITE", result)
self.sendLine("bob", "JOIN #alice")
result = {msg.command for msg in self.getMessages("bob")}
self.assertNotIn(ERR_BANNEDFROMCHAN, result)
self.assertIn("JOIN", result)
self.sendLine("alice", "KICK #alice bob")
self.getMessages("alice")
result = {msg.command for msg in self.getMessages("bob")}
self.assertIn("KICK", result)
# INVITE gets "used up" after one JOIN
self.sendLine("bob", "JOIN #alice")
result = {msg.command for msg in self.getMessages("bob")}
self.assertIn(ERR_BANNEDFROMCHAN, result)
self.assertNotIn("JOIN", result)

View File

@ -1,9 +1,72 @@
"""
RPL_ISUPPORT: `format <https://modern.ircdocs.horse/#rplisupport-005>`__
and various `tokens <https://modern.ircdocs.horse/#rplisupport-parameters>`__
"""
import re import re
from irctest import cases, runner from irctest import cases, runner
class IsupportTestCase(cases.BaseServerTestCase): class IsupportTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("Modern")
@cases.mark_isupport("PREFIX")
def testPrefix(self):
"""https://modern.ircdocs.horse/#prefix-parameter"""
self.connectClient("foo")
if "PREFIX" not in self.server_support:
raise runner.IsupportTokenNotSupported("PREFIX")
if self.server_support["PREFIX"] == "":
# "The value is OPTIONAL and when it is not specified indicates that no
# prefixes are supported."
return
m = re.match(
r"\((?P<modes>[a-zA-Z]+)\)(?P<prefixes>\S+)", self.server_support["PREFIX"]
)
self.assertTrue(
m,
f"PREFIX={self.server_support['PREFIX']} does not have the expected "
f"format.",
)
modes = m.group("modes")
prefixes = m.group("prefixes")
# "There is a one-to-one mapping between prefixes and channel modes."
self.assertEqual(
len(modes), len(prefixes), "Mismatched length of prefix and channel modes."
)
# "The prefixes in this parameter are in descending order, from the prefix
# that gives the most privileges to the prefix that gives the least."
self.assertLess(modes.index("o"), modes.index("v"), "'o' is not before 'v'")
if "h" in modes:
self.assertLess(modes.index("o"), modes.index("h"), "'o' is not before 'h'")
self.assertLess(modes.index("h"), modes.index("v"), "'h' is not before 'v'")
if "q" in modes:
self.assertLess(modes.index("q"), modes.index("o"), "'q' is not before 'o'")
# Not technically in the spec, but it would be very confusing not to follow
# these conventions.
mode_to_prefix = dict(zip(modes, prefixes))
self.assertEqual(mode_to_prefix["o"], "@", "Prefix char for mode +o is not @")
self.assertEqual(mode_to_prefix["v"], "+", "Prefix char for mode +v is not +")
if "h" in modes:
self.assertEqual(
mode_to_prefix["h"], "%", "Prefix char for mode +h is not %"
)
if "q" in modes:
self.assertEqual(
mode_to_prefix["q"], "~", "Prefix char for mode +q is not ~"
)
if "a" in modes:
self.assertEqual(
mode_to_prefix["a"], "&", "Prefix char for mode +a is not &"
)
@cases.mark_specifications("Modern", "ircdocs") @cases.mark_specifications("Modern", "ircdocs")
@cases.mark_isupport("TARGMAX") @cases.mark_isupport("TARGMAX")
def testTargmax(self): def testTargmax(self):
@ -17,7 +80,7 @@ class IsupportTestCase(cases.BaseServerTestCase):
self.connectClient("foo") self.connectClient("foo")
if "TARGMAX" not in self.server_support: if "TARGMAX" not in self.server_support:
raise runner.NotImplementedByController("TARGMAX") raise runner.IsupportTokenNotSupported("TARGMAX")
parts = self.server_support["TARGMAX"].split(",") parts = self.server_support["TARGMAX"].split(",")
for part in parts: for part in parts:

View File

@ -1,5 +1,30 @@
from irctest import cases """
The JOIN command (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.1>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.1>`__,
`Modern <https://modern.ircdocs.horse/#join-message>`__)
"""
from irctest import cases, runner
from irctest.irc_utils import ambiguities from irctest.irc_utils import ambiguities
from irctest.numerics import (
ERR_BADCHANMASK,
ERR_FORBIDDENCHANNEL,
ERR_NOSUCHCHANNEL,
RPL_ENDOFNAMES,
RPL_NAMREPLY,
)
from irctest.patma import ANYSTR, StrRe
ERR_BADCHANNAME = "479" # Hybrid only, and conflicts with others
JOIN_ERROR_NUMERICS = {
ERR_BADCHANMASK,
ERR_NOSUCHCHANNEL,
ERR_FORBIDDENCHANNEL,
ERR_BADCHANNAME,
}
class JoinTestCase(cases.BaseServerTestCase): class JoinTestCase(cases.BaseServerTestCase):
@ -19,13 +44,22 @@ class JoinTestCase(cases.BaseServerTestCase):
self.connectClient("foo") self.connectClient("foo")
self.sendLine(1, "JOIN #chan") self.sendLine(1, "JOIN #chan")
received_commands = {m.command for m in self.getMessages(1)} received_commands = {m.command for m in self.getMessages(1)}
expected_commands = {"353", "366"} # RPL_NAMREPLY # RPL_ENDOFNAMES expected_commands = {RPL_NAMREPLY, RPL_ENDOFNAMES, "JOIN"}
self.assertTrue( acceptable_commands = expected_commands | {"MODE"}
expected_commands.issubset(received_commands), self.assertLessEqual( # set inclusion
expected_commands,
received_commands,
"Server sent {} commands, but at least {} were expected.".format( "Server sent {} commands, but at least {} were expected.".format(
received_commands, expected_commands received_commands, expected_commands
), ),
) )
self.assertLessEqual( # ditto
received_commands,
acceptable_commands,
"Server sent {} commands, but only {} were expected.".format(
received_commands, acceptable_commands
),
)
@cases.mark_specifications("RFC2812") @cases.mark_specifications("RFC2812")
def testJoinNamreply(self): def testJoinNamreply(self):
@ -110,3 +144,95 @@ class JoinTestCase(cases.BaseServerTestCase):
'"foo" with an optional "+" or "@" prefix, but got: ' '"foo" with an optional "+" or "@" prefix, but got: '
"{msg}", "{msg}",
) )
def testJoinPartiallyInvalid(self):
"""TODO: specify this in Modern"""
self.connectClient("foo")
if int(self.targmax.get("JOIN") or "4") < 2:
raise runner.OptionalExtensionNotSupported("multi-channel JOIN")
self.sendLine(1, "JOIN #valid,inv@lid")
messages = self.getMessages(1)
received_commands = {m.command for m in messages}
expected_commands = {RPL_NAMREPLY, RPL_ENDOFNAMES, "JOIN"}
acceptable_commands = expected_commands | JOIN_ERROR_NUMERICS | {"MODE"}
self.assertLessEqual(
expected_commands,
received_commands,
"Server sent {} commands, but at least {} were expected.".format(
received_commands, expected_commands
),
)
self.assertLessEqual(
received_commands,
acceptable_commands,
"Server sent {} commands, but only {} were expected.".format(
received_commands, acceptable_commands
),
)
nb_errors = 0
for m in messages:
if m.command in JOIN_ERROR_NUMERICS:
nb_errors += 1
self.assertMessageMatch(m, params=["foo", "inv@lid", ANYSTR])
self.assertEqual(
nb_errors,
1,
fail_msg="Expected 1 error when joining channels '#valid' and 'inv@lid', "
"got {got}",
)
@cases.mark_capabilities("batch", "labeled-response")
def testJoinPartiallyInvalidLabeledResponse(self):
"""TODO: specify this in Modern"""
self.connectClient(
"foo", capabilities=["batch", "labeled-response"], skip_if_cap_nak=True
)
if int(self.targmax.get("JOIN") or "4") < 2:
raise runner.OptionalExtensionNotSupported("multi-channel JOIN")
self.sendLine(1, "@label=label1 JOIN #valid,inv@lid")
messages = self.getMessages(1)
first_msg = messages.pop(0)
last_msg = messages.pop(-1)
self.assertMessageMatch(
first_msg, command="BATCH", params=[StrRe(r"\+.*"), "labeled-response"]
)
batch_id = first_msg.params[0][1:]
self.assertMessageMatch(last_msg, command="BATCH", params=["-" + batch_id])
received_commands = {m.command for m in messages}
expected_commands = {RPL_NAMREPLY, RPL_ENDOFNAMES, "JOIN"}
acceptable_commands = expected_commands | JOIN_ERROR_NUMERICS | {"MODE"}
self.assertLessEqual(
expected_commands,
received_commands,
"Server sent {} commands, but at least {} were expected.".format(
received_commands, expected_commands
),
)
self.assertLessEqual(
received_commands,
acceptable_commands,
"Server sent {} commands, but only {} were expected.".format(
received_commands, acceptable_commands
),
)
nb_errors = 0
for m in messages:
self.assertIn("batch", m.tags)
self.assertEqual(m.tags["batch"], batch_id)
if m.command in JOIN_ERROR_NUMERICS:
nb_errors += 1
self.assertMessageMatch(m, params=["foo", "inv@lid", ANYSTR])
self.assertEqual(
nb_errors,
1,
fail_msg="Expected 1 error when joining channels '#valid' and 'inv@lid', "
"got {got}",
)

View File

@ -1,3 +1,10 @@
"""
The KICK command (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.1>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.>`__,
`Modern <https://modern.ircdocs.horse/#kick-message>`__)
"""
import pytest import pytest
from irctest import cases, client_mock, runner from irctest import cases, client_mock, runner
@ -89,6 +96,10 @@ class KickTestCase(cases.BaseServerTestCase):
self.assertMessageMatch(m3, command="KICK", params=["#chan", "bar", ANYSTR]) self.assertMessageMatch(m3, command="KICK", params=["#chan", "bar", ANYSTR])
@cases.mark_specifications("RFC2812") @cases.mark_specifications("RFC2812")
@cases.xfailIfSoftware(
["Charybdis", "ircu2", "irc2", "Solanum"],
"uses the nick of the kickee rather than the kicker.",
)
def testKickDefaultComment(self): def testKickDefaultComment(self):
""" """
"If a "comment" is "If a "comment" is
@ -219,13 +230,8 @@ class KickTestCase(cases.BaseServerTestCase):
self.connectClient("qux") self.connectClient("qux")
self.joinChannel(4, "#chan") self.joinChannel(4, "#chan")
targmax = dict( if self.targmax.get("KICK", "1") == "1":
item.split(":", 1) raise runner.OptionalExtensionNotSupported("Multi-target KICK")
for item in self.server_support.get("TARGMAX", "").split(",")
if item
)
if targmax.get("KICK", "1") == "1":
raise runner.NotImplementedByController("Multi-target KICK")
# TODO: check foo is an operator # TODO: check foo is an operator

View File

@ -1,8 +1,8 @@
""" """
`IRCv3 labeled-response <https://ircv3.net/specs/extensions/labeled-response>`_
This specification is a little hard to test because all labels are optional; This specification is a little hard to test because all labels are optional;
so there may be many false positives. so there may be many false positives.
<https://ircv3.net/specs/extensions/labeled-response.html>
""" """
import re import re
@ -10,10 +10,11 @@ import re
import pytest import pytest
from irctest import cases from irctest import cases
from irctest.patma import ANYDICT, AnyOptStr, NotStrRe, RemainingKeys, StrRe from irctest.numerics import ERR_UNKNOWNCOMMAND
from irctest.patma import ANYDICT, ANYOPTSTR, NotStrRe, RemainingKeys, StrRe
class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class LabeledResponsesTestCase(cases.BaseServerTestCase):
@cases.mark_capabilities("echo-message", "batch", "labeled-response") @cases.mark_capabilities("echo-message", "batch", "labeled-response")
def testLabeledPrivmsgResponsesToMultipleClients(self): def testLabeledPrivmsgResponsesToMultipleClients(self):
self.connectClient( self.connectClient(
@ -298,7 +299,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
tags={ tags={
"+draft/reply": msgid, "+draft/reply": msgid,
"+draft/react": "l😃l", "+draft/react": "l😃l",
RemainingKeys(NotStrRe("label")): AnyOptStr(), RemainingKeys(NotStrRe("label")): ANYOPTSTR,
}, },
) )
self.assertNotIn( self.assertNotIn(
@ -366,7 +367,7 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
tags={ tags={
"+draft/reply": msgid, "+draft/reply": msgid,
"+draft/react": "l😃l", "+draft/react": "l😃l",
RemainingKeys(NotStrRe("label")): AnyOptStr(), RemainingKeys(NotStrRe("label")): ANYOPTSTR,
}, },
fail_msg="No TAGMSG received by the target after sending one out", fail_msg="No TAGMSG received by the target after sending one out",
) )
@ -502,3 +503,19 @@ class LabeledResponsesTestCase(cases.BaseServerTestCase, cases.OptionalityHelper
ack = ms[0] ack = ms[0]
self.assertMessageMatch(ack, command="ACK", tags={"label": "98765"}) self.assertMessageMatch(ack, command="ACK", tags={"label": "98765"})
@cases.mark_capabilities("labeled-response")
def testUnknownCommand(self):
self.connectClient(
"bar", capabilities=["batch", "labeled-response"], skip_if_cap_nak=True
)
# this command doesn't exist, but the error response should still
# be labeled:
self.sendLine(1, "@label=deadbeef NONEXISTENT_COMMAND")
ms = self.getMessages(1)
self.assertEqual(len(ms), 1)
unknowncommand = ms[0]
self.assertMessageMatch(
unknowncommand, command=ERR_UNKNOWNCOMMAND, tags={"label": "deadbeef"}
)

View File

@ -0,0 +1,136 @@
from irctest import cases, runner
from irctest.numerics import ERR_UNKNOWNCOMMAND, RPL_ENDOFLINKS, RPL_LINKS
from irctest.patma import ANYSTR, StrRe
class LinksTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
def testLinksSingleServer(self):
"""
Only testing the parameter-less case.
https://datatracker.ietf.org/doc/html/rfc1459#section-4.3.3
https://datatracker.ietf.org/doc/html/rfc2812#section-3.4.5
https://github.com/ircdocs/modern-irc/pull/175
"
364 RPL_LINKS
"<mask> <server> :<hopcount> <server info>"
365 RPL_ENDOFLINKS
"<mask> :End of /LINKS list"
- In replying to the LINKS message, a server must send
replies back using the RPL_LINKS numeric and mark the
end of the list using an RPL_ENDOFLINKS reply.
"
-- https://datatracker.ietf.org/doc/html/rfc1459#page-51
-- https://datatracker.ietf.org/doc/html/rfc2812#page-48
RPL_LINKS: "<client> * <server> :<hopcount> <server info>"
RPL_ENDOFLINKS: "<client> * :End of /LINKS list"
-- https://github.com/ircdocs/modern-irc/pull/175/files
"""
self.connectClient("nick")
self.sendLine(1, "LINKS")
messages = self.getMessages(1)
if messages[0].command == ERR_UNKNOWNCOMMAND:
raise runner.OptionalCommandNotSupported("LINKS")
# Ignore '/LINKS has been disabled' from ircu2
messages = [m for m in messages if m.command != "NOTICE"]
self.assertMessageMatch(
messages.pop(-1),
command=RPL_ENDOFLINKS,
params=["nick", "*", ANYSTR],
)
if not messages:
# This server probably redacts links
return
self.assertMessageMatch(
messages[0],
command=RPL_LINKS,
params=[
"nick",
"My.Little.Server",
"My.Little.Server",
StrRe("0 (0042 )?test server"),
],
)
@cases.mark_services
class ServicesLinksTestCase(cases.BaseServerTestCase):
# On every IRCd but Ergo, services are linked.
# Ergo does not implement LINKS at all, so this test is skipped.
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
def testLinksWithServices(self):
"""
Only testing the parameter-less case.
https://datatracker.ietf.org/doc/html/rfc1459#section-4.3.3
https://datatracker.ietf.org/doc/html/rfc2812#section-3.4.5
"
364 RPL_LINKS
"<mask> <server> :<hopcount> <server info>"
365 RPL_ENDOFLINKS
"<mask> :End of /LINKS list"
- In replying to the LINKS message, a server must send
replies back using the RPL_LINKS numeric and mark the
end of the list using an RPL_ENDOFLINKS reply.
"
-- https://datatracker.ietf.org/doc/html/rfc1459#page-51
-- https://datatracker.ietf.org/doc/html/rfc2812#page-48
RPL_LINKS: "<client> * <server> :<hopcount> <server info>"
RPL_ENDOFLINKS: "<client> * :End of /LINKS list"
-- https://github.com/ircdocs/modern-irc/pull/175/files
"""
self.connectClient("nick")
self.sendLine(1, "LINKS")
messages = self.getMessages(1)
if messages[0].command == ERR_UNKNOWNCOMMAND:
raise runner.OptionalCommandNotSupported("LINKS")
# Ignore '/LINKS has been disabled' from ircu2
messages = [m for m in messages if m.command != "NOTICE"]
self.assertMessageMatch(
messages.pop(-1),
command=RPL_ENDOFLINKS,
params=["nick", "*", ANYSTR],
)
if not messages:
# This server redacts links
return
messages.sort(key=lambda m: m.params[-1])
self.assertMessageMatch(
messages.pop(0),
command=RPL_LINKS,
params=[
"nick",
"My.Little.Server",
"My.Little.Server",
StrRe("0 (0042 )?test server"),
],
)
self.assertMessageMatch(
messages.pop(0),
command=RPL_LINKS,
params=[
"nick",
"services.example.org",
"My.Little.Server",
StrRe("1 .+"), # SID instead of description for Anope...
],
)
self.assertEqual(messages, [])

View File

@ -1,40 +1,71 @@
from irctest import cases """
The LIST command (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.6>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.6>`__,
`Modern <https://modern.ircdocs.horse/#list-message>`__)
"""
import time
from irctest import cases, runner
from irctest.numerics import RPL_LIST, RPL_LISTEND, RPL_LISTSTART
class ListTestCase(cases.BaseServerTestCase): class _BasedListTestCase(cases.BaseServerTestCase):
def _parseChanList(self, client):
channels = set()
while True:
m = self.getMessage(client)
if m.command == RPL_LISTEND:
break
if m.command == RPL_LIST:
if m.params[1].startswith("&"):
# skip local pseudo-channels listed by ngircd and ircu
continue
channels.add(m.params[1])
return channels
class ListTestCase(_BasedListTestCase):
@cases.mark_specifications("RFC1459", "RFC2812") @cases.mark_specifications("RFC1459", "RFC2812")
@cases.xfailIfSoftware(["irc2"], "irc2 deprecated LIST")
def testListEmpty(self): def testListEmpty(self):
"""<https://tools.ietf.org/html/rfc1459#section-4.2.6> """<https://tools.ietf.org/html/rfc1459#section-4.2.6>
<https://tools.ietf.org/html/rfc2812#section-3.2.6> <https://tools.ietf.org/html/rfc2812#section-3.2.6>
<https://modern.ircdocs.horse/#list-message>
""" """
self.connectClient("foo") self.connectClient("foo")
self.connectClient("bar") self.connectClient("bar")
self.getMessages(1) self.getMessages(1)
self.sendLine(2, "LIST") self.sendLine(2, "LIST")
m = self.getMessage(2) m = self.getMessage(2)
if m.command == "321": if m.command == RPL_LISTSTART:
# skip RPL_LISTSTART # skip
m = self.getMessage(2) m = self.getMessage(2)
while m.command == "322" and m.params[1] == "&SERVER": # skip local pseudo-channels listed by ngircd and ircu
# ngircd adds this pseudo-channel while m.command == RPL_LIST and m.params[1].startswith("&"):
m = self.getMessage(2) m = self.getMessage(2)
self.assertNotEqual( self.assertNotEqual(
m.command, m.command,
"322", # RPL_LIST RPL_LIST,
"LIST response gives (at least) one channel, whereas there " "is none.", "LIST response gives (at least) one channel, whereas there " "is none.",
) )
self.assertMessageMatch( self.assertMessageMatch(
m, m,
command="323", # RPL_LISTEND command=RPL_LISTEND,
fail_msg="Second reply to LIST is not 322 (RPL_LIST) " fail_msg="Second reply to LIST is not 322 (RPL_LIST) "
"or 323 (RPL_LISTEND), or but: {msg}", "or 323 (RPL_LISTEND), or but: {msg}",
) )
@cases.mark_specifications("RFC1459", "RFC2812") @cases.mark_specifications("RFC1459", "RFC2812")
@cases.xfailIfSoftware(["irc2"], "irc2 deprecated LIST")
def testListOne(self): def testListOne(self):
"""When a channel exists, LIST should get it in a reply. """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/rfc1459#section-4.2.6>
<https://tools.ietf.org/html/rfc2812#section-3.2.6> <https://tools.ietf.org/html/rfc2812#section-3.2.6>
<https://modern.ircdocs.horse/#list-message>
""" """
self.connectClient("foo") self.connectClient("foo")
self.connectClient("bar") self.connectClient("bar")
@ -42,34 +73,331 @@ class ListTestCase(cases.BaseServerTestCase):
self.getMessages(1) self.getMessages(1)
self.sendLine(2, "LIST") self.sendLine(2, "LIST")
m = self.getMessage(2) m = self.getMessage(2)
if m.command == "321": if m.command == RPL_LISTSTART:
# skip RPL_LISTSTART # skip
m = self.getMessage(2) m = self.getMessage(2)
self.assertNotEqual( self.assertNotEqual(
m.command, m.command,
"323", # RPL_LISTEND RPL_LISTEND,
fail_msg="LIST response ended (ie. 323, aka RPL_LISTEND) " fail_msg="LIST response ended (ie. 323, aka RPL_LISTEND) "
"without listing any channel, whereas there is one.", "without listing any channel, whereas there is one.",
) )
self.assertMessageMatch( self.assertMessageMatch(
m, m,
command="322", # RPL_LIST command=RPL_LIST,
fail_msg="Second reply to LIST is not 322 (RPL_LIST), " fail_msg="Second reply to LIST is not 322 (RPL_LIST), "
"nor 323 (RPL_LISTEND) but: {msg}", "nor 323 (RPL_LISTEND) but: {msg}",
) )
m = self.getMessage(2) m = self.getMessage(2)
while m.command == "322" and m.params[1] == "&SERVER": # skip local pseudo-channels listed by ngircd and ircu
# ngircd adds this pseudo-channel while m.command == RPL_LIST and m.params[1].startswith("&"):
m = self.getMessage(2) m = self.getMessage(2)
self.assertNotEqual( self.assertNotEqual(
m.command, m.command,
"322", # RPL_LIST RPL_LIST,
fail_msg="LIST response gives (at least) two channels, " fail_msg="LIST response gives (at least) two channels, "
"whereas there is only one.", "whereas there is only one.",
) )
self.assertMessageMatch( self.assertMessageMatch(
m, m,
command="323", # RPL_LISTEND command=RPL_LISTEND,
fail_msg="Third reply to LIST is not 322 (RPL_LIST) " fail_msg="Third reply to LIST is not 322 (RPL_LIST) "
"or 323 (RPL_LISTEND), or but: {msg}", "or 323 (RPL_LISTEND), or but: {msg}",
) )
@cases.mark_specifications("RFC1459", "RFC2812")
@cases.xfailIfSoftware(
["Charybdis", "Solanum"],
"Charybdis and Solanum insert ERR_NOSUCHNICK reply in LIST",
)
def testListNonexistent(self):
"""LIST on a nonexistent channel does not send an error
response.
<https://tools.ietf.org/html/rfc1459#section-4.2.6>
<https://tools.ietf.org/html/rfc2812#section-3.2.6>
"""
self.connectClient("bar")
self.sendLine(1, "LIST #nonexistent")
responses = {msg.command for msg in self.getMessages(1)}
# successful response MUST include RPL_LISTEND:
self.assertIn(RPL_LISTEND, responses)
# and MUST NOT include RPL_LIST (since there is no matching channel)
# or any error numerics:
self.assertLessEqual(responses, {RPL_LISTSTART, RPL_LISTEND})
@cases.mark_isupport("ELIST")
@cases.mark_specifications("Modern")
def testListMask(self):
"""
"M: Searching based on mask."
-- <https://modern.ircdocs.horse/#elist-parameter>
-- https://datatracker.ietf.org/doc/html/draft-hardy-irc-isupport-00#section-4.8
"""
self.connectClient("foo")
if "M" not in self.server_support.get("ELIST", ""):
raise runner.OptionalExtensionNotSupported("ELIST=M")
self.connectClient("bar")
self.sendLine(1, "JOIN #chan1")
self.getMessages(1)
self.sendLine(1, "JOIN #chan2")
self.getMessages(1)
self.sendLine(2, "LIST *an1")
self.assertEqual(self._parseChanList(2), {"#chan1"})
self.sendLine(2, "LIST *an2")
self.assertEqual(self._parseChanList(2), {"#chan2"})
self.sendLine(2, "LIST #c*n2")
self.assertEqual(self._parseChanList(2), {"#chan2"})
self.sendLine(2, "LIST *an3")
self.assertEqual(self._parseChanList(2), set())
self.sendLine(2, "LIST #ch*")
self.assertEqual(self._parseChanList(2), {"#chan1", "#chan2"})
@cases.mark_isupport("ELIST")
@cases.mark_specifications("Modern")
def testListNotMask(self):
"""
" N: Searching based on a non-matching mask. i.e., the opposite of M."
-- <https://modern.ircdocs.horse/#elist-parameter>
-- https://datatracker.ietf.org/doc/html/draft-hardy-irc-isupport-00#section-4.8
"""
self.connectClient("foo")
if "N" not in self.server_support.get("ELIST", ""):
raise runner.OptionalExtensionNotSupported("ELIST=N")
self.sendLine(1, "JOIN #chan1")
self.getMessages(1)
self.sendLine(1, "JOIN #chan2")
self.getMessages(1)
self.connectClient("bar")
self.sendLine(2, "LIST !*an1")
self.assertEqual(self._parseChanList(2), {"#chan2"})
self.sendLine(2, "LIST !*an2")
self.assertEqual(self._parseChanList(2), {"#chan1"})
self.sendLine(2, "LIST !#c*n2")
self.assertEqual(self._parseChanList(2), {"#chan1"})
self.sendLine(2, "LIST !*an3")
self.assertEqual(self._parseChanList(2), {"#chan1", "#chan2"})
self.sendLine(2, "LIST !#ch*")
self.assertEqual(self._parseChanList(2), set())
@cases.mark_isupport("ELIST")
@cases.mark_specifications("Modern")
def testListUsers(self):
"""
"U: Searching based on user count within the channel, via the "<val" and
">val" modifiers to search for a channel that has less or more than val users,
respectively."
-- <https://modern.ircdocs.horse/#elist-parameter>
-- https://datatracker.ietf.org/doc/html/draft-hardy-irc-isupport-00#section-4.8
"""
self.connectClient("foo")
if "U" not in self.server_support.get("ELIST", ""):
raise runner.OptionalExtensionNotSupported("ELIST=U")
self.sendLine(1, "JOIN #chan1")
self.getMessages(1)
self.sendLine(1, "JOIN #chan2")
self.getMessages(1)
self.connectClient("bar")
self.sendLine(2, "JOIN #chan2")
self.getMessages(2)
self.connectClient("baz")
self.sendLine(3, "LIST >0")
self.assertEqual(self._parseChanList(3), {"#chan1", "#chan2"})
self.sendLine(3, "LIST <1")
self.assertEqual(self._parseChanList(3), set())
self.sendLine(3, "LIST <100")
self.assertEqual(self._parseChanList(3), {"#chan1", "#chan2"})
self.sendLine(3, "LIST >1")
self.assertEqual(self._parseChanList(3), {"#chan2"})
self.sendLine(3, "LIST <2")
self.assertEqual(self._parseChanList(3), {"#chan1"})
self.sendLine(3, "LIST <100")
self.assertEqual(self._parseChanList(3), {"#chan1", "#chan2"})
class FaketimeListTestCase(_BasedListTestCase):
faketime = "+1y x30" # for every wall clock second, 1 minute passed for the server
def _sleep_minutes(self, n):
for _ in range(n):
if self.controller.faketime_enabled:
# From the server's point of view, 1 min will pass
time.sleep(2)
else:
time.sleep(60)
# reply to pings
self.getMessages(1)
self.getMessages(2)
@cases.mark_isupport("ELIST")
@cases.mark_specifications("Modern")
@cases.xfailIfSoftware(
["Plexus4", "Hybrid"],
"Hybrid and Plexus4 filter on ELIST=C with the opposite meaning",
)
@cases.xfailIf(
lambda self: bool(
self.controller.software_name == "UnrealIRCd"
and self.controller.software_version == 5
),
"UnrealIRCd <6.0.3 filters on ELIST=C with the opposite meaning",
)
def testListCreationTime(self):
"""
" C: Searching based on channel creation time, via the "C<val" and "C>val"
modifiers to search for a channel creation time that is higher or lower
than val."
-- <https://modern.ircdocs.horse/#elist-parameter>
-- https://datatracker.ietf.org/doc/html/draft-hardy-irc-isupport-00#section-4.8
Unfortunately, this is ambiguous, because "val" is a time delta (in minutes),
not a timestamp.
On InspIRCd and Charybdis/Solanum, "C<val" is interpreted as "the channel was
created less than <val> minutes ago
On UnrealIRCd, Plexus, and Hybrid, it is interpreted as "the channel's creation
time is a timestamp lower than <val> minutes ago" (ie. the exact opposite)
"C: Searching based on channel creation time, via the "C<val" and "C>val"
modifiers to search for a channel that was created either less than `val`
minutes ago, or more than `val` minutes ago, respectively"
-- https://github.com/ircdocs/modern-irc/pull/171
"""
self.connectClient("foo")
if "C" not in self.server_support.get("ELIST", ""):
raise runner.OptionalExtensionNotSupported("ELIST=C")
self.connectClient("bar")
self.sendLine(1, "JOIN #chan1")
self.getMessages(1)
# Helps debugging
self.sendLine(1, "TIME")
self.getMessages(1)
self._sleep_minutes(2)
# Helps debugging
self.sendLine(1, "TIME")
self.getMessages(1)
self.sendLine(1, "JOIN #chan2")
self.getMessages(1)
self._sleep_minutes(1)
self.sendLine(2, "LIST C>2")
self.assertEqual(self._parseChanList(2), {"#chan1"})
self.sendLine(2, "LIST C<2")
self.assertEqual(self._parseChanList(2), {"#chan2"})
self.sendLine(2, "LIST C<0")
if self.controller.software_name == "InspIRCd":
self.assertEqual(self._parseChanList(2), {"#chan1", "#chan2"})
else:
self.assertEqual(self._parseChanList(2), set())
self.sendLine(2, "LIST C>0")
self.assertEqual(self._parseChanList(2), {"#chan1", "#chan2"})
self.sendLine(2, "LIST C<10")
self.assertEqual(self._parseChanList(2), {"#chan1", "#chan2"})
@cases.mark_isupport("ELIST")
@cases.mark_specifications("Modern")
@cases.xfailIf(
lambda self: bool(
self.controller.software_name == "UnrealIRCd"
and self.controller.software_version == 5
),
"UnrealIRCd <6.0.3 advertises ELIST=T but does not implement it",
)
def testListTopicTime(self):
"""
"T: Searching based on topic time, via the "T<val" and "T>val"
modifiers to search for a topic time that is lower or higher than
val respectively."
-- <https://modern.ircdocs.horse/#elist-parameter>
-- https://datatracker.ietf.org/doc/html/draft-hardy-irc-isupport-00#section-4.8
See testListCreationTime's docstring for comments on this.
"T: Searching based on topic set time, via the "T<val" and "T>val" modifiers
to search for a topic time that was set less than `val` minutes ago, or more
than `val` minutes ago, respectively."
-- https://github.com/ircdocs/modern-irc/pull/171
"""
self.connectClient("foo")
if "T" not in self.server_support.get("ELIST", ""):
raise runner.OptionalExtensionNotSupported("ELIST=T")
self.connectClient("bar")
self.sendLine(1, "JOIN #chan1")
self.sendLine(1, "JOIN #chan2")
self.getMessages(1)
self.sendLine(1, "TOPIC #chan1 :First channel")
self.getMessages(1)
# Helps debugging
self.sendLine(1, "TIME")
self.getMessages(1)
self._sleep_minutes(2)
# Helps debugging
self.sendLine(1, "TIME")
self.getMessages(1)
self.sendLine(1, "TOPIC #chan2 :Second channel")
self.getMessages(1)
self._sleep_minutes(1)
self.sendLine(1, "LIST T>2")
self.assertEqual(self._parseChanList(1), {"#chan1"})
self.sendLine(1, "LIST T<2")
self.assertEqual(self._parseChanList(1), {"#chan2"})
self.sendLine(1, "LIST T<0")
if self.controller.software_name == "InspIRCd":
# Insp internally represents "LIST T>0" like "LIST"
self.assertEqual(self._parseChanList(1), {"#chan1", "#chan2"})
else:
self.assertEqual(self._parseChanList(1), set())
self.sendLine(1, "LIST T>0")
self.assertEqual(self._parseChanList(1), {"#chan1", "#chan2"})
self.sendLine(1, "LIST T<10")
self.assertEqual(self._parseChanList(1), {"#chan1", "#chan2"})

View File

@ -1,3 +1,11 @@
"""
The LUSERS command (`RFC 2812
<https://datatracker.ietf.org/doc/html/rfc2812#section-3.4.2>`__,
`Modern <https://modern.ircdocs.horse/#lusers-message>`__),
which provides statistics on user counts.
"""
from dataclasses import dataclass from dataclasses import dataclass
import re import re
from typing import Optional from typing import Optional
@ -14,6 +22,7 @@ from irctest.numerics import (
RPL_LUSERUNKNOWN, RPL_LUSERUNKNOWN,
RPL_YOUREOPER, RPL_YOUREOPER,
) )
from irctest.patma import ANYSTR, StrRe
# 3 numbers, delimited by spaces, possibly negative (eek) # 3 numbers, delimited by spaces, possibly negative (eek)
LUSERCLIENT_REGEX = re.compile(r"^.*( [-0-9]* ).*( [-0-9]* ).*( [-0-9]* ).*$") LUSERCLIENT_REGEX = re.compile(r"^.*( [-0-9]* ).*( [-0-9]* ).*( [-0-9]* ).*$")
@ -50,10 +59,11 @@ class LusersTestCase(cases.BaseServerTestCase):
self.assertIn(lusers.LocalTotal, (total, None)) self.assertIn(lusers.LocalTotal, (total, None))
self.assertIn(lusers.LocalMax, (max_, None)) self.assertIn(lusers.LocalMax, (max_, None))
def getLusers(self, client): def getLusers(self, client, allow_missing_265_266):
self.sendLine(client, "LUSERS") self.sendLine(client, "LUSERS")
messages = self.getMessages(client) messages = self.getMessages(client)
by_numeric = dict((msg.command, msg) for msg in messages) by_numeric = dict((msg.command, msg) for msg in messages)
self.assertEqual(len(by_numeric), len(messages), "Duplicated numerics")
result = LusersResult() result = LusersResult()
@ -73,12 +83,31 @@ class LusersTestCase(cases.BaseServerTestCase):
raise ValueError("corrupt reply for 251 RPL_LUSERCLIENT", luserclient_param) raise ValueError("corrupt reply for 251 RPL_LUSERCLIENT", luserclient_param)
if RPL_LUSEROP in by_numeric: if RPL_LUSEROP in by_numeric:
self.assertMessageMatch(
by_numeric[RPL_LUSEROP], params=[client, StrRe("[0-9]+"), ANYSTR]
)
result.Opers = int(by_numeric[RPL_LUSEROP].params[1]) result.Opers = int(by_numeric[RPL_LUSEROP].params[1])
if RPL_LUSERUNKNOWN in by_numeric: if RPL_LUSERUNKNOWN in by_numeric:
self.assertMessageMatch(
by_numeric[RPL_LUSERUNKNOWN], params=[client, StrRe("[0-9]+"), ANYSTR]
)
result.Unregistered = int(by_numeric[RPL_LUSERUNKNOWN].params[1]) result.Unregistered = int(by_numeric[RPL_LUSERUNKNOWN].params[1])
if RPL_LUSERCHANNELS in by_numeric: if RPL_LUSERCHANNELS in by_numeric:
self.assertMessageMatch(
by_numeric[RPL_LUSERCHANNELS], params=[client, StrRe("[0-9]+"), ANYSTR]
)
result.Channels = int(by_numeric[RPL_LUSERCHANNELS].params[1]) result.Channels = int(by_numeric[RPL_LUSERCHANNELS].params[1])
self.assertMessageMatch(by_numeric[RPL_LUSERCLIENT], params=[client, ANYSTR])
self.assertMessageMatch(by_numeric[RPL_LUSERME], params=[client, ANYSTR])
if (
allow_missing_265_266
and RPL_LOCALUSERS not in by_numeric
and RPL_GLOBALUSERS not in by_numeric
):
return
# FIXME: RPL_LOCALUSERS and RPL_GLOBALUSERS are only in Modern, not in RFC2812 # FIXME: RPL_LOCALUSERS and RPL_GLOBALUSERS are only in Modern, not in RFC2812
localusers = by_numeric[RPL_LOCALUSERS] localusers = by_numeric[RPL_LOCALUSERS]
globalusers = by_numeric[RPL_GLOBALUSERS] globalusers = by_numeric[RPL_GLOBALUSERS]
@ -114,23 +143,55 @@ class BasicLusersTestCase(LusersTestCase):
@cases.mark_specifications("RFC2812") @cases.mark_specifications("RFC2812")
def testLusers(self): def testLusers(self):
self.connectClient("bar", name="bar") self.connectClient("bar", name="bar")
lusers = self.getLusers("bar") self.getLusers("bar", True)
self.connectClient("qux", name="qux")
self.getLusers("qux", True)
self.sendLine("qux", "QUIT")
self.assertDisconnected("qux")
self.getLusers("bar", True)
@cases.mark_specifications("Modern")
@cases.xfailIfSoftware(
["ircu2", "Nefarious", "snircd"],
"test depends on Modern behavior, not just RFC2812",
)
def testLusersFull(self):
self.connectClient("bar", name="bar")
lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=1, max_=1) self.assertLusersResult(lusers, unregistered=0, total=1, max_=1)
self.connectClient("qux", name="qux") self.connectClient("qux", name="qux")
lusers = self.getLusers("qux") lusers = self.getLusers("qux", False)
self.assertLusersResult(lusers, unregistered=0, total=2, max_=2) self.assertLusersResult(lusers, unregistered=0, total=2, max_=2)
self.sendLine("qux", "QUIT") self.sendLine("qux", "QUIT")
self.assertDisconnected("qux") self.assertDisconnected("qux")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=1, max_=2) self.assertLusersResult(lusers, unregistered=0, total=1, max_=2)
class LusersUnregisteredTestCase(LusersTestCase): class LusersUnregisteredTestCase(LusersTestCase):
@cases.mark_specifications("RFC2812") @cases.mark_specifications("RFC2812")
def testLusers(self): @cases.xfailIfSoftware(
self.doLusersTest() ["Nefarious"],
"Nefarious doesn't seem to distinguish unregistered users from normal ones",
)
def testLusersRfc2812(self):
self.doLusersTest(True)
@cases.mark_specifications("Modern")
@cases.xfailIfSoftware(
["Nefarious"],
"Nefarious doesn't seem to distinguish unregistered users from normal ones",
)
@cases.xfailIfSoftware(
["ircu2", "Nefarious", "snircd"],
"test depends on Modern behavior, not just RFC2812",
)
def testLusersFull(self):
self.doLusersTest(False)
def _synchronize(self, client_name): def _synchronize(self, client_name):
"""Synchronizes using a PING, but accept ERR_NOTREGISTERED as a response.""" """Synchronizes using a PING, but accept ERR_NOTREGISTERED as a response."""
@ -145,34 +206,39 @@ class LusersUnregisteredTestCase(LusersTestCase):
"got neither PONG or ERR_NOTREGISTERED" "got neither PONG or ERR_NOTREGISTERED"
) )
def doLusersTest(self): def doLusersTest(self, allow_missing_265_266):
self.connectClient("bar", name="bar") self.connectClient("bar", name="bar")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", allow_missing_265_266)
self.assertLusersResult(lusers, unregistered=0, total=1, max_=1) if lusers:
self.assertLusersResult(lusers, unregistered=0, total=1, max_=1)
self.addClient("qux") self.addClient("qux")
self.sendLine("qux", "NICK qux") self.sendLine("qux", "NICK qux")
self._synchronize("qux") self._synchronize("qux")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", allow_missing_265_266)
self.assertLusersResult(lusers, unregistered=1, total=1, max_=1) if lusers:
self.assertLusersResult(lusers, unregistered=1, total=1, max_=1)
self.addClient("bat") self.addClient("bat")
self.sendLine("bat", "NICK bat") self.sendLine("bat", "NICK bat")
self._synchronize("bat") self._synchronize("bat")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", allow_missing_265_266)
self.assertLusersResult(lusers, unregistered=2, total=1, max_=1) if lusers:
self.assertLusersResult(lusers, unregistered=2, total=1, max_=1)
# complete registration on one client # complete registration on one client
self.sendLine("qux", "USER u s e r") self.sendLine("qux", "USER u s e r")
self.getRegistrationMessage("qux") self.getRegistrationMessage("qux")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", allow_missing_265_266)
self.assertLusersResult(lusers, unregistered=1, total=2, max_=2) if lusers:
self.assertLusersResult(lusers, unregistered=1, total=2, max_=2)
# QUIT the other without registering # QUIT the other without registering
self.sendLine("bat", "QUIT") self.sendLine("bat", "QUIT")
self.assertDisconnected("bat") self.assertDisconnected("bat")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", allow_missing_265_266)
self.assertLusersResult(lusers, unregistered=0, total=2, max_=2) if lusers:
self.assertLusersResult(lusers, unregistered=0, total=2, max_=2)
class LusersUnregisteredDefaultInvisibleTestCase(LusersUnregisteredTestCase): class LusersUnregisteredDefaultInvisibleTestCase(LusersUnregisteredTestCase):
@ -187,9 +253,13 @@ class LusersUnregisteredDefaultInvisibleTestCase(LusersUnregisteredTestCase):
) )
@cases.mark_specifications("Ergo") @cases.mark_specifications("Ergo")
@cases.xfailIfSoftware(
["Nefarious"],
"Nefarious doesn't seem to distinguish unregistered users from normal ones",
)
def testLusers(self): def testLusers(self):
self.doLusersTest() self.doLusersTest(False)
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=2, max_=2) self.assertLusersResult(lusers, unregistered=0, total=2, max_=2)
self.assertEqual(lusers.GlobalInvisible, 2) self.assertEqual(lusers.GlobalInvisible, 2)
self.assertEqual(lusers.GlobalVisible, 0) self.assertEqual(lusers.GlobalVisible, 0)
@ -199,7 +269,7 @@ class LuserOpersTestCase(LusersTestCase):
@cases.mark_specifications("Ergo") @cases.mark_specifications("Ergo")
def testLuserOpers(self): def testLuserOpers(self):
self.connectClient("bar", name="bar") self.connectClient("bar", name="bar")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=1, max_=1) self.assertLusersResult(lusers, unregistered=0, total=1, max_=1)
self.assertIn(lusers.Opers, (0, None)) self.assertIn(lusers.Opers, (0, None))
@ -207,7 +277,7 @@ class LuserOpersTestCase(LusersTestCase):
self.sendLine("bar", "OPER operuser operpassword") self.sendLine("bar", "OPER operuser operpassword")
msgs = self.getMessages("bar") msgs = self.getMessages("bar")
self.assertIn(RPL_YOUREOPER, {msg.command for msg in msgs}) self.assertIn(RPL_YOUREOPER, {msg.command for msg in msgs})
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=1, max_=1) self.assertLusersResult(lusers, unregistered=0, total=1, max_=1)
self.assertEqual(lusers.Opers, 1) self.assertEqual(lusers.Opers, 1)
@ -215,7 +285,7 @@ class LuserOpersTestCase(LusersTestCase):
self.connectClient("qux", name="qux") self.connectClient("qux", name="qux")
self.sendLine("qux", "OPER operuser operpassword") self.sendLine("qux", "OPER operuser operpassword")
self.getMessages("qux") self.getMessages("qux")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=2, max_=2) self.assertLusersResult(lusers, unregistered=0, total=2, max_=2)
self.assertEqual(lusers.Opers, 2) self.assertEqual(lusers.Opers, 2)
@ -223,14 +293,14 @@ class LuserOpersTestCase(LusersTestCase):
self.sendLine("bar", "MODE bar -o") self.sendLine("bar", "MODE bar -o")
msgs = self.getMessages("bar") msgs = self.getMessages("bar")
self.assertIn("MODE", {msg.command for msg in msgs}) self.assertIn("MODE", {msg.command for msg in msgs})
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=2, max_=2) self.assertLusersResult(lusers, unregistered=0, total=2, max_=2)
self.assertEqual(lusers.Opers, 1) self.assertEqual(lusers.Opers, 1)
# remove oper by quit # remove oper by quit
self.sendLine("qux", "QUIT") self.sendLine("qux", "QUIT")
self.assertDisconnected("qux") self.assertDisconnected("qux")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=1, max_=2) self.assertLusersResult(lusers, unregistered=0, total=1, max_=2)
self.assertEqual(lusers.Opers, 0) self.assertEqual(lusers.Opers, 0)
@ -247,13 +317,13 @@ class ErgoInvisibleDefaultTestCase(LusersTestCase):
@cases.mark_specifications("Ergo") @cases.mark_specifications("Ergo")
def testLusers(self): def testLusers(self):
self.connectClient("bar", name="bar") self.connectClient("bar", name="bar")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=1, max_=1) self.assertLusersResult(lusers, unregistered=0, total=1, max_=1)
self.assertEqual(lusers.GlobalInvisible, 1) self.assertEqual(lusers.GlobalInvisible, 1)
self.assertEqual(lusers.GlobalVisible, 0) self.assertEqual(lusers.GlobalVisible, 0)
self.connectClient("qux", name="qux") self.connectClient("qux", name="qux")
lusers = self.getLusers("qux") lusers = self.getLusers("qux", False)
self.assertLusersResult(lusers, unregistered=0, total=2, max_=2) self.assertLusersResult(lusers, unregistered=0, total=2, max_=2)
self.assertEqual(lusers.GlobalInvisible, 2) self.assertEqual(lusers.GlobalInvisible, 2)
self.assertEqual(lusers.GlobalVisible, 0) self.assertEqual(lusers.GlobalVisible, 0)
@ -261,7 +331,7 @@ class ErgoInvisibleDefaultTestCase(LusersTestCase):
# remove +i with MODE # remove +i with MODE
self.sendLine("bar", "MODE bar -i") self.sendLine("bar", "MODE bar -i")
msgs = self.getMessages("bar") msgs = self.getMessages("bar")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertIn("MODE", {msg.command for msg in msgs}) self.assertIn("MODE", {msg.command for msg in msgs})
self.assertLusersResult(lusers, unregistered=0, total=2, max_=2) self.assertLusersResult(lusers, unregistered=0, total=2, max_=2)
self.assertEqual(lusers.GlobalInvisible, 1) self.assertEqual(lusers.GlobalInvisible, 1)
@ -270,7 +340,7 @@ class ErgoInvisibleDefaultTestCase(LusersTestCase):
# disconnect invisible user # disconnect invisible user
self.sendLine("qux", "QUIT") self.sendLine("qux", "QUIT")
self.assertDisconnected("qux") self.assertDisconnected("qux")
lusers = self.getLusers("bar") lusers = self.getLusers("bar", False)
self.assertLusersResult(lusers, unregistered=0, total=1, max_=2) self.assertLusersResult(lusers, unregistered=0, total=1, max_=2)
self.assertEqual(lusers.GlobalInvisible, 0) self.assertEqual(lusers.GlobalInvisible, 0)
self.assertEqual(lusers.GlobalVisible, 1) self.assertEqual(lusers.GlobalVisible, 1)

View File

@ -1,5 +1,5 @@
""" """
https://ircv3.net/specs/extensions/message-tags.html `IRCv3 message-tags <https://ircv3.net/specs/extensions/message-tags>`_
""" """
import pytest import pytest
@ -10,7 +10,7 @@ from irctest.numerics import ERR_INPUTTOOLONG
from irctest.patma import ANYDICT, ANYSTR, StrRe from irctest.patma import ANYDICT, ANYSTR, StrRe
class MessageTagsTestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class MessageTagsTestCase(cases.BaseServerTestCase):
@pytest.mark.arbitrary_client_tags @pytest.mark.arbitrary_client_tags
@cases.mark_capabilities("message-tags") @cases.mark_capabilities("message-tags")
def testBasic(self): def testBasic(self):

View File

@ -1,6 +1,5 @@
""" """
Section 3.2 of RFC 2812 The PRIVMSG and NOTICE commands.
<https://tools.ietf.org/html/rfc2812#section-3.3>
""" """
from irctest import cases from irctest import cases
@ -52,6 +51,15 @@ class NoticeTestCase(cases.BaseServerTestCase):
) )
@cases.mark_specifications("RFC1459", "RFC2812") @cases.mark_specifications("RFC1459", "RFC2812")
@cases.xfailIfSoftware(
["InspIRCd"],
"replies with ERR_NOSUCHCHANNEL to NOTICE to non-existent channels",
)
@cases.xfailIfSoftware(
["UnrealIRCd"],
"replies with ERR_NOSUCHCHANNEL to NOTICE to non-existent channels: "
"https://bugs.unrealircd.org/view.php?id=5949",
)
def testNoticeNonexistentChannel(self): def testNoticeNonexistentChannel(self):
""" """
"automatic replies must never be "automatic replies must never be
@ -72,6 +80,9 @@ class NoticeTestCase(cases.BaseServerTestCase):
class TagsTestCase(cases.BaseServerTestCase): class TagsTestCase(cases.BaseServerTestCase):
@cases.mark_capabilities("message-tags") @cases.mark_capabilities("message-tags")
@cases.xfailIfSoftware(
["UnrealIRCd"], "https://bugs.unrealircd.org/view.php?id=5947"
)
def testLineTooLong(self): def testLineTooLong(self):
self.connectClient("bar", capabilities=["message-tags"], skip_if_cap_nak=True) self.connectClient("bar", capabilities=["message-tags"], skip_if_cap_nak=True)
self.connectClient( self.connectClient(

View File

@ -1,6 +1,5 @@
""" """
Tests METADATA features. `Deprecated IRCv3 Metadata <https://ircv3.net/specs/core/metadata-3.2>`_
<http://ircv3.net/specs/core/metadata-3.2.html>
""" """
from irctest import cases from irctest import cases

View File

@ -1,9 +1,8 @@
""" """
<http://ircv3.net/specs/core/monitor-3.2.html> `IRCv3 MONITOR <https://ircv3.net/specs/extensions/monitor>`_
""" """
from irctest import cases from irctest import cases, runner
from irctest.basecontrollers import NotImplementedByController
from irctest.client_mock import NoMessageException from irctest.client_mock import NoMessageException
from irctest.numerics import ( from irctest.numerics import (
RPL_ENDOFMONLIST, RPL_ENDOFMONLIST,
@ -17,7 +16,7 @@ from irctest.patma import ANYSTR, StrRe
class MonitorTestCase(cases.BaseServerTestCase): class MonitorTestCase(cases.BaseServerTestCase):
def check_server_support(self): def check_server_support(self):
if "MONITOR" not in self.server_support: if "MONITOR" not in self.server_support:
raise NotImplementedByController("MONITOR") raise runner.IsupportTokenNotSupported("MONITOR")
def assertMononline(self, client, nick, m=None): def assertMononline(self, client, nick, m=None):
if not m: if not m:
@ -188,15 +187,12 @@ class MonitorTestCase(cases.BaseServerTestCase):
self.sendLine(1, "MONITOR + *!username@127.0.0.1") self.sendLine(1, "MONITOR + *!username@127.0.0.1")
try: try:
m = self.getMessage(1) m = self.getMessage(1)
self.assertNotEqual( self.assertMessageMatch(m, command="731")
m.command,
"731",
m,
fail_msg="Got 731 (RPL_MONOFFLINE) after adding a monitor "
"on a mask: {msg}",
)
except NoMessageException: except NoMessageException:
pass pass
else:
m = self.getMessage(1)
self.assertMessageMatch(m, command="731")
self.connectClient("bar") self.connectClient("bar")
try: try:
m = self.getMessage(1) m = self.getMessage(1)

View File

@ -1,6 +1,5 @@
""" """
Tests multi-prefix. `IRCv3 multi-prefix <https://ircv3.net/specs/extensions/multi-prefix>`_
<http://ircv3.net/specs/extensions/multi-prefix-3.1.html>
""" """
from irctest import cases from irctest import cases

View File

@ -1,5 +1,5 @@
""" """
draft/multiline `Draft IRCv3 multiline <https://ircv3.net/specs/extensions/multiline>`_
""" """
from irctest import cases from irctest import cases
@ -12,7 +12,7 @@ CONCAT_TAG = "draft/multiline-concat"
base_caps = ["message-tags", "batch", "echo-message", "server-time", "labeled-response"] base_caps = ["message-tags", "batch", "echo-message", "server-time", "labeled-response"]
class MultilineTestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class MultilineTestCase(cases.BaseServerTestCase):
@cases.mark_capabilities("draft/multiline") @cases.mark_capabilities("draft/multiline")
def testBasic(self): def testBasic(self):
self.connectClient( self.connectClient(

View File

@ -1,9 +1,114 @@
from irctest import cases """
from irctest.numerics import RPL_ENDOFNAMES The NAMES command (`RFC 1459
from irctest.patma import ANYSTR <https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5>`__,
`Modern <https://modern.ircdocs.horse/#names-message>`__)
"""
from irctest import cases, runner
from irctest.numerics import RPL_ENDOFNAMES, RPL_NAMREPLY
from irctest.patma import ANYSTR, StrRe
class NamesTestCase(cases.BaseServerTestCase): class NamesTestCase(cases.BaseServerTestCase):
def _testNames(self, symbol):
self.connectClient("nick1")
self.sendLine(1, "JOIN #chan")
self.getMessages(1)
self.connectClient("nick2")
self.sendLine(2, "JOIN #chan")
self.getMessages(2)
self.getMessages(1)
self.sendLine(1, "NAMES #chan")
# TODO: It is technically allowed to have one line for each;
# but noone does that.
self.assertMessageMatch(
self.getMessage(1),
command=RPL_NAMREPLY,
params=[
"nick1",
*(["="] if symbol else []),
"#chan",
StrRe("(nick2 @nick1|@nick1 nick2)"),
],
)
self.assertMessageMatch(
self.getMessage(1),
command=RPL_ENDOFNAMES,
params=["nick1", "#chan", ANYSTR],
)
@cases.mark_specifications("RFC1459", deprecated=True)
def testNames1459(self):
"""
https://modern.ircdocs.horse/#names-message
https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
"""
self._testNames(symbol=False)
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
def testNames2812(self):
"""
https://modern.ircdocs.horse/#names-message
https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
"""
self._testNames(symbol=True)
def _testNamesMultipleChannels(self, symbol):
self.connectClient("nick1")
if self.targmax.get("NAMES", "1") == "1":
raise runner.OptionalExtensionNotSupported("Multi-target NAMES")
self.sendLine(1, "JOIN #chan1")
self.sendLine(1, "JOIN #chan2")
self.getMessages(1)
self.sendLine(1, "NAMES #chan1,#chan2")
# TODO: order is unspecified
self.assertMessageMatch(
self.getMessage(1),
command=RPL_NAMREPLY,
params=["nick1", *(["="] if symbol else []), "#chan1", "@nick1"],
)
self.assertMessageMatch(
self.getMessage(1),
command=RPL_NAMREPLY,
params=["nick1", *(["="] if symbol else []), "#chan2", "@nick1"],
)
self.assertMessageMatch(
self.getMessage(1),
command=RPL_ENDOFNAMES,
params=["nick1", "#chan1,#chan2", ANYSTR],
)
@cases.mark_isupport("TARGMAX")
@cases.mark_specifications("RFC1459", deprecated=True)
def testNamesMultipleChannels1459(self):
"""
https://modern.ircdocs.horse/#names-message
https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
"""
self._testNamesMultipleChannels(symbol=False)
@cases.mark_isupport("TARGMAX")
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
def testNamesMultipleChannels2812(self):
"""
https://modern.ircdocs.horse/#names-message
https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
"""
self._testNamesMultipleChannels(symbol=True)
@cases.mark_specifications("RFC1459", "RFC2812", "Modern") @cases.mark_specifications("RFC1459", "RFC2812", "Modern")
def testNamesInvalidChannel(self): def testNamesInvalidChannel(self):
""" """
@ -47,3 +152,101 @@ class NamesTestCase(cases.BaseServerTestCase):
command=RPL_ENDOFNAMES, command=RPL_ENDOFNAMES,
params=["foo", "#nonexisting", ANYSTR], params=["foo", "#nonexisting", ANYSTR],
) )
def _testNamesNoArgumentPublic(self, symbol):
self.connectClient("nick1")
self.getMessages(1)
self.sendLine(1, "JOIN #chan1")
self.connectClient("nick2")
self.sendLine(2, "JOIN #chan2")
self.sendLine(2, "MODE #chan2 -sp")
self.getMessages(1)
self.getMessages(2)
self.sendLine(1, "NAMES")
# TODO: order is unspecified
self.assertMessageMatch(
self.getMessage(1),
command=RPL_NAMREPLY,
params=["nick1", *(["="] if symbol else []), "#chan1", "@nick1"],
)
self.assertMessageMatch(
self.getMessage(1),
command=RPL_NAMREPLY,
params=["nick1", *(["="] if symbol else []), "#chan2", "@nick2"],
)
self.assertMessageMatch(
self.getMessage(1),
command=RPL_ENDOFNAMES,
params=["nick1", ANYSTR, ANYSTR],
)
@cases.mark_specifications("RFC1459", deprecated=True)
def testNamesNoArgumentPublic1459(self):
"""
"If no <channel> parameter is given, a list of all channels and their
occupants is returned."
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
"""
self._testNamesNoArgumentPublic(symbol=False)
@cases.mark_specifications("RFC2812", deprecated=True)
def testNamesNoArgumentPublic2812(self):
"""
"If no <channel> parameter is given, a list of all channels and their
occupants is returned."
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
"""
self._testNamesNoArgumentPublic(symbol=True)
def _testNamesNoArgumentPrivate(self, symbol):
self.connectClient("nick1")
self.getMessages(1)
self.sendLine(1, "JOIN #chan1")
self.connectClient("nick2")
self.sendLine(2, "JOIN #chan2")
self.sendLine(2, "MODE #chan2 +sp")
self.getMessages(1)
self.getMessages(2)
self.sendLine(1, "NAMES")
self.assertMessageMatch(
self.getMessage(1),
command=RPL_NAMREPLY,
params=["nick1", *(["="] if symbol else []), "#chan1", "@nick1"],
)
self.assertMessageMatch(
self.getMessage(1),
command=RPL_ENDOFNAMES,
params=["nick1", ANYSTR, ANYSTR],
)
@cases.mark_specifications("RFC1459", deprecated=True)
def testNamesNoArgumentPrivate1459(self):
"""
"If no <channel> parameter is given, a list of all channels and their
occupants is returned. At the end of this list, a list of users who
are visible but either not on any channel or not on a visible channel
are listed as being on `channel' "*"."
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
"""
self._testNamesNoArgumentPrivate(symbol=False)
@cases.mark_specifications("RFC2812", deprecated=True)
def testNamesNoArgumentPrivate2812(self):
"""
"If no <channel> parameter is given, a list of all channels and their
occupants is returned. At the end of this list, a list of users who
are visible but either not on any channel or not on a visible channel
are listed as being on `channel' "*"."
-- https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.5
-- https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.5
"""
self._testNamesNoArgumentPrivate(symbol=True)

View File

@ -1,3 +1,12 @@
"""
The PART command (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-6.1>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-5.2>`__,
`Modern <https://modern.ircdocs.horse/#part-message>`__)
TODO: cross-reference Modern
"""
import time import time
from irctest import cases from irctest import cases

View File

@ -1,3 +1,7 @@
"""
The PING and PONG commands
"""
from irctest import cases from irctest import cases
from irctest.numerics import ERR_NEEDMOREPARAMS, ERR_NOORIGIN from irctest.numerics import ERR_NEEDMOREPARAMS, ERR_NOORIGIN
from irctest.patma import ANYSTR from irctest.patma import ANYSTR

View File

@ -1,3 +1,12 @@
"""
The QUITcommand (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.1.6>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.1>`__,
`Modern <https://modern.ircdocs.horse/#quit-message>`__)
TODO: cross-reference RFC 1459 and Modern
"""
import time import time
from irctest import cases from irctest import cases
@ -7,6 +16,7 @@ from irctest.patma import StrRe
class ChannelQuitTestCase(cases.BaseServerTestCase): class ChannelQuitTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("RFC2812") @cases.mark_specifications("RFC2812")
@cases.xfailIfSoftware(["ircu2", "Nefarious", "snircd"], "ircu2 does not echo QUIT")
def testQuit(self): def testQuit(self):
"""“Once a user has joined a channel, he receives information about """“Once a user has joined a channel, he receives information about
all commands his server receives affecting the channel. This all commands his server receives affecting the channel. This

View File

@ -1,9 +1,12 @@
"""
`Ergo <https://ergo.chat/>`_-specific tests of responses to DoS attacks
using long lines.
"""
from irctest import cases from irctest import cases
class ReadqTestCase(cases.BaseServerTestCase): class ReadqTestCase(cases.BaseServerTestCase):
"""Test responses to DoS attacks using long lines."""
@cases.mark_specifications("Ergo") @cases.mark_specifications("Ergo")
@cases.mark_capabilities("message-tags") @cases.mark_capabilities("message-tags")
def testReadqTags(self): def testReadqTags(self):

View File

@ -1,11 +1,14 @@
""" """
Regression tests for bugs in oragono. Regression tests for bugs in `Ergo <https://ergo.chat/>`_.
""" """
import time from irctest import cases, runner
from irctest.numerics import (
from irctest import cases ERR_ERRONEUSNICKNAME,
from irctest.numerics import ERR_ERRONEUSNICKNAME, ERR_NICKNAMEINUSE, RPL_WELCOME ERR_NICKNAMEINUSE,
RPL_HELLO,
RPL_WELCOME,
)
from irctest.patma import ANYDICT from irctest.patma import ANYDICT
@ -57,6 +60,12 @@ class RegressionsTestCase(cases.BaseServerTestCase):
@cases.mark_capabilities("message-tags", "batch", "echo-message", "server-time") @cases.mark_capabilities("message-tags", "batch", "echo-message", "server-time")
def testTagCap(self): def testTagCap(self):
if self.controller.software_name == "UnrealIRCd":
raise runner.ImplementationChoice(
"Arbitrary +draft/reply values (TODO: adapt this test to use real "
"values so their pass Unreal's validation) "
"https://bugs.unrealircd.org/view.php?id=5948"
)
# regression test for oragono #754 # regression test for oragono #754
self.connectClient( self.connectClient(
"alice", "alice",
@ -99,13 +108,13 @@ class RegressionsTestCase(cases.BaseServerTestCase):
) )
@cases.mark_specifications("RFC1459") @cases.mark_specifications("RFC1459")
@cases.xfailIfSoftware(["ngIRCd"], "wat")
def testStarNick(self): def testStarNick(self):
self.addClient(1) self.addClient(1)
self.sendLine(1, "NICK *") self.sendLine(1, "NICK *")
self.sendLine(1, "USER u s e r") self.sendLine(1, "USER u s e r")
replies = {"NOTICE"} replies = {"NOTICE"}
time.sleep(2) # give time to slow servers, like irc2 to reply while replies <= {"NOTICE", RPL_HELLO}:
while replies == {"NOTICE"}:
replies = set(msg.command for msg in self.getMessages(1, synchronize=False)) replies = set(msg.command for msg in self.getMessages(1, synchronize=False))
self.assertIn(ERR_ERRONEUSNICKNAME, replies) self.assertIn(ERR_ERRONEUSNICKNAME, replies)
self.assertNotIn(RPL_WELCOME, replies) self.assertNotIn(RPL_WELCOME, replies)

View File

@ -1,3 +1,7 @@
"""
RELAYMSG command of `Ergo <https://ergo.chat/>`_
"""
from irctest import cases from irctest import cases
from irctest.irc_utils.junkdrawer import random_name from irctest.irc_utils.junkdrawer import random_name
from irctest.patma import ANYSTR from irctest.patma import ANYSTR

View File

@ -1,3 +1,7 @@
"""
Roleplay features of `Ergo <https://ergo.chat/>`_
"""
from irctest import cases from irctest import cases
from irctest.irc_utils.junkdrawer import random_name from irctest.irc_utils.junkdrawer import random_name
from irctest.numerics import ERR_CANNOTSENDRP from irctest.numerics import ERR_CANNOTSENDRP

View File

@ -12,9 +12,9 @@ class RegistrationTestCase(cases.BaseServerTestCase):
@cases.mark_services @cases.mark_services
class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class SaslTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
def testPlain(self): def testPlain(self):
"""PLAIN authentication with correct username/password.""" """PLAIN authentication with correct username/password."""
self.controller.registerUser(self, "foo", "sesame") self.controller.registerUser(self, "foo", "sesame")
@ -54,7 +54,7 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
) )
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
def testPlainNonAscii(self): def testPlainNonAscii(self):
password = "é" * 100 password = "é" * 100
authstring = base64.b64encode( authstring = base64.b64encode(
@ -82,7 +82,7 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
) )
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
def testPlainNoAuthzid(self): def testPlainNoAuthzid(self):
"""“message = [authzid] UTF8NUL authcid UTF8NUL passwd """“message = [authzid] UTF8NUL authcid UTF8NUL passwd
@ -170,7 +170,22 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
) )
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
@cases.xfailIf(
lambda self: (
self.controller.services_controller is not None
and self.controller.services_controller.software_name == "Anope"
),
"Anope does not handle split AUTHENTICATE (reported on IRC)",
)
@cases.xfailIf(
lambda self: (
self.controller.services_controller is not None
and self.controller.services_controller.software_name == "Dlk-Services"
),
"Dlk does not handle split AUTHENTICATE "
"https://github.com/DalekIRC/Dalek-Services/issues/28",
)
def testPlainLarge(self): def testPlainLarge(self):
"""Test the client splits large AUTHENTICATE messages whose payload """Test the client splits large AUTHENTICATE messages whose payload
is not a multiple of 400. is not a multiple of 400.
@ -232,7 +247,14 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
# message's length too big for it to be valid. # message's length too big for it to be valid.
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
@cases.xfailIf(
lambda self: (
self.controller.services_controller is not None
and self.controller.services_controller.software_name == "Anope"
),
"Anope does not handle split AUTHENTICATE (reported on IRC)",
)
def testPlainLargeEquals400(self): def testPlainLargeEquals400(self):
"""Test the client splits large AUTHENTICATE messages whose payload """Test the client splits large AUTHENTICATE messages whose payload
is not a multiple of 400. is not a multiple of 400.
@ -277,7 +299,7 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
# message's length too big for it to be valid. # message's length too big for it to be valid.
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.OptionalityHelper.skipUnlessHasMechanism("SCRAM-SHA-256") @cases.skipUnlessHasMechanism("SCRAM-SHA-256")
def testScramSha256Success(self): def testScramSha256Success(self):
self.controller.registerUser(self, "Scramtest", "sesame") self.controller.registerUser(self, "Scramtest", "sesame")
@ -333,7 +355,7 @@ class SaslTestCase(cases.BaseServerTestCase, cases.OptionalityHelper):
self.confirmSuccessfulAuth() self.confirmSuccessfulAuth()
@cases.mark_specifications("IRCv3") @cases.mark_specifications("IRCv3")
@cases.OptionalityHelper.skipUnlessHasMechanism("SCRAM-SHA-256") @cases.skipUnlessHasMechanism("SCRAM-SHA-256")
def testScramSha256Failure(self): def testScramSha256Failure(self):
self.controller.registerUser(self, "Scramtest", "sesame") self.controller.registerUser(self, "Scramtest", "sesame")

View File

@ -1,3 +1,10 @@
"""
STATUSMSG ISUPPORT token and related PRIVMSG (`Modern
<https://modern.ircdocs.horse/#statusmsg-parameter>`__)
TODO: cross-reference Modern
"""
from irctest import cases, runner from irctest import cases, runner
from irctest.numerics import RPL_NAMREPLY from irctest.numerics import RPL_NAMREPLY
@ -10,6 +17,11 @@ class StatusmsgTestCase(cases.BaseServerTestCase):
self.assertEqual(self.server_support["STATUSMSG"], "~&@%+") self.assertEqual(self.server_support["STATUSMSG"], "~&@%+")
@cases.mark_isupport("STATUSMSG") @cases.mark_isupport("STATUSMSG")
@cases.xfailIfSoftware(
["ircu2", "Nefarious", "snircd"],
"STATUSMSG is present in ISUPPORT, but it not actually supported as PRIVMSG "
"target (only for WALLCOPS/WALLCHOPS/...)",
)
def testStatusmsgFromOp(self): def testStatusmsgFromOp(self):
"""Test that STATUSMSG are sent to the intended recipients, """Test that STATUSMSG are sent to the intended recipients,
with the intended prefixes.""" with the intended prefixes."""
@ -61,6 +73,11 @@ class StatusmsgTestCase(cases.BaseServerTestCase):
self.assertEqual(len(unprivilegedMessages), 0) self.assertEqual(len(unprivilegedMessages), 0)
@cases.mark_isupport("STATUSMSG") @cases.mark_isupport("STATUSMSG")
@cases.xfailIfSoftware(
["ircu2", "Nefarious", "snircd"],
"STATUSMSG is present in ISUPPORT, but it not actually supported as PRIVMSG "
"target (only for WALLCOPS/WALLCHOPS/...)",
)
def testStatusmsgFromRegular(self): def testStatusmsgFromRegular(self):
"""Test that STATUSMSG are sent to the intended recipients, """Test that STATUSMSG are sent to the intended recipients,
with the intended prefixes.""" with the intended prefixes."""

View File

@ -0,0 +1,48 @@
import math
import time
from irctest import cases
from irctest.numerics import RPL_TIME
from irctest.patma import ANYSTR, StrRe
class TimeTestCase(cases.BaseServerTestCase):
def testTime(self):
self.connectClient("user")
time_before = math.floor(time.time())
self.sendLine(1, "TIME")
msg = self.getMessage(1)
time_after = math.ceil(time.time())
if len(msg.params) == 5:
# ircu2, snircd
self.assertMessageMatch(
msg,
command=RPL_TIME,
params=["user", "My.Little.Server", StrRe("[0-9]+"), "0", ANYSTR],
)
self.assertIn(
int(msg.params[2]),
range(time_before, time_after + 1),
"Timestamp not in expected range",
)
elif len(msg.params) == 4:
# bahamut
self.assertMessageMatch(
msg,
command=RPL_TIME,
params=["user", "My.Little.Server", StrRe("[0-9]+"), ANYSTR],
)
self.assertIn(
int(msg.params[2]),
range(time_before, time_after + 1),
"Timestamp not in expected range",
)
else:
# Common case
self.assertMessageMatch(
msg, command=RPL_TIME, params=["user", "My.Little.Server", ANYSTR]
)

View File

@ -1,3 +1,10 @@
"""
The TOPIC command (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.2.1>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.2.1>`__,
`Modern <https://modern.ircdocs.horse/#topic-message>`__)
"""
from irctest import cases, client_mock, runner from irctest import cases, client_mock, runner
from irctest.numerics import ERR_CHANOPRIVSNEEDED, RPL_NOTOPIC, RPL_TOPIC, RPL_TOPICTIME from irctest.numerics import ERR_CHANOPRIVSNEEDED, RPL_NOTOPIC, RPL_TOPIC, RPL_TOPICTIME

View File

@ -0,0 +1,124 @@
"""
Test the registered-only DM user mode (commonly +R).
"""
from irctest import cases
from irctest.numerics import ERR_NEEDREGGEDNICK
@cases.mark_services
class RegisteredOnlyUmodeTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("Ergo")
def testRegisteredOnlyUserMode(self):
"""Test the +R registered-only mode."""
self.controller.registerUser(self, "evan", "sesame")
self.controller.registerUser(self, "carmen", "pink")
self.connectClient(
"evan",
name="evan",
account="evan",
password="sesame",
capabilities=["sasl"],
)
self.connectClient("shivaram", name="shivaram")
self.sendLine("evan", "MODE evan +R")
self.assertMessageMatch(
self.getMessage("evan"), command="MODE", params=["evan", "+R"]
)
# this DM should be blocked by +R registered-only
self.getMessages("shivaram")
self.sendLine("shivaram", "PRIVMSG evan :hey there")
self.assertMessageMatch(
self.getMessage("shivaram"),
command=ERR_NEEDREGGEDNICK,
)
self.assertEqual(self.getMessages("evan"), [])
self.connectClient(
"carmen",
name="carmen",
account="carmen",
password="pink",
capabilities=["sasl"],
)
self.getMessages("evan")
self.sendLine("carmen", "PRIVMSG evan :hey there")
self.assertEqual(self.getMessages("carmen"), [])
# this message should go through fine:
self.assertMessageMatch(
self.getMessage("evan"),
command="PRIVMSG",
params=["evan", "hey there"],
)
@cases.mark_specifications("Ergo")
def testRegisteredOnlyUserModeAcceptCommand(self):
"""Test that the ACCEPT command can authorize another user
to send the accept-er direct messages, overriding the
+R registered-only mode."""
self.controller.registerUser(self, "evan", "sesame")
self.connectClient(
"evan",
name="evan",
account="evan",
password="sesame",
capabilities=["sasl"],
)
self.connectClient("shivaram", name="shivaram")
self.sendLine("evan", "MODE evan +R")
self.assertMessageMatch(
self.getMessage("evan"), command="MODE", params=["evan", "+R"]
)
self.sendLine("evan", "ACCEPT shivaram")
self.getMessages("evan")
self.sendLine("shivaram", "PRIVMSG evan :hey there")
self.assertEqual(self.getMessages("shivaram"), [])
self.assertMessageMatch(
self.getMessage("evan"),
command="PRIVMSG",
params=["evan", "hey there"],
)
self.sendLine("evan", "ACCEPT -shivaram")
self.getMessages("evan")
self.sendLine("shivaram", "PRIVMSG evan :how's it going")
self.assertMessageMatch(
self.getMessage("shivaram"),
command=ERR_NEEDREGGEDNICK,
)
self.assertEqual(self.getMessages("evan"), [])
@cases.mark_specifications("Ergo")
def testRegisteredOnlyUserModeAutoAcceptOnDM(self):
"""Test that sending someone a DM automatically authorizes them to
reply, overriding the +R registered-only mode."""
self.controller.registerUser(self, "evan", "sesame")
self.connectClient(
"evan",
name="evan",
account="evan",
password="sesame",
capabilities=["sasl"],
)
self.connectClient("shivaram", name="shivaram")
self.sendLine("evan", "MODE evan +R")
self.assertMessageMatch(
self.getMessage("evan"), command="MODE", params=["evan", "+R"]
)
self.sendLine("evan", "PRIVMSG shivaram :hey there")
self.getMessages("evan")
self.assertMessageMatch(
self.getMessage("shivaram"),
command="PRIVMSG",
params=["shivaram", "hey there"],
)
self.sendLine("shivaram", "PRIVMSG evan :how's it going")
self.assertEqual(self.getMessages("shivaram"), [])
self.assertMessageMatch(
self.getMessage("evan"),
command="PRIVMSG",
params=["evan", "how's it going"],
)

View File

@ -1,8 +1,15 @@
"""
`Ergo <https://ergo.chat/>`_-specific tests of non-Unicode filtering
TODO: turn this into a test of `IRCv3 UTF8ONLY
<https://ircv3.net/specs/extensions/utf8-only>`_
"""
from irctest import cases from irctest import cases
from irctest.patma import ANYSTR from irctest.patma import ANYSTR
class Utf8TestCase(cases.BaseServerTestCase, cases.OptionalityHelper): class Utf8TestCase(cases.BaseServerTestCase):
@cases.mark_specifications("Ergo") @cases.mark_specifications("Ergo")
def testUtf8Validation(self): def testUtf8Validation(self):
self.connectClient( self.connectClient(

View File

@ -1,3 +1,9 @@
"""
The WALLOPS command (`RFC 2812
<https://datatracker.ietf.org/doc/html/rfc2812#section-3.7>`__,
`Modern <https://modern.ircdocs.horse/#wallops-message>`__)
"""
from irctest import cases, runner from irctest import cases, runner
from irctest.numerics import ERR_NOPRIVILEGES, ERR_UNKNOWNCOMMAND, RPL_YOUREOPER from irctest.numerics import ERR_NOPRIVILEGES, ERR_UNKNOWNCOMMAND, RPL_YOUREOPER
from irctest.patma import ANYSTR, StrRe from irctest.patma import ANYSTR, StrRe
@ -38,7 +44,7 @@ class WallopsTestCase(cases.BaseServerTestCase):
messages = self.getMessages(1) messages = self.getMessages(1)
if ERR_UNKNOWNCOMMAND in (message.command for message in messages): if ERR_UNKNOWNCOMMAND in (message.command for message in messages):
raise runner.NotImplementedByController("WALLOPS") raise runner.OptionalCommandNotSupported("WALLOPS")
for message in messages: for message in messages:
self.assertMessageMatch( self.assertMessageMatch(
message, message,
@ -60,6 +66,9 @@ class WallopsTestCase(cases.BaseServerTestCase):
) )
@cases.mark_specifications("Modern") @cases.mark_specifications("Modern")
@cases.xfailIfSoftware(
["irc2"], "irc2 ignores the command instead of replying ERR_UNKNOWNCOMMAND"
)
def testWallopsPrivileges(self): def testWallopsPrivileges(self):
""" """
https://github.com/ircdocs/modern-irc/pull/118 https://github.com/ircdocs/modern-irc/pull/118
@ -68,7 +77,7 @@ class WallopsTestCase(cases.BaseServerTestCase):
self.sendLine(1, "WALLOPS :hi everyone") self.sendLine(1, "WALLOPS :hi everyone")
message = self.getMessage(1) message = self.getMessage(1)
if message.command == ERR_UNKNOWNCOMMAND: if message.command == ERR_UNKNOWNCOMMAND:
raise runner.NotImplementedByController("WALLOPS") raise runner.OptionalCommandNotSupported("WALLOPS")
self.assertMessageMatch( self.assertMessageMatch(
message, command=ERR_NOPRIVILEGES, params=["nick1", ANYSTR] message, command=ERR_NOPRIVILEGES, params=["nick1", ANYSTR]
) )

536
irctest/server_tests/who.py Normal file
View File

@ -0,0 +1,536 @@
"""
The WHO command (`Modern <https://modern.ircdocs.horse/#who-message>`__)
and `IRCv3 WHOX <https://ircv3.net/specs/extensions/whox>`_
TODO: cross-reference RFC 1459 and RFC 2812
"""
import re
import pytest
from irctest import cases, runner
from irctest.numerics import RPL_ENDOFWHO, RPL_WHOREPLY, RPL_WHOSPCRPL, RPL_YOUREOPER
from irctest.patma import ANYSTR, InsensitiveStr, StrRe
def realname_regexp(realname):
return (
"[0-9]+ " # is 0 for every IRCd I can find, except ircu2 (which returns 3)
+ "(0042 )?" # for irc2...
+ re.escape(realname)
)
class BaseWhoTestCase:
def _init(self, auth=False):
self.nick = "coolNick"
self.username = "myusernam" # may be truncated if longer than this
self.realname = "My UniqueReal Name"
self.addClient()
if auth:
self.controller.registerUser(self, "coolAcct", "sesame")
self.requestCapabilities(1, ["sasl"], skip_if_cap_nak=True)
self.authenticateClient(1, "coolAcct", "sesame")
self.sendLine(1, f"NICK {self.nick}")
self.sendLine(1, f"USER {self.username} 0 * :{self.realname}")
if auth:
self.sendLine(1, "CAP END")
self.skipToWelcome(1)
self.getMessages(1)
self.sendLine(1, "JOIN #chan")
self.getMessages(1)
self.connectClient("otherNick")
self.getMessages(2)
self.sendLine(2, "JOIN #chan")
self.getMessages(2)
def _checkReply(self, reply, flags):
host_re = "[0-9A-Za-z_:.-]+"
if reply.params[1] == "*":
# Unreal, ...
self.assertMessageMatch(
reply,
command=RPL_WHOREPLY,
params=[
"otherNick",
"*", # no chan
StrRe("~?" + self.username),
StrRe(host_re),
"My.Little.Server",
"coolNick",
flags,
StrRe(realname_regexp(self.realname)),
],
)
else:
# Solanum, Insp, ...
self.assertMessageMatch(
reply,
command=RPL_WHOREPLY,
params=[
"otherNick",
"#chan",
StrRe("~?" + self.username),
StrRe(host_re),
"My.Little.Server",
"coolNick",
flags + "@",
StrRe(realname_regexp(self.realname)),
],
)
class WhoTestCase(BaseWhoTestCase, cases.BaseServerTestCase):
@cases.mark_specifications("Modern")
def testWhoStar(self):
if self.controller.software_name == "Bahamut":
raise runner.OptionalExtensionNotSupported("WHO mask")
self._init()
self.sendLine(2, "WHO *")
messages = self.getMessages(2)
self.assertEqual(len(messages), 3, "Unexpected number of messages")
(*replies, end) = messages
# Get them in deterministic order
replies.sort(key=lambda msg: msg.params[5])
self._checkReply(replies[0], "H")
# " `<mask>` MUST be exactly the `<mask>` parameter sent by the client
# in its `WHO` message. This means the case MUST be preserved."
# -- https://github.com/ircdocs/modern-irc/pull/138/files
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", "*", ANYSTR],
)
@pytest.mark.parametrize(
"mask", ["coolNick", "coolnick", "coolni*"], ids=["exact", "casefolded", "mask"]
)
@cases.mark_specifications("Modern")
def testWhoNick(self, mask):
if "*" in mask and self.controller.software_name == "Bahamut":
raise runner.OptionalExtensionNotSupported("WHO mask")
self._init()
self.sendLine(2, f"WHO {mask}")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self._checkReply(reply, "H")
# " `<mask>` MUST be exactly the `<mask>` parameter sent by the client
# in its `WHO` message. This means the case MUST be preserved."
# -- https://github.com/ircdocs/modern-irc/pull/138/files
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr(mask), ANYSTR],
)
@pytest.mark.skip("Not consistently implemented")
@pytest.mark.parametrize(
"mask",
["*usernam", "*UniqueReal*", "127.0.0.1"],
ids=["username", "realname-mask", "hostname"],
)
def testWhoUsernameRealName(self, mask):
if "*" in mask and self.controller.software_name == "Bahamut":
raise runner.OptionalExtensionNotSupported("WHO mask")
self._init()
self.sendLine(2, f"WHO :{mask}")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self._checkReply(reply, "H")
# " `<mask>` MUST be exactly the `<mask>` parameter sent by the client
# in its `WHO` message. This means the case MUST be preserved."
# -- https://github.com/ircdocs/modern-irc/pull/138/files
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr(mask), ANYSTR],
)
@pytest.mark.skip("Not consistently implemented")
def testWhoRealNameSpaces(self):
self._init()
self.sendLine(2, "WHO :*UniqueReal Name")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self._checkReply(reply, "H")
# What to do here? This?
# self.assertMessageMatch(
# end,
# command=RPL_ENDOFWHO,
# params=[
# "otherNick",
# InsensitiveStr("*UniqueReal"),
# InsensitiveStr("Name"),
# ANYSTR,
# ],
# )
@pytest.mark.parametrize(
"mask", ["coolNick", "coolnick", "coolni*"], ids=["exact", "casefolded", "mask"]
)
@cases.mark_specifications("Modern")
def testWhoNickAway(self, mask):
if "*" in mask and self.controller.software_name == "Bahamut":
raise runner.OptionalExtensionNotSupported("WHO mask")
self._init()
self.sendLine(1, "AWAY :be right back")
self.getMessages(1)
self.getMessages(2)
self.sendLine(2, f"WHO {mask}")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self._checkReply(reply, "G")
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr(mask), ANYSTR],
)
@pytest.mark.parametrize(
"mask", ["coolNick", "coolnick", "coolni*"], ids=["exact", "casefolded", "mask"]
)
@cases.mark_specifications("Modern")
def testWhoNickOper(self, mask):
if "*" in mask and self.controller.software_name == "Bahamut":
raise runner.OptionalExtensionNotSupported("WHO mask")
self._init()
self.sendLine(1, "OPER operuser operpassword")
self.assertIn(
RPL_YOUREOPER,
[m.command for m in self.getMessages(1)],
fail_msg="OPER failed",
)
self.getMessages(2)
self.sendLine(2, f"WHO {mask}")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self._checkReply(reply, "H*")
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr(mask), ANYSTR],
)
@pytest.mark.parametrize(
"mask", ["coolNick", "coolnick", "coolni*"], ids=["exact", "casefolded", "mask"]
)
@cases.mark_specifications("Modern")
def testWhoNickAwayAndOper(self, mask):
if "*" in mask and self.controller.software_name == "Bahamut":
raise runner.OptionalExtensionNotSupported("WHO mask")
self._init()
self.sendLine(1, "OPER operuser operpassword")
self.assertIn(
RPL_YOUREOPER,
[m.command for m in self.getMessages(1)],
fail_msg="OPER failed",
)
self.sendLine(1, "AWAY :be right back")
self.getMessages(1)
self.getMessages(2)
self.sendLine(2, f"WHO {mask}")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self._checkReply(reply, "G*")
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr(mask), ANYSTR],
)
@pytest.mark.parametrize("mask", ["#chan", "#CHAN"], ids=["exact", "casefolded"])
@cases.mark_specifications("Modern")
def testWhoChan(self, mask):
if "*" in mask and self.controller.software_name == "Bahamut":
raise runner.OptionalExtensionNotSupported("WHO mask")
self._init()
self.sendLine(1, "OPER operuser operpassword")
self.assertIn(
RPL_YOUREOPER,
[m.command for m in self.getMessages(1)],
fail_msg="OPER failed",
)
self.sendLine(1, "AWAY :be right back")
self.getMessages(1)
self.getMessages(2)
self.sendLine(2, f"WHO {mask}")
messages = self.getMessages(2)
self.assertEqual(len(messages), 3, "Unexpected number of messages")
(*replies, end) = messages
# Get them in deterministic order
replies.sort(key=lambda msg: msg.params[5])
host_re = "[0-9A-Za-z_:.-]+"
self.assertMessageMatch(
replies[0],
command=RPL_WHOREPLY,
params=[
"otherNick",
"#chan",
StrRe("~?" + self.username),
StrRe(host_re),
"My.Little.Server",
"coolNick",
"G*@",
StrRe(realname_regexp(self.realname)),
],
)
self.assertMessageMatch(
replies[1],
command=RPL_WHOREPLY,
params=[
"otherNick",
"#chan",
ANYSTR,
ANYSTR,
"My.Little.Server",
"otherNick",
"H",
StrRe("[0-9]+ .*"),
],
)
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr(mask), ANYSTR],
)
@cases.mark_specifications("IRCv3")
@cases.mark_isupport("WHOX")
def testWhoxFull(self):
"""https://github.com/ircv3/ircv3-specifications/pull/482"""
self._testWhoxFull("%tcuihsnfdlaor,123")
@cases.mark_specifications("IRCv3")
@cases.mark_isupport("WHOX")
def testWhoxFullReversed(self):
"""https://github.com/ircv3/ircv3-specifications/pull/482"""
self._testWhoxFull("%" + "".join(reversed("tcuihsnfdlaor")) + ",123")
def _testWhoxFull(self, chars):
self._init()
if "WHOX" not in self.server_support:
raise runner.IsupportTokenNotSupported("WHOX")
self.sendLine(2, f"WHO coolNick {chars}")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self.assertMessageMatch(
reply,
command=RPL_WHOSPCRPL,
params=[
"otherNick",
"123",
StrRe(r"(#chan|\*)"),
StrRe("~?myusernam"),
ANYSTR,
ANYSTR,
"My.Little.Server",
"coolNick",
StrRe("H@?"),
ANYSTR, # hopcount
StrRe("[0-9]"), # seconds idle
"0", # account name
ANYSTR, # op level
"My UniqueReal Name",
],
)
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr("coolNick"), ANYSTR],
)
def testWhoxToken(self):
"""https://github.com/ircv3/ircv3-specifications/pull/482"""
self._init()
if "WHOX" not in self.server_support:
raise runner.IsupportTokenNotSupported("WHOX")
self.sendLine(2, "WHO coolNick %tn,321")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self.assertMessageMatch(
reply,
command=RPL_WHOSPCRPL,
params=[
"otherNick",
"321",
"coolNick",
],
)
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr("coolNick"), ANYSTR],
)
@cases.mark_services
class WhoServicesTestCase(BaseWhoTestCase, cases.BaseServerTestCase):
@cases.mark_specifications("IRCv3")
@cases.mark_isupport("WHOX")
def testWhoxAccount(self):
self._init(auth=True)
if "WHOX" not in self.server_support:
raise runner.IsupportTokenNotSupported("WHOX")
self.sendLine(2, "WHO coolNick %na")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self.assertMessageMatch(
reply,
command=RPL_WHOSPCRPL,
params=[
"otherNick",
"coolNick",
"coolAcct",
],
)
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr("coolNick"), ANYSTR],
)
@cases.mark_specifications("IRCv3")
@cases.mark_isupport("WHOX")
def testWhoxNoAccount(self):
self._init(auth=False)
if "WHOX" not in self.server_support:
raise runner.IsupportTokenNotSupported("WHOX")
self.sendLine(2, "WHO coolNick %na")
messages = self.getMessages(2)
self.assertEqual(len(messages), 2, "Unexpected number of messages")
(reply, end) = messages
self.assertMessageMatch(
reply,
command=RPL_WHOSPCRPL,
params=[
"otherNick",
"coolNick",
"0",
],
)
self.assertMessageMatch(
end,
command=RPL_ENDOFWHO,
params=["otherNick", InsensitiveStr("coolNick"), ANYSTR],
)
class WhoInvisibleTestCase(cases.BaseServerTestCase):
@cases.mark_specifications("Modern")
def testWhoInvisible(self):
if self.controller.software_name == "Bahamut":
raise runner.OptionalExtensionNotSupported("WHO mask")
self.connectClient("evan", name="evan")
self.sendLine("evan", "MODE evan +i")
self.getMessages("evan")
self.connectClient("shivaram", name="shivaram")
self.getMessages("shivaram")
self.sendLine("shivaram", "WHO eva*")
reply_cmds = {msg.command for msg in self.getMessages("shivaram")}
self.assertEqual(reply_cmds, {RPL_ENDOFWHO})
# invisibility should not be respected for plain nicknames, only for masks:
self.sendLine("shivaram", "WHO evan")
replies = self.getMessages("shivaram")
reply_cmds = {msg.command for msg in replies}
self.assertEqual(reply_cmds, {RPL_WHOREPLY, RPL_ENDOFWHO})
# invisibility should not be respected if the users share a channel
self.joinChannel("evan", "#test")
self.joinChannel("shivaram", "#test")
self.sendLine("shivaram", "WHO eva*")
replies = self.getMessages("shivaram")
reply_cmds = {msg.command for msg in replies}
self.assertEqual(reply_cmds, {RPL_WHOREPLY, RPL_ENDOFWHO})

View File

@ -1,3 +1,9 @@
"""
The WHOIS command (`Modern <https://modern.ircdocs.horse/#whois-message>`__)
TODO: cross-reference RFC 1459 and RFC 2812
"""
import pytest import pytest
from irctest import cases from irctest import cases
@ -23,6 +29,9 @@ from irctest.patma import ANYSTR, StrRe
class _WhoisTestMixin(cases.BaseServerTestCase): class _WhoisTestMixin(cases.BaseServerTestCase):
def _testWhoisNumerics(self, authenticate, away, oper): def _testWhoisNumerics(self, authenticate, away, oper):
if oper and self.controller.software_name == "Charybdis":
pytest.xfail("charybdis uses RPL_WHOISSPECIAL instead of RPL_WHOISOPERATOR")
if authenticate: if authenticate:
self.connectClient("nick1") self.connectClient("nick1")
self.controller.registerUser(self, "val", "sesame") self.controller.registerUser(self, "val", "sesame")
@ -62,7 +71,10 @@ class _WhoisTestMixin(cases.BaseServerTestCase):
last_message, last_message,
command=RPL_ENDOFWHOIS, command=RPL_ENDOFWHOIS,
params=["nick1", "nick2", ANYSTR], params=["nick1", "nick2", ANYSTR],
fail_msg=f"Last message was not RPL_ENDOFWHOIS ({RPL_ENDOFWHOIS})", fail_msg=(
f"Expected RPL_ENDOFWHOIS ({RPL_ENDOFWHOIS}) as last message, "
f"got {{msg}}"
),
) )
unexpected_messages = [] unexpected_messages = []
@ -158,7 +170,7 @@ class _WhoisTestMixin(cases.BaseServerTestCase):
) )
class WhoisTestCase(_WhoisTestMixin, cases.BaseServerTestCase, cases.OptionalityHelper): class WhoisTestCase(_WhoisTestMixin, cases.BaseServerTestCase):
@pytest.mark.parametrize( @pytest.mark.parametrize(
"server", "server",
["", "My.Little.Server", "coolNick"], ["", "My.Little.Server", "coolNick"],
@ -199,22 +211,20 @@ class WhoisTestCase(_WhoisTestMixin, cases.BaseServerTestCase, cases.Optionality
def testWhoisNumerics(self, away, oper): def testWhoisNumerics(self, away, oper):
"""Tests all numerics are in the exhaustive list defined in the Modern spec. """Tests all numerics are in the exhaustive list defined in the Modern spec.
TBD modern PR""" <https://modern.ircdocs.horse/#whois-message>"""
self._testWhoisNumerics(authenticate=False, away=away, oper=oper) self._testWhoisNumerics(oper=oper, authenticate=False, away=away)
@cases.mark_services @cases.mark_services
class ServicesWhoisTestCase( class ServicesWhoisTestCase(_WhoisTestMixin, cases.BaseServerTestCase):
_WhoisTestMixin, cases.BaseServerTestCase, cases.OptionalityHelper
):
@pytest.mark.parametrize("oper", [False, True], ids=["normal", "oper"]) @pytest.mark.parametrize("oper", [False, True], ids=["normal", "oper"])
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
@cases.mark_specifications("Modern") @cases.mark_specifications("Modern")
def testWhoisNumerics(self, oper): def testWhoisNumerics(self, oper):
"""Tests all numerics are in the exhaustive list defined in the Modern spec, """Tests all numerics are in the exhaustive list defined in the Modern spec,
on an authenticated user. on an authenticated user.
TBD modern PR""" <https://modern.ircdocs.horse/#whois-message>"""
self._testWhoisNumerics(oper=oper, authenticate=True, away=False) self._testWhoisNumerics(oper=oper, authenticate=True, away=False)
@cases.mark_specifications("Ergo") @cases.mark_specifications("Ergo")
@ -291,7 +301,7 @@ class ServicesWhoisTestCase(
"RPL_WHOISCHANNELS should be sent for a non-invisible nick", "RPL_WHOISCHANNELS should be sent for a non-invisible nick",
) )
@cases.OptionalityHelper.skipUnlessHasMechanism("PLAIN") @cases.skipUnlessHasMechanism("PLAIN")
@cases.mark_specifications("ircdocs") @cases.mark_specifications("ircdocs")
def testWhoisAccount(self): def testWhoisAccount(self):
"""Test numeric 330, RPL_WHOISACCOUNT. """Test numeric 330, RPL_WHOISACCOUNT.

View File

@ -0,0 +1,474 @@
"""
The WHOSWAS command (`RFC 1459
<https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3>`__,
`RFC 2812 <https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3>`__,
`Modern <https://modern.ircdocs.horse/#whowas-message>`__)
TODO: cross-reference Modern
"""
import pytest
from irctest import cases, runner
from irctest.exceptions import ConnectionClosed
from irctest.numerics import (
ERR_NEEDMOREPARAMS,
ERR_NONICKNAMEGIVEN,
ERR_WASNOSUCHNICK,
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}"
)
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
def testWhowasEnd(self):
"""
"At the end of all reply batches, there must be RPL_ENDOFWHOWAS"
-- https://datatracker.ietf.org/doc/html/rfc1459#page-50
-- https://datatracker.ietf.org/doc/html/rfc2812#page-45
"Servers MUST reply with either ERR_WASNOSUCHNICK or [...],
both followed with RPL_ENDOFWHOWAS"
-- https://modern.ircdocs.horse/#whowas-message
"""
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})",
)
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", "Modern")
@cases.xfailIfSoftware(
["InspIRCd"],
"Feature not released yet: https://github.com/inspircd/inspircd/pull/1967",
)
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
-- https://modern.ircdocs.horse/#whowas-message
"""
self._testWhowasMultiple(second_result=True, whowas_command="WHOWAS nick2")
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
@cases.xfailIfSoftware(
["InspIRCd"],
"Feature not released yet: https://github.com/inspircd/inspircd/pull/1968",
)
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
-- https://modern.ircdocs.horse/#whowas-message
"""
self._testWhowasMultiple(second_result=False, whowas_command="WHOWAS nick2 1")
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
@cases.xfailIfSoftware(
["InspIRCd"],
"Feature not released yet: https://github.com/inspircd/inspircd/pull/1968",
)
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
-- https://modern.ircdocs.horse/#whowas-message
"""
self._testWhowasMultiple(second_result=True, whowas_command="WHOWAS nick2 2")
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
@cases.xfailIfSoftware(
["InspIRCd"],
"Feature not released yet: https://github.com/inspircd/inspircd/pull/1968",
)
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
"If given, <count> SHOULD be a positive number. Otherwise, a full search
"is done.
-- https://modern.ircdocs.horse/#whowas-message
"""
self._testWhowasMultiple(second_result=True, whowas_command="WHOWAS nick2 -1")
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
@cases.xfailIfSoftware(
["ircu2"], "Fix not released yet: https://github.com/UndernetIRC/ircu2/pull/19"
)
@cases.xfailIfSoftware(
["InspIRCd"],
"Feature not released yet: https://github.com/inspircd/inspircd/pull/1967",
)
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
"If given, <count> SHOULD be a positive number. Otherwise, a full search
"is done.
-- https://modern.ircdocs.horse/#whowas-message
"""
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
-- https://modern.ircdocs.horse/#whowas-message
"""
if self.controller.software_name == "Bahamut":
raise runner.OptionalExtensionNotSupported("WHOWAS mask")
self._testWhowasMultiple(second_result=True, whowas_command="WHOWAS *ck2")
@cases.mark_specifications("RFC1459", "RFC2812", deprecated=True)
def testWhowasNoParamRfc(self):
"""
https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3
https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
and:
"At the end of all reply batches, there must be RPL_ENDOFWHOWAS
(even if there was only one reply and it was an error)."
-- https://datatracker.ietf.org/doc/html/rfc1459#page-50
-- https://datatracker.ietf.org/doc/html/rfc2812#page-45
"""
# But no one seems to follow this. Most implementations use ERR_NEEDMOREPARAMS
# instead of ERR_NONICKNAMEGIVEN; and I couldn't find any that returns
# RPL_ENDOFWHOWAS either way.
self.connectClient("nick1")
self.sendLine(1, "WHOWAS")
self.assertMessageMatch(
self.getMessage(1),
command=ERR_NONICKNAMEGIVEN,
params=["nick1", ANYSTR],
)
self.assertMessageMatch(
self.getMessage(1),
command=RPL_ENDOFWHOWAS,
params=["nick1", "nick2", ANYSTR],
)
@cases.mark_specifications("Modern")
def testWhowasNoParamModern(self):
"""
"If the `<nick>` argument is missing, they SHOULD send a single reply, using
either ERR_NONICKNAMEGIVEN or ERR_NEEDMOREPARAMS"
-- https://modern.ircdocs.horse/#whowas-message
"""
# But no one seems to follow this. Most implementations use ERR_NEEDMOREPARAMS
# instead of ERR_NONICKNAMEGIVEN; and I couldn't find any that returns
# RPL_ENDOFWHOWAS either way.
self.connectClient("nick1")
self.sendLine(1, "WHOWAS")
m = self.getMessage(1)
if m.command == ERR_NONICKNAMEGIVEN:
self.assertMessageMatch(
m,
command=ERR_NONICKNAMEGIVEN,
params=["nick1", ANYSTR],
)
else:
self.assertMessageMatch(
m,
command=ERR_NEEDMOREPARAMS,
params=["nick1", "WHOWAS", ANYSTR],
)
@cases.mark_specifications("RFC1459", "RFC2812", "Modern")
@cases.xfailIfSoftware(
["Charybdis"],
"fails because of a typo (solved in "
"https://github.com/solanum-ircd/solanum/commit/"
"08b7b6bd7e60a760ad47b58cbe8075b45d66166f)",
)
def testWhowasNoSuchNick(self):
"""
https://datatracker.ietf.org/doc/html/rfc1459#section-4.5.3
https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
-- https://modern.ircdocs.horse/#whowas-message
and:
"At the end of all reply batches, there must be RPL_ENDOFWHOWAS
(even if there was only one reply and it was an error)."
-- https://datatracker.ietf.org/doc/html/rfc1459#page-50
-- https://datatracker.ietf.org/doc/html/rfc2812#page-45
and:
"Servers MUST reply with either ERR_WASNOSUCHNICK or [...],
both followed with RPL_ENDOFWHOWAS"
-- https://modern.ircdocs.horse/#whowas-message
"""
self.connectClient("nick1")
self.sendLine(1, "WHOWAS nick2")
self.assertMessageMatch(
self.getMessage(1),
command=ERR_WASNOSUCHNICK,
params=["nick1", "nick2", ANYSTR],
)
self.assertMessageMatch(
self.getMessage(1),
command=RPL_ENDOFWHOWAS,
params=["nick1", "nick2", ANYSTR],
)
@cases.mark_specifications("RFC2812")
@cases.mark_isupport("TARGMAX")
@pytest.mark.parametrize("targets", ["nick2,nick3", "nick3,nick2"])
def testWhowasMultiTarget(self, targets):
"""
https://datatracker.ietf.org/doc/html/rfc2812#section-3.6.3
"""
if self.controller.software_name == "Bahamut":
pytest.xfail(
"Bahamut returns entries in query order instead of chronological order"
)
self.connectClient("nick1")
if self.targmax.get("WHOWAS", "1") == "1":
raise runner.OptionalExtensionNotSupported("Multi-target WHOWAS")
self.connectClient("nick2", ident="ident2")
self.sendLine(2, "QUIT :bye")
try:
self.getMessages(2)
except ConnectionClosed:
pass
self.connectClient("nick3", ident="ident3")
self.sendLine(3, "QUIT :bye")
try:
self.getMessages(3)
except ConnectionClosed:
pass
self.sendLine(1, f"WHOWAS {targets}")
messages = self.getMessages(1)
self.assertMessageMatch(
messages.pop(0),
command=RPL_WHOWASUSER,
params=[
"nick1",
"nick3",
StrRe("~?ident3"),
ANYSTR,
"*",
"Realname",
],
)
while messages[0].command in (RPL_WHOISACTUALLY, RPL_WHOISSERVER):
# don't care
messages.pop(0)
# 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", targets, ANYSTR],
fail_msg=f"Last message was not RPL_ENDOFWHOWAS ({RPL_ENDOFWHOWAS})",
)

Some files were not shown because too many files have changed in this diff Show More