ghūl compiler integration test runner
June 5, 2026 · View on GitHub
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.
Test Folder Structure
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.
Expectation Comparison Workflow
- 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.
Command Line Usage
ghul-test [--use-dotnet-build] [--runtime-dll <path>] <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.--runtime-dll <path>– use the suppliedghul-runtime.dllfor compiled test binaries instead of the version that ships withghul-test. The path must point to an existing file. Takes precedence over theGHUL_RUNTIME_DLLenvironment variable. Has no effect under--use-dotnet-build, which resolves the runtime via the test project's ownPackageReference.<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 unless overridden by--runtime-dll/GHUL_RUNTIME_DLL.GHUL_RUNTIME_DLL– path to aghul-runtime.dllto use for compiled test binaries, overriding the version that ships withghul-test. Equivalent to passing--runtime-dll; the CLI flag wins if both are set.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.
Runtime Library Handling
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.
By default the runtime DLL is sourced from the published compiler's directory (LOCAL mode) or the test runner's own install directory (CI mode). When the integration tests need to run against a runtime version other than the one ghul-test itself was packaged with — for example, when CI builds a compiler that depends on a newer ghul.runtime than the pinned ghul.test ships with — pass --runtime-dll <path> or set GHUL_RUNTIME_DLL to override the discovered location with an explicit DLL path.
MSBuild Projects
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.
Dependencies
The runner relies on several standard Unix utilities being available in the environment: grep, sort, diff and ln. A .NET 10 SDK installation is required; mono is not supported.
Writing New Tests
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.