This is a very simple snapshot based test runner which is used by the ghūl programming language compiler integration tests. It compares test expectations, in the form of snapshot text files, against the actual outputs of the compiler and test executables and flags any differences.
A test directory must contain at least two things:
- One or more
.ghulsource files – the sources to compile. - A
ghulflagsfile – flags passed directly to the compiler when building the test.
Any directory containing a ghulflags file is treated as a test. Subdirectories without this file are ignored by the queue logic.
Optional expectation and configuration files may also be present:
| File | Purpose |
|---|---|
fail.expected |
If present, the build is expected to fail. Its mere presence enables this behaviour; the file contents are ignored. |
err.expected |
Expected compiler error output. Actual errors are extracted from compiler.out, sorted, and diffed against this file. |
warn.expected |
Expected compiler warning output. Warnings undergo the same grep and sort process as errors. |
run.expected |
Expected stdout from running the compiled binary. |
il.expected |
Expected IL disassembly output (from the il.out file). |
ghulflags |
Mandatory file containing additional command line flags for the compiler. |
disabled* |
Any file beginning with disabled causes the test to be skipped. |
A basic “hello world” example can be found in the integration-tests folder of this repository.
- The runner invokes the compiler using the arguments from
ghulflagsand the test’s.ghulsources. Compiler stdout/stderr is written tocompiler.out. grepextracts error and warning lines fromcompiler.outintoerr.grepandwarn.greprespectively.- These files are sorted with
sort(withLC_COLLATEset toCfor stable output) intoerr.sortandwarn.sort. diffcompareserr.sorttoerr.expectedandwarn.sorttowarn.expected. Whitespace differences are ignored and carriage returns are stripped.- If compilation succeeded,
ghul-runtime.dllis symlinked into the test directory and the binary is executed viadotnet. Output is captured inrun.outand compared torun.expected. - If an
il.expectedfile exists,diffis run against the generatedil.outfile as well.
Any mismatches cause a failure report containing a unified diff of the actual versus expected output.
ghul-test [--use-dotnet-build] <test-folder> [...]
--use-dotnet-build– expects each test folder to be an MSBuild project. For ghūl projects the file should end with.ghulproj. The runner builds the project withdotnet buildinstead of invoking the compiler directly.<test-folder>– one or more directories containing tests. Each is recursively searched for subdirectories with aghulflagsfile if not using--use-dotnet-build.
Environment variables influence behaviour:
HOSTandTARGET– specify the CLI used to run the compiler and the compiled binary (defaultdotnet).CI– when set to1ortrue, enables CI mode. In this modeghul-runtime.dllis taken from the test runner’s own location.TEST_PROCESSES– number of worker processes to use. If unset, a value derived from CPU count is used.
The runner prints progress for each test and a final summary indicating total, enabled, passed and failed counts.
When the compiler is invoked directly (the default and CI modes), the produced executable expects to find ghul-runtime.dll beside it. The runner therefore creates a symbolic link in the test directory pointing to the runtime library. This link is not needed when --use-dotnet-build is used. After the test completes successfully, the link is deleted during cleanup.
This runner does not execute arbitrary MSBuild projects. It either drives the compiler directly on .ghul source files or, when --use-dotnet-build is supplied, assumes the folder already contains a valid MSBuild project (for ghūl projects this means a *.ghulproj file). Only a small set of standard .NET assemblies is referenced so complex projects are out of scope.
The runner relies on several standard Unix utilities being available in the environment: grep, sort, diff and ln. A .NET 8 SDK installation is required; mono is not supported.
This repository includes helper scripts under ./scripts:
create.sh– create a new test directory from the built-in template.capture.sh– update expectation files after running a test.
- Run
./scripts/create.shand provide the new test name. - Edit the generated
.ghulsources andghulflagsas required. - Execute
ghul-test <path-to-test>(expect it to fail initially). The runner produces.outfiles with the actual output. - Run
./scripts/capture.sh <path-to-test>to copy the.outfiles over the corresponding*.expectedfiles. - Re-run
ghul-testand verify the test now passes. - Commit the test directory along with the expectations.
Refer to the ghūl compiler integration tests for many real‑world examples of this structure.