mirror of
https://github.com/NixOS/nix
synced 2025-06-27 00:11:17 +02:00
251 lines
7.6 KiB
Markdown
251 lines
7.6 KiB
Markdown
# Hacking
|
|
|
|
This section provides some notes on how to hack on Nix. To get the
|
|
latest version of Nix from GitHub:
|
|
|
|
```console
|
|
$ git clone https://github.com/NixOS/nix.git
|
|
$ cd nix
|
|
```
|
|
|
|
To build Nix for the current operating system/architecture use
|
|
|
|
```console
|
|
$ nix-build
|
|
```
|
|
|
|
or if you have a flake-enabled nix:
|
|
|
|
```console
|
|
$ nix build
|
|
```
|
|
|
|
This will build `defaultPackage` attribute defined in the `flake.nix`
|
|
file. To build for other platforms add one of the following suffixes to
|
|
it: aarch64-linux, i686-linux, x86\_64-darwin, x86\_64-linux. i.e.
|
|
|
|
```console
|
|
$ nix-build -A defaultPackage.x86_64-linux
|
|
```
|
|
|
|
To build all dependencies and start a shell in which all environment
|
|
variables are set up so that those dependencies can be found:
|
|
|
|
```console
|
|
$ nix-shell
|
|
```
|
|
|
|
or if you have a flake-enabled nix:
|
|
|
|
```console
|
|
$ nix develop
|
|
```
|
|
|
|
To get a shell with a different compilation environment (e.g. stdenv,
|
|
gccStdenv, clangStdenv, clang11Stdenv, ccacheStdenv):
|
|
|
|
```console
|
|
$ nix-shell -A devShells.x86_64-linux.clang11StdenvPackages
|
|
```
|
|
|
|
or if you have a flake-enabled nix:
|
|
|
|
```console
|
|
$ nix develop .#clang11StdenvPackages
|
|
```
|
|
|
|
Note: you can use `ccacheStdenv` to drastically improve rebuild
|
|
time. By default, ccache keeps artifacts in `~/.cache/ccache/`.
|
|
|
|
To build Nix itself in this shell:
|
|
|
|
```console
|
|
[nix-shell]$ ./bootstrap.sh
|
|
[nix-shell]$ ./configure $configureFlags --prefix=$(pwd)/outputs/out
|
|
[nix-shell]$ make -j $NIX_BUILD_CORES
|
|
```
|
|
|
|
To install it in `$(pwd)/outputs` and test it:
|
|
|
|
```console
|
|
[nix-shell]$ make install
|
|
[nix-shell]$ make installcheck -j $NIX_BUILD_CORES
|
|
[nix-shell]$ ./outputs/out/bin/nix --version
|
|
nix (Nix) 3.0
|
|
```
|
|
|
|
If you have a flakes-enabled Nix you can replace:
|
|
|
|
```console
|
|
$ nix-shell
|
|
```
|
|
|
|
by:
|
|
|
|
```console
|
|
$ nix develop
|
|
```
|
|
|
|
## Running tests
|
|
|
|
### Unit-tests
|
|
|
|
The unit-tests for each Nix library (`libexpr`, `libstore`, etc..) are defined
|
|
under `src/{library_name}/tests` using the
|
|
[googletest](https://google.github.io/googletest/) framework.
|
|
|
|
You can run the whole testsuite with `make check`, or the tests for a specific component with `make libfoo-tests_RUN`. Finer-grained filtering is also possible using the [--gtest_filter](https://google.github.io/googletest/advanced.html#running-a-subset-of-the-tests) command-line option.
|
|
|
|
### Functional tests
|
|
|
|
The functional tests reside under the `tests` directory and are listed in `tests/local.mk`.
|
|
Each test is a bash script.
|
|
|
|
The whole test suite can be run with:
|
|
|
|
```shell-session
|
|
$ make install && make installcheck
|
|
ran test tests/foo.sh... [PASS]
|
|
ran test tests/bar.sh... [PASS]
|
|
...
|
|
```
|
|
|
|
Individual tests can be run with `make`:
|
|
|
|
```shell-session
|
|
$ make tests/${testName}.sh.test
|
|
ran test tests/${testName}.sh... [PASS]
|
|
```
|
|
|
|
or without `make`:
|
|
|
|
```shell-session
|
|
$ ./mk/run-test.sh tests/${testName}.sh
|
|
ran test tests/${testName}.sh... [PASS]
|
|
```
|
|
|
|
To see the complete output, one can also run:
|
|
|
|
```shell-session
|
|
$ ./mk/debug-test.sh tests/${testName}.sh
|
|
+ foo
|
|
output from foo
|
|
+ bar
|
|
output from bar
|
|
...
|
|
```
|
|
|
|
The test script will then be traced with `set -x` and the output displayed as it happens, regardless of whether the test succeeds or fails.
|
|
|
|
#### Debugging failing functional tests
|
|
|
|
When a functional test fails, it usually does so somewhere in the middle of the script.
|
|
|
|
To figure out what's wrong, it is convenient to run the test regularly up to the failing `nix` command, and then run that command with a debugger like GDB.
|
|
|
|
For example, if the script looks like:
|
|
|
|
```bash
|
|
foo
|
|
nix blah blub
|
|
bar
|
|
```
|
|
edit it like so:
|
|
|
|
```diff
|
|
foo
|
|
-nix blah blub
|
|
+gdb --args nix blah blub
|
|
bar
|
|
```
|
|
|
|
Then, running the test with `./mk/debug-test.sh` will drop you into GDB once the script reaches that point:
|
|
|
|
```shell-session
|
|
$ ./mk/debug-test.sh tests/${testName}.sh
|
|
...
|
|
+ gdb blash blub
|
|
GNU gdb (GDB) 12.1
|
|
...
|
|
(gdb)
|
|
```
|
|
|
|
One can debug the Nix invocation in all the usual ways.
|
|
For example, enter `run` to start the Nix invocation.
|
|
|
|
### Integration tests
|
|
|
|
The integration tests are defined in the Nix flake under the `hydraJobs.tests` attribute.
|
|
These tests include everything that needs to interact with external services or run Nix in a non-trivial distributed setup.
|
|
Because these tests are expensive and require more than what the standard github-actions setup provides, they only run on the master branch (on <https://hydra.nixos.org/jobset/nix/master>).
|
|
|
|
You can run them manually with `nix build .#hydraJobs.tests.{testName}` or `nix-build -A hydraJobs.tests.{testName}`
|
|
|
|
### Installer tests
|
|
|
|
After a one-time setup, the Nix repository's GitHub Actions continuous integration (CI) workflow can test the installer each time you push to a branch.
|
|
|
|
Creating a Cachix cache for your installer tests and adding its authorization token to GitHub enables [two installer-specific jobs in the CI workflow](https://github.com/NixOS/nix/blob/88a45d6149c0e304f6eb2efcc2d7a4d0d569f8af/.github/workflows/ci.yml#L50-L91):
|
|
|
|
- The `installer` job generates installers for the platforms below and uploads them to your Cachix cache:
|
|
- `x86_64-linux`
|
|
- `armv6l-linux`
|
|
- `armv7l-linux`
|
|
- `x86_64-darwin`
|
|
|
|
- The `installer_test` job (which runs on `ubuntu-latest` and `macos-latest`) will try to install Nix with the cached installer and run a trivial Nix command.
|
|
|
|
#### One-time setup
|
|
|
|
1. Have a GitHub account with a fork of the [Nix repository](https://github.com/NixOS/nix).
|
|
2. At cachix.org:
|
|
- Create or log in to an account.
|
|
- Create a Cachix cache using the format `<github-username>-nix-install-tests`.
|
|
- Navigate to the new cache > Settings > Auth Tokens.
|
|
- Generate a new Cachix auth token and copy the generated value.
|
|
3. At github.com:
|
|
- Navigate to your Nix fork > Settings > Secrets > Actions > New repository secret.
|
|
- Name the secret `CACHIX_AUTH_TOKEN`.
|
|
- Paste the copied value of the Cachix cache auth token.
|
|
|
|
#### Using the CI-generated installer for manual testing
|
|
|
|
After the CI run completes, you can check the output to extract the installer URL:
|
|
1. Click into the detailed view of the CI run.
|
|
2. Click into any `installer_test` run (the URL you're here to extract will be the same in all of them).
|
|
3. Click into the `Run cachix/install-nix-action@v...` step and click the detail triangle next to the first log line (it will also be `Run cachix/install-nix-action@v...`)
|
|
4. Copy the value of `install_url`
|
|
5. To generate an install command, plug this `install_url` and your GitHub username into this template:
|
|
|
|
```console
|
|
sh <(curl -L <install_url>) --tarball-url-prefix https://<github-username>-nix-install-tests.cachix.org/serve
|
|
```
|
|
|
|
<!-- #### Manually generating test installers
|
|
|
|
There's obviously a manual way to do this, and it's still the only way for
|
|
platforms that lack GA runners.
|
|
|
|
I did do this back in Fall 2020 (before the GA approach encouraged here). I'll
|
|
sketch what I recall in case it encourages someone to fill in detail, but: I
|
|
didn't know what I was doing at the time and had to fumble/ask around a lot--
|
|
so I don't want to uphold any of it as "right". It may have been dumb or
|
|
the _hard_ way from the getgo. Fundamentals may have changed since.
|
|
|
|
Here's the build command I used to do this on and for x86_64-darwin:
|
|
nix build --out-link /tmp/foo ".#checks.x86_64-darwin.binaryTarball"
|
|
|
|
I used the stable out-link to make it easier to script the next steps:
|
|
link=$(readlink /tmp/foo)
|
|
cp $link/*-darwin.tar.xz ~/somewheres
|
|
|
|
I've lost the last steps and am just going from memory:
|
|
|
|
From here, I think I had to extract and modify the `install` script to point
|
|
it at this tarball (which I scped to my own site, but it might make more sense
|
|
to just share them locally). I extracted this script once and then just
|
|
search/replaced in it for each new build.
|
|
|
|
The installer now supports a `--tarball-url-prefix` flag which _may_ have
|
|
solved this need?
|
|
-->
|