Add tests documentation to docs page
All checks were successful
CI / build (push) Successful in 2m43s

This commit is contained in:
Fabian Hauser 2025-03-24 15:35:23 +02:00
parent c9844276b8
commit c2783867b6
8 changed files with 152 additions and 107 deletions

View file

@ -1,6 +1,7 @@
# Summary
- [Repository README](README.md)
- [Testing](checks/README.md)
- [Deployment](deploy/README.md)
---

View file

@ -4,7 +4,7 @@
pkgs,
deployPkgs,
...
}@inputs:
}:
{
${system} = {
@ -17,7 +17,7 @@
'';
nixos-modules = pkgs.callPackage ./nixos-modules {
inherit (self.lib) loadSubmodulesFrom;
inherit (self.lib) getSubDirs isFolderWithFile;
};
#TODO(#29): Integration/System tests

View file

@ -1,9 +1,60 @@
{
linkFarmFromDrvs,
callPackage,
loadSubmodulesFrom,
isFolderWithFile,
getSubDirs,
lib,
testers,
}:
let
tests = map (test: callPackage test { }) (loadSubmodulesFrom ./.);
inherit (lib)
filter
path
mkDefault
readFile
attrNames
concatStringsSep
pipe
;
modulesBaseDir = ../../nixos-modules;
mkTest =
name:
let
getFilePath = file: path.append modulesBaseDir "./${name}/${file}";
in
linkFarmFromDrvs "nixos-modules" tests
testers.runNixOSTest {
inherit name;
imports = [
(import (getFilePath "test.nix") {
inherit name;
inherit lib;
})
];
defaults.imports = [ (getFilePath "default.nix") ];
# Calls a `test(...)` python function in the test's python file with the list of nodes and helper functions.
# Helper symbols may be added as function args when needed and can be found in:
# https://github.com/NixOS/nixpkgs/blob/master/nixos/lib/test-driver/src/test_driver/driver.py#L121
testScript = mkDefault (
{ nodes, ... }:
let
script = readFile (getFilePath "test.py");
nodeArgs = pipe nodes [
attrNames
(map (val: "${val}=${val}"))
(concatStringsSep ", ")
];
in
''
${script}
test(${nodeArgs}, subtest=subtest)
''
);
};
in
pipe modulesBaseDir [
getSubDirs
(filter (isFolderWithFile "test.nix" modulesBaseDir))
(map mkTest)
(linkFarmFromDrvs "nixos-modules")
]

View file

@ -1,45 +0,0 @@
{
testers,
curl,
lib,
gnugrep,
...
}:
testers.runNixOSTest {
name = "static-page";
nodes.webserver =
{ ... }:
{
# Service under test
imports = [ ../../../nixos-modules/static-page ];
qois.static-page = {
enable = true;
pages = lib.mkForce {
"localhost" = {
domainAliases = [ "example.com" ];
};
};
};
# Test environment
environment.systemPackages = [
curl
gnugrep
];
# Disable TLS services
services.nginx.virtualHosts =
let
tlsOff = {
forceSSL = lib.mkForce false;
enableACME = lib.mkForce false;
};
in
{
"localhost" = tlsOff;
"example.com" = tlsOff;
};
};
testScript = lib.readFile ./test.py;
}

View file

@ -1,46 +0,0 @@
webserver.wait_for_unit("nginx")
webserver.wait_for_open_port(80)
# Preparations
webserverRoot = "/var/lib/nginx-localhost/root"
indexContent = "It works!"
webserver.succeed(f"mkdir {webserverRoot}")
webserver.succeed(f"echo '{indexContent}' > {webserverRoot}/index.html")
webserver.succeed(f"chown -R nginx-localhost\: {webserverRoot}")
# Helpers
def expect_http_code(node, code, url):
http_code = node.succeed(
f"curl -s --no-location -o /dev/null -w '%{{http_code}}' '{url}'")
assert http_code == code, \
f"expected {code} but got following response:\n{http_code}"
def expect_http_location(node, location, url):
redirect_url = node.succeed(
f"curl -s --no-location -o /dev/null -w '%{{redirect_url}}' '{url}'")
assert redirect_url == location, \
f"expected redirect to {location} but got:\n{redirect_url}"
def expect_http_content(node, expectedContent, url):
content = node.succeed(f"curl --no-location --silent '{url}'")
assert content.strip() == expectedContent.strip(), \
f"expected:\n{expectedContent}\n at {
url} but got following content:\n'{content}'"
# Tests
with subtest("website is successfully served on localhost"):
expect_http_code(webserver, "200", "http://localhost/index.html")
expect_http_content(webserver, indexContent, "http://localhost/index.html")
with subtest("example.com is a hosts alias and redirects to localhost"):
webserver.succeed("grep example.com /etc/hosts")
url = "http://example.com/index.html"
expect_http_code(webserver, "301", url)
expect_http_location(
webserver, "http://localhost/index.html", url)

View file

@ -1,18 +1,26 @@
{ pkgs, ... }:
let
lib = pkgs.lib;
foldersWithNix =
path:
let
folders = lib.attrNames (lib.filterAttrs (n: t: t == "directory") (builtins.readDir path));
isFolderWithDefaultNix = folder: lib.pathExists (lib.path.append path "./${folder}/default.nix");
in
lib.filter isFolderWithDefaultNix folders;
inherit (pkgs.lib)
attrNames
filterAttrs
filter
pathExists
path
;
# Get a list of all subdirectories of a directory.
getSubDirs = base: attrNames (filterAttrs (n: t: t == "directory") (builtins.readDir base));
# Check if a folder with a base path and folder name contains a file with a specific name
isFolderWithFile =
fileName: basePath: folderName:
(pathExists (path.append basePath "./${folderName}/${fileName}"));
# Get a list of subfolders that contain a default.nix file.
foldersWithNix = base: filter (isFolderWithFile "default.nix" base) (getSubDirs base);
in
{
inherit foldersWithNix;
inherit getSubDirs isFolderWithFile foldersWithNix;
# Get a list of default.nix files that are nix submodules of the current folder.
loadSubmodulesFrom =
path: map (folder: lib.path.append path "./${folder}/default.nix") (foldersWithNix path);
basePath: map (folder: path.append basePath "./${folder}/default.nix") (foldersWithNix basePath);
}

View file

@ -0,0 +1,30 @@
{
...
}:
{
nodes.webserver =
{ pkgs, lib, ... }:
let
inherit (pkgs) curl gnugrep;
inherit (lib) mkForce genAttrs const;
in
{
# Setup simple localhost page with an example.com redirect
qois.static-page = {
enable = true;
pages."localhost".domainAliases = [ "example.com" ];
};
# Disable TLS services
services.nginx.virtualHosts = genAttrs [ "localhost" "example.com" ] (const {
forceSSL = mkForce false;
enableACME = mkForce false;
});
# Test environment
environment.systemPackages = [
curl
gnugrep
];
};
}

View file

@ -0,0 +1,46 @@
def test(subtest, webserver):
webserver.wait_for_unit("nginx")
webserver.wait_for_open_port(80)
# Preparations
webserverRoot = "/var/lib/nginx-localhost/root"
indexContent = "It works!"
webserver.succeed(f"mkdir {webserverRoot}")
webserver.succeed(f"echo '{indexContent}' > {webserverRoot}/index.html")
webserver.succeed(f"chown -R nginx-localhost\: {webserverRoot}")
# Helpers
def curl_variable_test(node, variable, expected, url):
value = node.succeed(
f"curl -s --no-location -o /dev/null -w '%{{{variable}}}' '{url}'")
assert value == expected, \
f"expected {variable} to be '{expected}' but got '{value}'"
def expect_http_code(node, code, url):
curl_variable_test(node, "http_code", code, url)
def expect_http_location(node, location, url):
curl_variable_test(node, "redirect_url", location, url)
def expect_http_content(node, expectedContent, url):
content = node.succeed(f"curl --no-location --silent '{url}'")
assert content.strip() == expectedContent.strip(), f'''
expected content:
{expectedContent}
at {url} but got following content:
{content}
'''
# Tests
with subtest("website is successfully served on localhost"):
expect_http_code(webserver, "200", "http://localhost/index.html")
expect_http_content(webserver, indexContent,
"http://localhost/index.html")
with subtest("example.com is a hosts alias and redirect"):
webserver.succeed("grep example.com /etc/hosts")
url = "http://example.com/index.html"
expect_http_code(webserver, "301", url)
expect_http_location(
webserver, "http://localhost/index.html", url)