mirror of
https://github.com/NixOS/nix.dev.git
synced 2024-10-18 00:06:26 -04:00
File sets tutorial (#802)
* source file selection tutorial Co-authored-by: Valentin Gagarin <valentin.gagarin@tweag.io> Co-authored-by: Robert Hensing <roberth@users.noreply.github.com>
This commit is contained in:
parent
7941b348a0
commit
614b6fe50f
|
@ -59,7 +59,7 @@ myst_heading_anchors = 3
|
|||
|
||||
myst_number_code_blocks = [ "nix", "python" ]
|
||||
|
||||
copybutton_prompt_text = r"\$ "
|
||||
copybutton_prompt_text = r"\$ |nix-repl> "
|
||||
copybutton_prompt_is_regexp = True
|
||||
|
||||
# Add any paths that contain templates here, relative to this directory.
|
||||
|
|
620
source/tutorials/file-sets.md
Normal file
620
source/tutorials/file-sets.md
Normal file
|
@ -0,0 +1,620 @@
|
|||
(file-sets)=
|
||||
# Working with local files
|
||||
<!-- TODO: Switch all mentions of unstable to stable once 23.11 is out
|
||||
Also remove mention of using unstable
|
||||
Also use https://nixos.org/manual/nixpkgs/unstable/#sec-functions-library-fileset
|
||||
instead of https://nixos.org/manual/nixpkgs/unstable/#sec-fileset
|
||||
-->
|
||||
|
||||
To build a local project in a Nix derivation, source files must be accessible to its [`builder` executable](https://nixos.org/manual/nix/stable/language/derivations#attr-builder).
|
||||
Since by default, the `builder` runs in an [isolated environment](https://nixos.org/manual/nix/stable/command-ref/conf-file.html#conf-sandbox) that only allows reading from the Nix store, the Nix language has built-in features to copy local files to the store and expose the resulting store paths.
|
||||
|
||||
Using these features directly can be tricky however:
|
||||
|
||||
- Coercion of paths to strings, such as the wide-spread pattern of `src = ./.`,
|
||||
makes the derivation dependent on the name of the current directory.
|
||||
It always adds the entire directory, including unneeded files that cause unnecessary new builds when they change.
|
||||
|
||||
- The [`builtins.path`](https://nixos.org/manual/nix/stable/language/builtins.html#builtins-path) function
|
||||
(and equivalently [`lib.sources.cleanSourceWith`](https://nixos.org/manual/nixpkgs/stable/#function-library-lib.sources.cleanSourceWith))
|
||||
can address these problems.
|
||||
However, it's often hard to express the desired path selection using the `filter` function interface.
|
||||
|
||||
In this tutorial you'll learn how to use the Nixpkgs [`lib.fileset` library](https://nixos.org/manual/nixpkgs/unstable/#sec-fileset) to work with local files in derivations.
|
||||
It abstracts over built-in functionality and offers a safer and more convenient interface.
|
||||
|
||||
## File sets
|
||||
|
||||
A _file set_ is a data type representing a collection of local files.
|
||||
File sets can be created, composed, and manipulated with the various functions of the library.
|
||||
|
||||
You can explore and learn about the library with [`nix repl`](https://nixos.org/manual/nix/stable/command-ref/new-cli/nix3-repl):
|
||||
|
||||
```shell-session
|
||||
$ nix repl -f https://github.com/NixOS/nixpkgs/tarball/master
|
||||
...
|
||||
nix-repl> fs = lib.fileset
|
||||
```
|
||||
|
||||
:::{note}
|
||||
We're using the `master` branch here, because the covered library features are part of NixOS 23.11 and up, which has not been released at the time of writing.
|
||||
:::
|
||||
|
||||
The [`trace`](https://nixos.org/manual/nixpkgs/unstable/#function-library-lib.fileset.trace) function pretty-prints the files included in a given file set:
|
||||
|
||||
```shell-session
|
||||
nix-repl> fs.trace ./. null
|
||||
trace: /home/user (all files in directory)
|
||||
null
|
||||
```
|
||||
|
||||
All functions that expect a file set for an argument can also accept a [path](https://nixos.org/manual/nix/stable/language/values#type-path).
|
||||
Such path arguments are then [implicitly turned into sets](https://nixos.org/manual/nixpkgs/unstable/#sec-fileset-path-coercion) that contain _all_ files under the given path.
|
||||
In the previous trace this is indicated by `(all files in directory)`.
|
||||
|
||||
:::{tip}
|
||||
The `trace` function pretty-prints its first argument and returns its second argument.
|
||||
But since you often just need the pretty-printing in `nix repl`, you can omit the second argument:
|
||||
|
||||
```shell-session
|
||||
nix-repl> fs.trace ./.
|
||||
trace: /home/user (all files in directory)
|
||||
«lambda @ /nix/store/1czr278x24s3bl6qdnifpvm5z03wfi2p-nixpkgs-src/lib/fileset/default.nix:555:8»
|
||||
```
|
||||
:::
|
||||
|
||||
Even though file sets conceptually contain local files, these files are *never* added to the Nix store unless explicitly requested.
|
||||
Therefore you don't have to worry as much about accidentally copying secrets into the world-readable store.
|
||||
|
||||
In this example, although we pretty-printed the home directory, no files were copied.
|
||||
This is in contrast to coercion of paths to strings such as in `"${./.}"`,
|
||||
which copies the whole directory to the Nix store on evaluation!
|
||||
|
||||
:::{warning}
|
||||
When using the [`flakes` and `nix-command` experimental features](https://nixos.org/manual/nix/stable/command-ref/new-cli/nix3-flake),
|
||||
a local directory within a Flake is always copied into the Nix store *completely* unless it is a Git repository!
|
||||
:::
|
||||
|
||||
This implicit coercion also works for files:
|
||||
|
||||
```shell-session
|
||||
$ touch some-file
|
||||
```
|
||||
|
||||
```shell-session
|
||||
nix-repl> fs.trace ./some-file
|
||||
trace: /home/user
|
||||
trace: - some-file (regular)
|
||||
```
|
||||
|
||||
In addition to the included file, this also prints its [file type](https://nixos.org/manual/nix/stable/language/builtins.html#builtins-readFileType).
|
||||
|
||||
|
||||
## Example project
|
||||
|
||||
To further experiment with the library, make a sample project.
|
||||
Create a new directory, enter it,
|
||||
and set up `niv` to pin the Nixpkgs dependency:
|
||||
|
||||
```shell-session
|
||||
$ mkdir fileset
|
||||
$ cd fileset
|
||||
$ nix-shell -p niv --run "niv init --nixpkgs nixos/nixpkgs --nixpkgs-branch master"
|
||||
```
|
||||
|
||||
Then create a `default.nix` file with the following contents:
|
||||
|
||||
```{code-block} nix
|
||||
:caption: default.nix
|
||||
{
|
||||
system ? builtins.currentSystem,
|
||||
sources ? import ./nix/sources.nix,
|
||||
}:
|
||||
let
|
||||
pkgs = import sources.nixpkgs {
|
||||
config = { };
|
||||
overlays = [ ];
|
||||
inherit system;
|
||||
};
|
||||
in
|
||||
pkgs.callPackage ./build.nix { }
|
||||
```
|
||||
|
||||
Add two source files to work with:
|
||||
|
||||
```shell-session
|
||||
$ echo hello > hello.txt
|
||||
$ echo world > world.txt
|
||||
```
|
||||
|
||||
## Adding files to the Nix store
|
||||
|
||||
Files in a given file set can be added to the Nix store with [`toSource`](https://nixos.org/manual/nixpkgs/unstable/#function-library-lib.fileset.toSource).
|
||||
The argument to this function requires a `root` attribute to determine which source directory to copy to the store.
|
||||
Only the files in the `fileset` attribute are included in the result.
|
||||
|
||||
Define `build.nix` as follows:
|
||||
|
||||
```{code-block} nix
|
||||
:caption: build.nix
|
||||
{ stdenv, lib }:
|
||||
let
|
||||
fs = lib.fileset;
|
||||
sourceFiles = ./hello.txt;
|
||||
in
|
||||
|
||||
fs.trace sourceFiles
|
||||
|
||||
stdenv.mkDerivation {
|
||||
name = "fileset";
|
||||
src = fs.toSource {
|
||||
root = ./.;
|
||||
fileset = sourceFiles;
|
||||
};
|
||||
postInstall = ''
|
||||
mkdir $out
|
||||
cp -v hello.txt $out
|
||||
'';
|
||||
}
|
||||
```
|
||||
|
||||
The call to `fs.trace` prints the file set that will be used as a derivation input.
|
||||
|
||||
Try building it:
|
||||
|
||||
:::{note}
|
||||
It will take a while to fetch Nixpkgs the first time around.
|
||||
:::
|
||||
|
||||
```
|
||||
$ nix-build
|
||||
trace: /home/user/fileset
|
||||
trace: - hello.txt (regular)
|
||||
this derivation will be built:
|
||||
/nix/store/3ci6avmjaijx5g8jhb218i183xi7bi2n-fileset.drv
|
||||
...
|
||||
'hello.txt' -> '/nix/store/sa4g6h13v0zbpfw6pzva860kp5aks44n-fileset/hello.txt'
|
||||
...
|
||||
/nix/store/sa4g6h13v0zbpfw6pzva860kp5aks44n-fileset
|
||||
```
|
||||
|
||||
But the real benefit of the file set library comes from its facilities for composing file sets in different ways.
|
||||
|
||||
## Difference
|
||||
|
||||
To be able to copy both files `hello.txt` and `world.txt` to the output, add the whole project directory as a source again:
|
||||
|
||||
```{code-block} diff
|
||||
:caption: build.nix
|
||||
{ stdenv, lib }:
|
||||
let
|
||||
fs = lib.fileset;
|
||||
- sourceFiles = ./hello.txt;
|
||||
+ sourceFiles = ./.;
|
||||
in
|
||||
|
||||
fs.trace sourceFiles
|
||||
|
||||
stdenv.mkDerivation {
|
||||
name = "fileset";
|
||||
src = fs.toSource {
|
||||
root = ./.;
|
||||
fileset = sourceFiles;
|
||||
};
|
||||
postInstall = ''
|
||||
mkdir $out
|
||||
- cp -v hello.txt $out
|
||||
+ cp -v {hello,world}.txt $out
|
||||
'';
|
||||
}
|
||||
```
|
||||
|
||||
This will work as expected:
|
||||
|
||||
```shell-session
|
||||
$ nix-build
|
||||
trace: /home/user/fileset (all files in directory)
|
||||
this derivation will be built:
|
||||
/nix/store/fsihp8872vv9ngbkc7si5jcbigs81727-fileset.drv
|
||||
...
|
||||
'hello.txt' -> '/nix/store/wmsxfgbylagmf033nkazr3qfc96y7mwk-fileset/hello.txt'
|
||||
'world.txt' -> '/nix/store/wmsxfgbylagmf033nkazr3qfc96y7mwk-fileset/world.txt'
|
||||
...
|
||||
/nix/store/wmsxfgbylagmf033nkazr3qfc96y7mwk-fileset
|
||||
```
|
||||
|
||||
However, if you run `nix-build` again, the output path will be different!
|
||||
|
||||
```shell-session
|
||||
$ nix-build
|
||||
trace: /home/user/fileset (all files in directory)
|
||||
this derivation will be built:
|
||||
/nix/store/nlh7ismrf27xsnl3m20vfz6rvwlbbbca-fileset.drv
|
||||
...
|
||||
'hello.txt' -> '/nix/store/xknflcvjaa8dj6a6vkg629zmcrgz10rh-fileset/hello.txt'
|
||||
'world.txt' -> '/nix/store/xknflcvjaa8dj6a6vkg629zmcrgz10rh-fileset/world.txt'
|
||||
...
|
||||
/nix/store/xknflcvjaa8dj6a6vkg629zmcrgz10rh-fileset
|
||||
```
|
||||
|
||||
The problem here is that `nix-build` by default creates a `result` symlink in the working directory, which points to the store path just produced:
|
||||
|
||||
```
|
||||
$ ls -l result
|
||||
result -> /nix/store/xknflcvjaa8dj6a6vkg629zmcrgz10rh-fileset
|
||||
```
|
||||
|
||||
Since `src` refers to the whole directory, and its contents change when `nix-build` succeeds, Nix will have to start over every time.
|
||||
|
||||
:::{note}
|
||||
This will also happen without the file set library, e.g. when setting `src = ./.;` directly.
|
||||
:::
|
||||
|
||||
The [`difference`](https://nixos.org/manual/nixpkgs/unstable/#function-library-lib.fileset.difference) function subtracts one file set from another.
|
||||
The result is a new file set that contains all files from the first argument that aren't in the second argument.
|
||||
|
||||
Use it to filter out `./result` by changing the `sourceFiles` definition:
|
||||
|
||||
```{code-block} diff
|
||||
:caption: build.nix
|
||||
{ stdenv, lib }:
|
||||
let
|
||||
fs = lib.fileset;
|
||||
- sourceFiles = ./.;
|
||||
+ sourceFiles = fs.difference ./. ./result;
|
||||
in
|
||||
```
|
||||
|
||||
Building this, the file set library will specify which files are taken from the directory:
|
||||
|
||||
```shell-session
|
||||
$ nix-build
|
||||
trace: /home/user/fileset
|
||||
trace: - build.nix (regular)
|
||||
trace: - default.nix (regular)
|
||||
trace: - hello.txt (regular)
|
||||
trace: - nix (all files in directory)
|
||||
trace: - world.txt (regular)
|
||||
this derivation will be built:
|
||||
/nix/store/zr19bv51085zz005yk7pw4s9sglmafvn-fileset.drv
|
||||
...
|
||||
'hello.txt' -> '/nix/store/vhyhk6ij39gjapqavz1j1x3zbiy3qc1a-fileset/hello.txt'
|
||||
'world.txt' -> '/nix/store/vhyhk6ij39gjapqavz1j1x3zbiy3qc1a-fileset/world.txt'
|
||||
...
|
||||
/nix/store/vhyhk6ij39gjapqavz1j1x3zbiy3qc1a-fileset
|
||||
```
|
||||
|
||||
An attempt to repeat the build will re-use the existing store path:
|
||||
|
||||
```
|
||||
$ nix-build
|
||||
trace: /home/user/fileset
|
||||
trace: - build.nix (regular)
|
||||
trace: - default.nix (regular)
|
||||
trace: - hello.txt (regular)
|
||||
trace: - nix (all files in directory)
|
||||
trace: - world.txt (regular)
|
||||
/nix/store/vhyhk6ij39gjapqavz1j1x3zbiy3qc1a-fileset
|
||||
```
|
||||
|
||||
## Missing files
|
||||
|
||||
Removing the `./result` symlink creates a new problem, though:
|
||||
|
||||
```shell-session
|
||||
$ rm result
|
||||
$ nix-build
|
||||
error: lib.fileset.difference: Second argument (negative set)
|
||||
(/home/user/fileset/result) is a path that does not exist.
|
||||
To create a file set from a path that may not exist, use `lib.fileset.maybeMissing`.
|
||||
```
|
||||
|
||||
Follow the instructions in the error message, and use `maybeMissing` to create a file set from a path that may not exist (in which case the file set will be empty):
|
||||
|
||||
<!-- https://nixos.org/manual/nixpkgs/unstable/#function-library-lib.fileset.maybeMissing -->
|
||||
|
||||
```{code-block} diff
|
||||
:caption: build.nix
|
||||
{ stdenv, lib }:
|
||||
let
|
||||
fs = lib.fileset;
|
||||
- sourceFiles = fs.difference ./. ./result;
|
||||
+ sourceFiles = fs.difference ./. (fs.maybeMissing ./result);
|
||||
in
|
||||
```
|
||||
|
||||
This now works, using the whole directory since `./result` is not present:
|
||||
|
||||
```
|
||||
$ nix-build
|
||||
trace: /home/user/fileset (all files in directory)
|
||||
this derivation will be built:
|
||||
/nix/store/zr19bv51085zz005yk7pw4s9sglmafvn-fileset.drv
|
||||
...
|
||||
/nix/store/vhyhk6ij39gjapqavz1j1x3zbiy3qc1a-fileset
|
||||
```
|
||||
|
||||
Another build attempt will produce a different trace, but the same output path:
|
||||
|
||||
```
|
||||
$ nix-build
|
||||
trace: /home/user/fileset
|
||||
trace: - build.nix (regular)
|
||||
trace: - default.nix (regular)
|
||||
trace: - hello.txt (regular)
|
||||
trace: - nix (all files in directory)
|
||||
trace: - world.txt (regular)
|
||||
/nix/store/vhyhk6ij39gjapqavz1j1x3zbiy3qc1a-fileset
|
||||
```
|
||||
|
||||
## Union (explicitly exclude files)
|
||||
|
||||
There is still a problem:
|
||||
Changing _any_ of the included files causes the derivation to be built again, even though it doesn't depend on those files.
|
||||
|
||||
Append an empty line to `build.nix`:
|
||||
|
||||
```shell-session
|
||||
$ echo >> build.nix
|
||||
```
|
||||
|
||||
Again, Nix will start from scratch:
|
||||
|
||||
```shell-session
|
||||
$ nix-build
|
||||
trace: /home/user/fileset
|
||||
trace: - default.nix (regular)
|
||||
trace: - nix (all files in directory)
|
||||
trace: - build.nix (regular)
|
||||
trace: - string.txt (regular)
|
||||
this derivation will be built:
|
||||
/nix/store/zmgpqlpfz2jq0w9rdacsnpx8ni4n77cn-filesets.drv
|
||||
...
|
||||
/nix/store/6pffjljjy3c7kla60nljk3fad4q4kkzn-filesets
|
||||
```
|
||||
|
||||
One way to fix this is to use [`unions`](https://nixos.org/manual/nixpkgs/unstable/#function-library-lib.fileset.unions).
|
||||
|
||||
Create a file set containing a union of the files to exclude (`fs.unions [ ... ]`), and subtract it (`difference`) from the complete directory (`./.`):
|
||||
|
||||
```{code-block} nix
|
||||
:caption: build.nix
|
||||
sourceFiles =
|
||||
fs.difference
|
||||
./.
|
||||
(fs.unions [
|
||||
(fs.maybeMissing ./result)
|
||||
./default.nix
|
||||
./build.nix
|
||||
./nix
|
||||
]);
|
||||
```
|
||||
|
||||
Changing any of the excluded files now doesn't necessarily cause a new build anymore:
|
||||
|
||||
```
|
||||
$ echo >> build.nix
|
||||
```
|
||||
|
||||
```
|
||||
$ nix-build
|
||||
trace: /home/user/fileset
|
||||
trace: - hello.txt (regular)
|
||||
trace: - world.txt (regular)
|
||||
/nix/store/ckn40y7hgqphhbhyrq64h9r6rvdh973r-fileset
|
||||
```
|
||||
|
||||
## Filter
|
||||
|
||||
The [`fileFilter`](https://nixos.org/manual/nixpkgs/unstable/#function-library-lib.fileset.fileFilter) function allows filtering file sets such that each included file satisfies the given criteria.
|
||||
|
||||
Use it to select all files with a name ending in `.nix`:
|
||||
|
||||
```{code-block} diff
|
||||
:caption: build.nix
|
||||
sourceFiles =
|
||||
fs.difference
|
||||
./.
|
||||
(fs.unions [
|
||||
(fs.maybeMissing ./result)
|
||||
- ./default.nix
|
||||
- ./build.nix
|
||||
+ (fs.fileFilter (file: lib.hasSuffix ".nix" file.name) ./.)
|
||||
./nix
|
||||
]);
|
||||
```
|
||||
|
||||
This does not change the result, even if we add a new `.nix` file.
|
||||
|
||||
```shell-session
|
||||
$ nix-build
|
||||
trace: /home/user/fileset
|
||||
trace: - hello.txt (regular)
|
||||
trace: - world.txt (regular)
|
||||
/nix/store/ckn40y7hgqphhbhyrq64h9r6rvdh973r-fileset
|
||||
```
|
||||
|
||||
Notably, the approach of using `difference ./.` explicitly selects the files to _exclude_, which means that new files added to the source directory are included by default.
|
||||
Depending on your project, this might be a better fit than the alternative in the next section.
|
||||
|
||||
## Union (explicitly include files)
|
||||
|
||||
To contrast the previous approach, `unions` can also be used to select only the files to _include_.
|
||||
This means that new files added to the current directory would be ignored by default.
|
||||
|
||||
Create some additional files:
|
||||
|
||||
```shell-session
|
||||
$ mkdir src
|
||||
$ touch build.sh src/select.{c,h}
|
||||
```
|
||||
|
||||
Then create a file set from only the files to be included explicitly:
|
||||
|
||||
```{code-block} nix
|
||||
:caption: build.nix
|
||||
{ stdenv, lib }:
|
||||
let
|
||||
fs = lib.fileset;
|
||||
sourceFiles = fs.unions [
|
||||
./hello.txt
|
||||
./world.txt
|
||||
./build.sh
|
||||
(fs.fileFilter
|
||||
(file:
|
||||
lib.hasSuffix ".c" file.name
|
||||
|| lib.hasSuffix ".h" file.name
|
||||
)
|
||||
./src
|
||||
)
|
||||
];
|
||||
in
|
||||
|
||||
fs.trace sourceFiles
|
||||
|
||||
stdenv.mkDerivation {
|
||||
name = "fileset";
|
||||
src = fs.toSource {
|
||||
root = ./.;
|
||||
fileset = sourceFiles;
|
||||
};
|
||||
postInstall = ''
|
||||
cp -vr . $out
|
||||
'';
|
||||
}
|
||||
```
|
||||
|
||||
The `postInstall` script is simplified to rely on the sources to be pre-filtered appropriately:
|
||||
|
||||
```shell-session
|
||||
$ nix-build
|
||||
trace: /home/user/fileset
|
||||
trace: - build.sh (regular)
|
||||
trace: - hello.txt (regular)
|
||||
trace: - src (all files in directory)
|
||||
trace: - world.txt (regular)
|
||||
this derivation will be built:
|
||||
/nix/store/sjzkn07d6a4qfp60p6dc64pzvmmdafff-fileset.drv
|
||||
...
|
||||
'.' -> '/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset'
|
||||
'./build.sh' -> '/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset/build.sh'
|
||||
'./hello.txt' -> '/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset/hello.txt'
|
||||
'./world.txt' -> '/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset/world.txt'
|
||||
'./src' -> '/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset/src'
|
||||
'./src/select.c' -> '/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset/src/select.c'
|
||||
'./src/select.h' -> '/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset/src/select.h'
|
||||
...
|
||||
/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset
|
||||
```
|
||||
|
||||
Only the specified files are used, even when a new one is added:
|
||||
|
||||
```shell-session
|
||||
$ touch src/select.o README.md
|
||||
|
||||
$ nix-build
|
||||
trace: - build.sh (regular)
|
||||
trace: - hello.txt (regular)
|
||||
trace: - src
|
||||
trace: - select.c (regular)
|
||||
trace: - select.h (regular)
|
||||
trace: - world.txt (regular)
|
||||
/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset
|
||||
```
|
||||
|
||||
## Matching files tracked by Git
|
||||
|
||||
If a directory is part of a Git repository, passing it to [`gitTracked`](https://nixos.org/manual/nixpkgs/unstable/#function-library-lib.fileset.toSource) gives you a file set that only includes files tracked by Git.
|
||||
|
||||
Create a local Git repository and add all files except `src/select.o` and `./result` to it:
|
||||
|
||||
```shell-session
|
||||
$ git init
|
||||
Initialized empty Git repository in /home/user/fileset/.git/
|
||||
$ git add -A
|
||||
$ git reset src/select.o result
|
||||
```
|
||||
|
||||
Re-use this selection of files with `gitTracked`:
|
||||
|
||||
```{code-block} nix
|
||||
:caption: build.nix
|
||||
sourceFiles = fs.gitTracked ./.;
|
||||
```
|
||||
|
||||
Build it again:
|
||||
|
||||
```shell-session
|
||||
$ nix-build
|
||||
warning: Git tree '/home/user/fileset' is dirty
|
||||
trace: /home/vg/src/nix.dev/fileset
|
||||
trace: - README.md (regular)
|
||||
trace: - build.nix (regular)
|
||||
trace: - build.sh (regular)
|
||||
trace: - default.nix (regular)
|
||||
trace: - hello.txt (regular)
|
||||
trace: - nix (all files in directory)
|
||||
trace: - src
|
||||
trace: - select.c (regular)
|
||||
trace: - select.h (regular)
|
||||
trace: - world.txt (regular)
|
||||
this derivation will be built:
|
||||
/nix/store/p9aw3fl5xcjbgg9yagykywvskzgrmk5y-fileset.drv
|
||||
...
|
||||
/nix/store/cw4bza1r27iimzrdbfl4yn5xr36d6k5l-fileset
|
||||
```
|
||||
|
||||
This includes too much though, as not all of these files are needed to build the derivation as originally intended.
|
||||
|
||||
:::{note}
|
||||
When using the [`flakes` and `nix-command` experimental features](https://nixos.org/manual/nix/stable/command-ref/new-cli/nix3-flake),
|
||||
it's [not really possible](https://github.com/NixOS/nix/issues/9292) to use this function, even with
|
||||
|
||||
|
||||
```shell-session
|
||||
$ nix build path:.
|
||||
```
|
||||
|
||||
However it's also not needed, because by default, `nix build` only allows access to files tracked by Git.
|
||||
:::
|
||||
|
||||
## Intersection
|
||||
|
||||
This is where `intersection` comes in.
|
||||
It allows creating a file set that consists only of files that are in _both_ of two given file sets.
|
||||
|
||||
Select all files that are both tracked by Git *and* relevant for the build:
|
||||
|
||||
```{code-block} nix
|
||||
:caption: build.nix
|
||||
sourceFiles =
|
||||
fs.intersection
|
||||
(fs.gitTracked ./.)
|
||||
(fs.unions [
|
||||
./hello.txt
|
||||
./world.txt
|
||||
./build.sh
|
||||
./src
|
||||
]);
|
||||
```
|
||||
|
||||
This will produce the same output as in the other approach and therefore re-use a previous build result:
|
||||
|
||||
```shell-session
|
||||
$ nix-build
|
||||
warning: Git tree '/home/user/fileset' is dirty
|
||||
trace: - build.sh (regular)
|
||||
trace: - hello.txt (regular)
|
||||
trace: - src
|
||||
trace: - select.c (regular)
|
||||
trace: - select.h (regular)
|
||||
trace: - world.txt (regular)
|
||||
/nix/store/zl4n1g6is4cmsqf02dci5b2h5zd0ia4r-fileset
|
||||
```
|
||||
|
||||
## Conclusion
|
||||
|
||||
We have shown some examples on how to use all of the fundamental file set functions.
|
||||
For more complex use cases, they can be composed as needed.
|
||||
|
||||
For the complete list and more details, see the [`lib.fileset` reference documentation](https://nixos.org/manual/nixpkgs/unstable/#sec-fileset).
|
|
@ -10,6 +10,7 @@ These sections contains series of lessons to get started.
|
|||
first-steps/index.md
|
||||
nix-language.md
|
||||
Packaging existing software <packaging-existing-software.md>
|
||||
file-sets.md
|
||||
nixos/index.md
|
||||
cross-compilation.md
|
||||
module-system/module-system.md
|
||||
|
|
|
@ -3,6 +3,7 @@ Docker
|
|||
Dolstra
|
||||
Eelco
|
||||
Frequently Asked Questions
|
||||
Git
|
||||
GitHub
|
||||
GitHub Actions
|
||||
GitHub Actions Cache
|
||||
|
|
Loading…
Reference in a new issue