Merge pull request #298967 from vbgl/ocaml-5.2.0
[NixPkgs.git] / doc / build-helpers / testers.chapter.md
blobb734cbbbd4e29bdb13cc3e7ac6611de1763e4a50
1 # Testers {#chap-testers}
3 This chapter describes several testing builders which are available in the `testers` namespace.
5 ## `hasPkgConfigModules` {#tester-hasPkgConfigModules}
7 <!-- Old anchor name so links still work -->
8 []{#tester-hasPkgConfigModule}
9 Checks whether a package exposes a given list of `pkg-config` modules.
10 If the `moduleNames` argument is omitted, `hasPkgConfigModules` will use `meta.pkgConfigModules`.
12 :::{.example #ex-haspkgconfigmodules-defaultvalues}
14 # Check that `pkg-config` modules are exposed using default values
16 ```nix
18   passthru.tests.pkg-config = testers.hasPkgConfigModules {
19     package = finalAttrs.finalPackage;
20   };
22   meta.pkgConfigModules = [ "libfoo" ];
24 ```
26 :::
28 :::{.example #ex-haspkgconfigmodules-explicitmodules}
30 # Check that `pkg-config` modules are exposed using explicit module names
32 ```nix
34   passthru.tests.pkg-config = testers.hasPkgConfigModules {
35     package = finalAttrs.finalPackage;
36     moduleNames = [ "libfoo" ];
37   };
39 ```
41 :::
43 ## `testVersion` {#tester-testVersion}
45 Checks that the output from running a command contains the specified version string in it as a whole word.
47 Although simplistic, this test assures that the main program can run.
48 While there's no substitute for a real test case, it does catch dynamic linking errors and such.
49 It also provides some protection against accidentally building the wrong version, for example when using an "old" hash in a fixed-output derivation.
51 By default, the command to be run will be inferred from the given `package` attribute:
52 it will check `meta.mainProgram` first, and fall back to `pname` or `name`.
53 The default argument to the command is `--version`, and the version to be checked will be inferred from the given `package` attribute as well.
55 :::{.example #ex-testversion-hello}
57 # Check a program version using all the default values
59 This example will run the command `hello --version`, and then check that the version of the `hello` package is in the output of the command.
61 ```nix
63   passthru.tests.version = testers.testVersion { package = hello; };
65 ```
67 :::
69 :::{.example #ex-testversion-different-commandversion}
71 # Check the program version using a specified command and expected version string
73 This example will run the command `leetcode -V`, and then check that `leetcode 0.4.2` is in the output of the command as a whole word (separated by whitespaces).
74 This means that an output like "leetcode 0.4.21" would fail the tests, and an output like "You're running leetcode 0.4.2" would pass the tests.
76 A common usage of the `version` attribute is to specify `version = "v${version}"`.
78 ```nix
80   version = "0.4.2";
82   passthru.tests.version = testers.testVersion {
83     package = leetcode-cli;
84     command = "leetcode -V";
85     version = "leetcode ${version}";
86   };
88 ```
90 :::
92 ## `testBuildFailure` {#tester-testBuildFailure}
94 Make sure that a build does not succeed. This is useful for testing testers.
96 This returns a derivation with an override on the builder, with the following effects:
98  - Fail the build when the original builder succeeds
99  - Move `$out` to `$out/result`, if it exists (assuming `out` is the default output)
100  - Save the build log to `$out/testBuildFailure.log` (same)
102 While `testBuildFailure` is designed to keep changes to the original builder's environment to a minimum, some small changes are inevitable:
104  - The file `$TMPDIR/testBuildFailure.log` is present. It should not be deleted.
105  - `stdout` and `stderr` are a pipe instead of a tty. This could be improved.
106  - One or two extra processes are present in the sandbox during the original builder's execution.
107  - The derivation and output hashes are different, but not unusual.
108  - The derivation includes a dependency on `buildPackages.bash` and `expect-failure.sh`, which is built to include a transitive dependency on `buildPackages.coreutils` and possibly more.
109    These are not added to `PATH` or any other environment variable, so they should be hard to observe.
111 :::{.example #ex-testBuildFailure-showingenvironmentchanges}
113 # Check that a build fails, and verify the changes made during build
115 ```nix
116 runCommand "example" {
117   failed = testers.testBuildFailure (runCommand "fail" {} ''
118     echo ok-ish >$out
119     echo failing though
120     exit 3
121   '');
122 } ''
123   grep -F 'ok-ish' $failed/result
124   grep -F 'failing though' $failed/testBuildFailure.log
125   [[ 3 = $(cat $failed/testBuildFailure.exit) ]]
126   touch $out
132 ## `testEqualContents` {#tester-equalContents}
134 Check that two paths have the same contents.
136 :::{.example #ex-testEqualContents-toyexample}
138 # Check that two paths have the same contents
140 ```nix
141 testers.testEqualContents {
142   assertion = "sed -e performs replacement";
143   expected = writeText "expected" ''
144     foo baz baz
145   '';
146   actual = runCommand "actual" {
147     # not really necessary for a package that's in stdenv
148     nativeBuildInputs = [ gnused ];
149     base = writeText "base" ''
150       foo bar baz
151     '';
152   } ''
153     sed -e 's/bar/baz/g' $base >$out
154   '';
160 ## `testEqualDerivation` {#tester-testEqualDerivation}
162 Checks that two packages produce the exact same build instructions.
164 This can be used to make sure that a certain difference of configuration, such as the presence of an overlay does not cause a cache miss.
166 When the derivations are equal, the return value is an empty file.
167 Otherwise, the build log explains the difference via `nix-diff`.
169 :::{.example #ex-testEqualDerivation-hello}
171 # Check that two packages produce the same derivation
173 ```nix
174 testers.testEqualDerivation
175   "The hello package must stay the same when enabling checks."
176   hello
177   (hello.overrideAttrs(o: { doCheck = true; }))
182 ## `invalidateFetcherByDrvHash` {#tester-invalidateFetcherByDrvHash}
184 Use the derivation hash to invalidate the output via name, for testing.
186 Type: `(a@{ name, ... } -> Derivation) -> a -> Derivation`
188 Normally, fixed output derivations can and should be cached by their output hash only, but for testing we want to re-fetch everytime the fetcher changes.
190 Changes to the fetcher become apparent in the drvPath, which is a hash of how to fetch, rather than a fixed store path.
191 By inserting this hash into the name, we can make sure to re-run the fetcher every time the fetcher changes.
193 This relies on the assumption that Nix isn't clever enough to reuse its database of local store contents to optimize fetching.
195 You might notice that the "salted" name derives from the normal invocation, not the final derivation.
196 `invalidateFetcherByDrvHash` has to invoke the fetcher function twice:
197 once to get a derivation hash, and again to produce the final fixed output derivation.
199 :::{.example #ex-invalidateFetcherByDrvHash-nix}
201 # Prevent nix from reusing the output of a fetcher
203 ```nix
205   tests.fetchgit = testers.invalidateFetcherByDrvHash fetchgit {
206     name = "nix-source";
207     url = "https://github.com/NixOS/nix";
208     rev = "9d9dbe6ed05854e03811c361a3380e09183f4f4a";
209     hash = "sha256-7DszvbCNTjpzGRmpIVAWXk20P0/XTrWZ79KSOGLrUWY=";
210   };
216 ## `runNixOSTest` {#tester-runNixOSTest}
218 A helper function that behaves exactly like the NixOS `runTest`, except it also assigns this Nixpkgs package set as the `pkgs` of the test and makes the `nixpkgs.*` options read-only.
220 If your test is part of the Nixpkgs repository, or if you need a more general entrypoint, see ["Calling a test" in the NixOS manual](https://nixos.org/manual/nixos/stable/index.html#sec-calling-nixos-tests).
222 :::{.example #ex-runNixOSTest-hello}
224 # Run a NixOS test using `runNixOSTest`
226 ```nix
227 pkgs.testers.runNixOSTest ({ lib, ... }: {
228   name = "hello";
229   nodes.machine = { pkgs, ... }: {
230     environment.systemPackages = [ pkgs.hello ];
231   };
232   testScript = ''
233     machine.succeed("hello")
234   '';
240 ## `nixosTest` {#tester-nixosTest}
242 Run a NixOS VM network test using this evaluation of Nixpkgs.
244 NOTE: This function is primarily for external use. NixOS itself uses `make-test-python.nix` directly. Packages defined in Nixpkgs [reuse NixOS tests via `nixosTests`, plural](#ssec-nixos-tests-linking).
246 It is mostly equivalent to the function `import ./make-test-python.nix` from the [NixOS manual](https://nixos.org/nixos/manual/index.html#sec-nixos-tests), except that the current application of Nixpkgs (`pkgs`) will be used, instead of letting NixOS invoke Nixpkgs anew.
248 If a test machine needs to set NixOS options under `nixpkgs`, it must set only the `nixpkgs.pkgs` option.
250 ### Parameter {#tester-nixosTest-parameter}
252 A [NixOS VM test network](https://nixos.org/nixos/manual/index.html#sec-nixos-tests), or path to it. Example:
254 ```nix
256   name = "my-test";
257   nodes = {
258     machine1 = { lib, pkgs, nodes, ... }: {
259       environment.systemPackages = [ pkgs.hello ];
260       services.foo.enable = true;
261     };
262     # machine2 = ...;
263   };
264   testScript = ''
265     start_all()
266     machine1.wait_for_unit("foo.service")
267     machine1.succeed("hello | foo-send")
268   '';
272 ### Result {#tester-nixosTest-result}
274 A derivation that runs the VM test.
276 Notable attributes:
278  * `nodes`: the evaluated NixOS configurations. Useful for debugging and exploring the configuration.
280  * `driverInteractive`: a script that launches an interactive Python session in the context of the `testScript`.