-
-
Notifications
You must be signed in to change notification settings - Fork 1.6k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
initialize test suite for git fetchers
solves #9388 This utilizes nixos vm tests to allow: - writing tests for fetchTree and fetchGit involving actual networking. - writing small independent test cases by automating local and remote repository setup per test case. This adds: - a gitea module setting up a gitea server - a setup module that simplifies writing test cases by automating the repo setup. - a simple git http test case Other improvements: For all nixos tests, add capability of overriding the nix version to test against. This should make it easier to prevent regressions. If a new test is added it can simply be ran against any older nix version without having to backport the test. For example, for running the container tests against nix 2.12.0: `nix build "$(nix eval --raw .#hydraJobs.tests.containers --impure --apply 't: (t.forNix "2.12.0").drvPath')^*" -L`
- Loading branch information
Showing
4 changed files
with
258 additions
and
8 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,60 @@ | ||
{ lib, config, ... }: | ||
{ | ||
name = "fetch-git"; | ||
|
||
imports = [ | ||
./testsupport/gitea.nix | ||
]; | ||
|
||
/* | ||
Test cases | ||
The following is set up automatically for each test case: | ||
- a repo with the {name} is created on the gitea server | ||
- a repo with the {name} is created on the client | ||
- the client repo is configured to push to the server repo | ||
Python variables: | ||
- repo.path: the path to the directory of the client repo | ||
- repo.git: the git command with the client repo as the working directory | ||
- repo.remote: the url to the server repo | ||
*/ | ||
testCases = [ | ||
{ | ||
name = "simple-http"; | ||
description = "can fetch a git repo via http"; | ||
script = '' | ||
# add a file to the repo | ||
client.succeed(f""" | ||
echo chiang-mai > {repo.path}/thailand \ | ||
&& {repo.git} add thailand \ | ||
&& {repo.git} commit -m 'commit1' | ||
""") | ||
# memoize the revision | ||
rev1 = client.succeed(f""" | ||
{repo.git} rev-parse HEAD | ||
""").strip() | ||
# push to the server | ||
client.succeed(f""" | ||
{repo.git} push origin main | ||
""") | ||
# fetch the repo via nix | ||
fetched1 = client.succeed(f""" | ||
nix eval --impure --raw --expr "(builtins.fetchGit {repo.remote}).outPath" | ||
""") | ||
# check if the committed file is there | ||
client.succeed(f""" | ||
test -f {fetched1}/thailand | ||
""") | ||
# check if the revision is the same | ||
rev1_fetched = client.succeed(f""" | ||
nix eval --impure --raw --expr "(builtins.fetchGit {repo.remote}).rev" | ||
""").strip() | ||
assert rev1 == rev1_fetched | ||
''; | ||
} | ||
]; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,55 @@ | ||
{ lib, ... }: { | ||
imports = [ | ||
../testsupport/setup.nix | ||
]; | ||
nodes = { | ||
gitea = { pkgs, ... }: { | ||
services.gitea.enable = true; | ||
services.gitea.settings.service.DISABLE_REGISTRATION = true; | ||
services.gitea.settings.log.LEVEL = "Info"; | ||
services.gitea.settings.database.LOG_SQL = false; | ||
networking.firewall.allowedTCPPorts = [ 3000 ]; | ||
environment.systemPackages = [ pkgs.gitea ]; | ||
}; | ||
client = { pkgs, ... }: { | ||
environment.systemPackages = [ pkgs.git ]; | ||
}; | ||
}; | ||
defaults = { pkgs, ... }: { | ||
environment.systemPackages = [ pkgs.jq ]; | ||
}; | ||
|
||
setupScript = '' | ||
import shlex | ||
gitea.wait_for_unit("gitea.service") | ||
gitea_admin = "test" | ||
gitea_admin_password = "test123test" | ||
gitea.succeed(f""" | ||
gitea --version >&2 | ||
su -l gitea -c 'GITEA_WORK_DIR=/var/lib/gitea gitea admin user create \ | ||
--username {gitea_admin} --password {gitea_admin_password} --email test@client' | ||
""") | ||
client.wait_for_unit("multi-user.target") | ||
gitea.wait_for_open_port(3000) | ||
gitea_admin_token = gitea.succeed(f""" | ||
curl --fail -X POST http://{gitea_admin}:{gitea_admin_password}@gitea:3000/api/v1/users/test/tokens \ | ||
-H 'Accept: application/json' -H 'Content-Type: application/json' \ | ||
-d {shlex.quote( '{"name":"token", "scopes":["all"]}' )} \ | ||
| jq -r '.sha1' | ||
""").strip() | ||
client.succeed(f""" | ||
echo "http://{gitea_admin}:{gitea_admin_password}@gitea:3000" >~/.git-credentials-admin | ||
git config --global credential.helper 'store --file ~/.git-credentials-admin' | ||
git config --global user.email "test@client" | ||
git config --global user.name "Test User" | ||
git config --global gc.autodetach 0 | ||
git config --global gc.auto 0 | ||
""") | ||
''; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,121 @@ | ||
{ lib, config, ... }: | ||
let | ||
inherit (lib) | ||
concatStrings | ||
mapAttrsToList | ||
mkOption | ||
types | ||
; | ||
|
||
indent = lib.replaceStrings ["\n"] ["\n "]; | ||
|
||
execTestCase = testCase: '' | ||
### TEST ${testCase.name}: ${testCase.description} ### | ||
with run_test("${testCase.description}"): | ||
repo = Repo("${testCase.name}") | ||
${indent testCase.script} | ||
''; | ||
in | ||
{ | ||
|
||
options = { | ||
setupScript = mkOption { | ||
type = types.lines; | ||
description = '' | ||
Python code that runs before the main test. | ||
Variables defined by this code will be available in the test. | ||
''; | ||
default = ""; | ||
}; | ||
testCases = mkOption { | ||
type = types.listOf (types.submodule { | ||
options.name = mkOption { | ||
type = types.str; | ||
description = '' | ||
The name of the test case. | ||
A repo will automatically be created in a directory with that name. | ||
''; | ||
}; | ||
options.description = mkOption { | ||
type = types.str; | ||
description = '' | ||
A description of the test case. | ||
''; | ||
}; | ||
options.script = mkOption { | ||
type = types.lines; | ||
description = '' | ||
Python code that runs the test. | ||
Variables defined by `setupScript` will be available here. | ||
''; | ||
}; | ||
}); | ||
description = '' | ||
The test cases. See `testScript`. | ||
''; | ||
}; | ||
}; | ||
|
||
config = { | ||
nodes.client = { | ||
environment.variables = { | ||
_NIX_FORCE_HTTP = "1"; | ||
}; | ||
nix.settings.experimental-features = ["nix-command" "flakes"]; | ||
}; | ||
setupScript = '' | ||
from contextlib import contextmanager | ||
class Repo: | ||
""" | ||
A class to create a git repository on the gitea server and locally. | ||
""" | ||
def __init__(self, name): | ||
self.name = name | ||
self.path = "/tmp/repos/" + name | ||
self.remote = "http://gitea:3000/test/" + name | ||
self.git = f"git -C {self.path}" | ||
self.create() | ||
def create(self): | ||
gitea.succeed(f""" | ||
curl --fail -X POST http://{gitea_admin}:{gitea_admin_password}@gitea:3000/api/v1/user/repos \ | ||
-H 'Accept: application/json' -H 'Content-Type: application/json' \ | ||
-d {shlex.quote( f'{{"name":"{self.name}", "default_branch": "main"}}' )} | ||
""") | ||
client.succeed(f""" | ||
mkdir -p {self.path} \ | ||
&& git init -b main {self.path} \ | ||
&& {self.git} remote add origin {self.remote} | ||
""") | ||
@contextmanager | ||
def run_test(description): | ||
""" | ||
A context manager to run a test case. | ||
Prints the description when tests starts or fails. | ||
""" | ||
print(f"\033[94mtesting: {description}\033[0m") | ||
try: | ||
yield | ||
except Exception: | ||
print(f"\033[91mfailed: {description}\033[0m") | ||
raise | ||
''; | ||
testScript = '' | ||
start_all(); | ||
${config.setupScript} | ||
### SETUP COMPLETE ### | ||
${lib.concatStringsSep "\n" (map execTestCase config.testCases)} | ||
''; | ||
}; | ||
|
||
} |