mirror of
https://github.com/privatevoid-net/nix-super.git
synced 2024-11-15 10:46:15 +02:00
140 lines
7.2 KiB
Markdown
140 lines
7.2 KiB
Markdown
|
# Import From Derivation
|
||
|
|
||
|
The value of a Nix expression can depend on the contents of a [store object](@docroot@/glossary.md#gloss-store-object).
|
||
|
|
||
|
Passing an expression `expr` that evaluates to a [store path](@docroot@/glossary.md#gloss-store-path) to any built-in function which reads from the filesystem constitutes Import From Derivation (IFD):
|
||
|
|
||
|
- [`import`](./builtins.md#builtins-import)` expr`
|
||
|
- [`builtins.readFile`](./builtins.md#builtins-readFile)` expr`
|
||
|
- [`builtins.readFileType`](./builtins.md#builtins-readFileType)` expr`
|
||
|
- [`builtins.readDir`](./builtins.md#builtins-readDir)` expr`
|
||
|
- [`builtins.pathExists`](./builtins.md#builtins-pathExists)` expr`
|
||
|
- [`builtins.filterSource`](./builtins.md#builtins-filterSource)` f expr`
|
||
|
- [`builtins.path`](./builtins.md#builtins-path)` { path = expr; }`
|
||
|
- [`builtins.hashFile`](./builtins.md#builtins-hashFile)` t expr`
|
||
|
- `builtins.scopedImport x drv`
|
||
|
|
||
|
When the store path needs to be accessed, evaluation will be paused, the corresponding store object [realised], and then evaluation resumed.
|
||
|
|
||
|
[realised]: @docroot@/glossary.md#gloss-realise
|
||
|
|
||
|
This has performance implications:
|
||
|
Evaluation can only finish when all required store objects are realised.
|
||
|
Since the Nix language evaluator is sequential, it only finds store paths to read from one at a time.
|
||
|
While realisation is always parallel, in this case it cannot be done for all required store paths at once, and is therefore much slower than otherwise.
|
||
|
|
||
|
Realising store objects during evaluation can be disabled by setting [`allow-import-from-derivation`](../command-ref/conf-file.md#conf-allow-import-from-derivation) to `false`.
|
||
|
Without IFD it is ensured that evaluation is complete and Nix can produce a build plan before starting any realisation.
|
||
|
|
||
|
## Example
|
||
|
|
||
|
In the following Nix expression, the inner derivation `drv` produces a file with contents `hello`.
|
||
|
|
||
|
```nix
|
||
|
# IFD.nix
|
||
|
let
|
||
|
drv = derivation {
|
||
|
name = "hello";
|
||
|
builder = "/bin/sh";
|
||
|
args = [ "-c" "echo -n hello > $out" ];
|
||
|
system = builtins.currentSystem;
|
||
|
};
|
||
|
in "${builtins.readFile drv} world"
|
||
|
```
|
||
|
|
||
|
```shellSession
|
||
|
nix-instantiate IFD.nix --eval --read-write-mode
|
||
|
```
|
||
|
|
||
|
```
|
||
|
building '/nix/store/348q1cal6sdgfxs8zqi9v8llrsn4kqkq-hello.drv'...
|
||
|
"hello world"
|
||
|
```
|
||
|
|
||
|
The contents of the derivation's output have to be [realised] before they can be read with [`readFile`](./builtins.md#builtins-readFile).
|
||
|
Only then evaluation can continue to produce the final result.
|
||
|
|
||
|
## Illustration
|
||
|
|
||
|
As a first approximation, the following data flow graph shows how evaluation and building are interleaved, if the value of a Nix expression depends on realising a [store object].
|
||
|
Boxes are data structures, arrow labels are transformations.
|
||
|
|
||
|
```
|
||
|
+----------------------+ +------------------------+
|
||
|
| Nix evaluator | | Nix store |
|
||
|
| .----------------. | | |
|
||
|
| | Nix expression | | | |
|
||
|
| '----------------' | | |
|
||
|
| | | | |
|
||
|
| evaluate | | |
|
||
|
| | | | |
|
||
|
| V | | |
|
||
|
| .------------. | | .------------------. |
|
||
|
| | derivation |----|-instantiate-|->| store derivation | |
|
||
|
| '------------' | | '------------------' |
|
||
|
| | | | |
|
||
|
| | | realise |
|
||
|
| | | | |
|
||
|
| | | V |
|
||
|
| .----------------. | | .--------------. |
|
||
|
| | Nix expression |<-|----read-----|----| store object | |
|
||
|
| '----------------' | | '--------------' |
|
||
|
| | | | |
|
||
|
| evaluate | | |
|
||
|
| | | | |
|
||
|
| V | | |
|
||
|
| .------------. | | |
|
||
|
| | value | | | |
|
||
|
| '------------' | | |
|
||
|
+----------------------+ +------------------------+
|
||
|
```
|
||
|
|
||
|
In more detail, the following sequence diagram shows how the expression is evaluated step by step, and where evaluation is blocked to wait for the build output to appear.
|
||
|
|
||
|
```
|
||
|
.-------. .-------------. .---------.
|
||
|
|Nix CLI| |Nix evaluator| |Nix store|
|
||
|
'-------' '-------------' '---------'
|
||
|
| | |
|
||
|
|evaluate IFD.nix| |
|
||
|
|--------------->| |
|
||
|
| | |
|
||
|
| evaluate `"${readFile drv} world"` |
|
||
|
| | |
|
||
|
| evaluate `readFile drv` |
|
||
|
| | |
|
||
|
| evaluate `drv` as string |
|
||
|
| | |
|
||
|
| |instantiate /nix/store/...-hello.drv|
|
||
|
| |----------------------------------->|
|
||
|
| : |
|
||
|
| : realise /nix/store/...-hello.drv |
|
||
|
| :----------------------------------->|
|
||
|
| : |
|
||
|
| |--------.
|
||
|
| : | |
|
||
|
| (evaluation blocked) | echo hello > $out
|
||
|
| : | |
|
||
|
| |<-------'
|
||
|
| : /nix/store/...-hello |
|
||
|
| |<-----------------------------------|
|
||
|
| | |
|
||
|
| resume `readFile /nix/store/...-hello` |
|
||
|
| | |
|
||
|
| | readFile /nix/store/...-hello |
|
||
|
| |----------------------------------->|
|
||
|
| | |
|
||
|
| | hello |
|
||
|
| |<-----------------------------------|
|
||
|
| | |
|
||
|
| resume `"${"hello"} world"` |
|
||
|
| | |
|
||
|
| resume `"hello world"` |
|
||
|
| | |
|
||
|
| "hello world" | |
|
||
|
|<---------------| |
|
||
|
.-------. .-------------. .---------.
|
||
|
|Nix CLI| |Nix evaluator| |Nix store|
|
||
|
'-------' '-------------' '---------'
|
||
|
```
|