Nix-Tutorium vom Nationalen Forschungsinstitut für Informatik und Automatisierung (INRIA): Teil 6
Lesezeit: 13 MinutenIm letzten inria-Beitrag, Convenient Management of Inputs with Nix Flakes, geht es um Flakes. Dabei handelt es sich um ein relativ neues Feature, das den Umgang mit Build-Inputs vereinfachen und perfekte Reproduzierbarkeit gewährleisten soll.
Darüber hinaus erfahren wir zunächst sehr wenig bis gar nichts darüber, was Flakes sind. Im Prinzip scheint es ein neues Format zu sein, mit dem Derivations, Entwicklungsumgebungen und andere Dinge definiert werden. Also ganz so, wie wir es seit dem zweiten Beitrag getan haben. Ich denke wir haben es nicht mit etwas wesentlich Neues zu tun.1 Es handelt sich eher um einen weiteren Schritt, um unsere Praxis zu verfeinern.
Im Einzelnen lernen wir:
- Wie man Flakes und das Kommandozeilenwerkzeug
nix
aktiviert - Wie wir Pakete innerhalb von Flakes definieren
- Wie wir in Flakes enthaltene Pakete bauen
- Wie wir in Flakes enthaltene Pakete ausführen
- Wie wir Entwicklungsumgebungen innerhalb von Flakes definieren und wie wir sie betreten
Flakes aktivieren
Auch wenn Flakes in der Nix-Community mittlerweile eine sehr breite Verbreitung gefunden haben, gelten sie noch immer als experimentelles Feature. Sie müssen ausdrücklich aktiviert werden, um verwendet werden zu können. Auch das Werkzeug, über das mit ihnen interagiert wird (der neue nix
-Befehl) muss freigegeben werden.
Eine Möglichkeit wäre, sie bei Verwendung auf der Kommandozeile freizuschalten. Dazu dient ein spezielles Flag:
--experimental-features 'nix-command flakes'
Nur die wenigsten werden das alles bei jeder Verwendung tippen wollen.
Der Nix-Paketmanager kann über eine Konfigurationsdatei eingestellt werden. Für individuelle Benutzer findet sich die Datei in ~/.config/nix/nix.conf
. Daneben gibt es eine Datei, mit der Nix systemweit konfiguriert werden kann (/etc/nix/nix.conf
). Um die entsprechenden Features zu aktivieren, kann folgende Zeile hinzugefügt werden:
experimental-features = nix-command flakes
Leider wird im Beitrag nicht weiter auf die Konfigurationsdatei eingegangen. Welche anderen Einstellungen können darin vorgenommen werden?
Flakes erstellen
Wir erfahren, dass neue Flakes auf der Grundlage von Templates erzeugt werden können. Von Haus aus kommt Nix mit einer Reihe solcher Vorlagen. Diese können wir uns mit einem Befehl anzeigen lassen:
nix flake show templates
Sie werden praktischerweise mit Beschreibungen versehen, die ihren intendierten Verwendungszweck schildern. Darüber hinaus können selbst neue Templates definiert werden. Der Beitrag sagt jedoch nichts dazu, wie wir das machen würden.
Um ein neues Flake auf der Grundlage einer vorhandenen Vorlage zu erstellen, kann folgender Befehl verwendet werden:
nix flake init -t templates#<Template-Name> <Verzeichnis>
Wenn kein Verzeichnis-Pfad übergeben wird, dann wird das neue Flake im Arbeitsverzeichnis erstellt. Es wird im Beitrag gesagt, dass wir ein Default-Template bestimmen können und dass standardmäßig trivial
als Default gesetzt ist. Wenn wir das Flake im Arbeitsverzeichnis und auf der Grundlage der trivial
-Vorlage erstellen wollen, kann der obige Befehl also verkürzt werden:
nix flake init -t templates
Leider erfahren wir nicht, wie der Default-Wert geändert werden kann. Mutmaßlich müsste man dazu wieder eine Änderung an der Nix-Konfigurationsdatei vornehmen?
Tatsächlich ist -t templates
selbst wiederum ein Default. Bei passenden Einstellungen kann der Befehl zum Erstellen eines Flakes demnach reduziert werden auf:
nix flake init
Das erstellte Flake ist nicht ohne Weiteres verfügbar. Flakes müssen notwendig in ein Versionsverwaltungssystem eingecheckt werden. Das bedeutet, dass wir nur dann mittels nix flake
mit einem Flake interagieren können, wenn die entsprechende Datei der Git Staging Area hinzugefügt wurde. Daraus folgt, dass das Verzeichnis, in dem sich die Flake-Dateien befinden, ein Git-Repo enthalten muss.
git init
git add flake.nix
Interaktion mit Flakes
Das nix
-Werkzeug stellt eine Reihe von Unterbefehlen bereit, um mit Flakes zu interagieren. Wir einleitend vermutet wurde, sind Flakes unter anderem ein Mechanismus, um Pakete (Derivations) zu definieren. Es gibt einen neuen Befehl, um alle im Flake enthaltenen Komponenten (darunter Pakete) aufzulisten:
nix flake show
Bisher haben wir Pakete mit nix-build
gebaut. Wir erfahren nun, dass wir die in einem Flake enthaltenen Pakete mit dem Unterbefehl nix build
bauen können. Es wird nicht ausdrücklich auf die Gemeinsamkeiten und Unterschide zwischen nix-build
und nix build
eingegangen. Zumindest die Syntax ist ein wenig anders:
nix build <Verzeichnis>#<Paketname>
Wenn sich im Arbeitsverzeichnis eine flake.nix
befindet, die ein Paket namens hello
enthält, dann können wir das Paket folgendermaßen bauen:
nix build .#hello
Wir erfahren, dass auch durch den neuen Unterbefehl eine Verlinkung erzeugt wird, die auf den Build-Output im Nix-Store zeigt (./result
).
Viele Pakete umfassen ausführbare Binärdateien. Wenn wir ein Flake gebaut haben, finden sich diese Dateien im Unterverzeichnis, das daraufhin im Nix-Store erstellt wurde. Das neue Kommandozeilenwerkzeug erlaubt es uns, diese Hintergründe zu vergessen. Stattdessen können wir uns vorstellen, wir würden die in Flakes enthaltenen Pakete direkt ausführen. Die Syntax dabei ist völlig analog zur Verwendung von nix build
:
nix run <Verzeichnis>#<Paketname>
Wir erfahren, dass bereits nix-build
einen Zugang bot, um die in einem Paket enthaltenen Binärdateien auszuführen. Wie nun oft gesagt, resultiert der Build von Paketen in Unterverzeichnissen im Nix-Store, die gegebenenfalls ausführbare Binärdateien enthalten. Der Link ./result
verweist auf dieses Unterverzeichnis. Für gewöhnlich finden sich die Binärdateien im Unterverzeichnis /bin
.
Das hello
-Paket hat genau diesen Aufbau. Um es auszuführen, kann somit folgender Befehl verwendet werden:
./result/bin/hello
Das gilt sowohl für nix-build
wie für nix build
.
Der Aufbau eines Flake
Wie zuvor wollen wir eine Datei schreiben, in der ein oder mehr Pakete (Derivations) definiert werden. Bisher haben wir dazu eine Funktion definiert. In einer flake.nix
wird nun eine Attributmenge definiert. Für gewöhnlich werden dabei genau drei Attribute festgelegt.2
Eines der drei Attribute, outputs
, enthält als Wert eine Funktion, die stark dem ähnelt, was wir schon kennen. Mit dem inputs
-Attribut werden Dependencies für unsere Pakete gesetzt. Durch das description
-Attribut schließlich kann der Zweck unserer Flakes (in Form eines Strings) beschrieben werden.
Hier das Flake, das bei der Verwendung der trivial
-Vorlage automatisch erstellt wird:
{
description = "A very basic flake";
inputs = {
nixpkgs.url = "github:nixos/nixpkgs/22.05";
};
outputs = { self, nixpkgs }: {
packages.x86_64-linux.hello = nixpkgs.legacyPackages.x86_64-linux.hello;
defaultPackage.x86_64-linux = self.packages.x86_64-linux.hello;
};
}
Inputs
Der Zweck des inputs
-Attributs liegt darin, die spezifische Version einer Dependency anzupinnen. Im Beispiel wird festgelegt, dass Version 22.05
vom Nixpkgs-Repo als Input verwendet werden soll.
Wie bei vielen Paketverwaltungssystemen werden die spezifischen Dependency-Versionen, die als Inputs verwendet werden, in einer lock
-Datei dokumentiert. Nix verwendet dazu flake.lock
. Für das Beispiel-Projekt sieht diese Datei folgendermaßen aus:
{
'nodes':
{
'nixpkgs':
{
'locked':
{
'lastModified': 1653936696,
'narHash': 'sha256-M6bJShji9AIDZ7Kh7CPwPBPb/T7RiVev2PAcOi4fxDQ=',
'owner': 'nixos',
'repo': 'nixpkgs',
'rev': 'ce6aa13369b667ac2542593170993504932eb836',
'type': 'github',
},
'original':
{
'owner': 'nixos',
'ref': '22.05',
'repo': 'nixpkgs',
'type': 'github',
},
},
'root': { 'inputs': { 'nixpkgs': 'nixpkgs' } },
},
'root': 'root',
'version': 7,
}
Die flake.nix
wird automatisch aktualisiert, wenn wir die Inputs unserer flake.nix
anpassen. Leider wird im Beitrag nicht darauf eingegangen, wann genau das passiert. Wenn flake.nix
seit unserer letzten Verwendung von nix flake
verändert wurde, scheint eine erneute Verwendung des Unterbefehls zu einer Anpassung der Lock-Datei zu führen. Wir erhalten folgende Warnung:
nix flake show
warning: updating lock file '/tmp/tuto-nix/flake.lock':
Inputs aktualisieren
Wir erfahren, dass Inputs aktualisiert werden können. Wir können entweder alle Inputs zugleich aktualisieren oder nur einzelne Inputs.
nix flake update
nix flake lock --update-input <Input-Name>
Leider wird nicht ausgeführt, was das genau bedeutet. Mutmaßlich werden daraufhin neuere Version der Dependencies verwendet, wenn wir die als Output definierten Derivations bauen oder ausführen.
Auf welcher Grundlage entscheidet Nix, ob eine neuere Version vorhanden ist? Wenn ich das richtig verstehe, dann ersetzen Flakes das Konzept von Kanälen. Ich denke die gesetzten Kanäle entscheiden deshalb nicht über den Ablauf der Aktualisierung.
Es stellt sich auch die Frage, was das Resultat einer Aktualisierung ist. Wird die flake.nix
oder die flake.lock
angepasst? Oder vielleicht beide?
Outputs
Über das outputs
-Attribut erfahren wir Folgendes:
The
outputs
field is a function taking as input the inputs and returning a set. This set should have a specific hierarchy. First the type of output (…), then the target architecture (…) and finally the name of the output.
Der Rückgabewert der Output-Funktion ist demnach eine Menge. Im Nix-Ökosystem heißt “Menge” immer Attributmenge. “Outputs” (Plural) legt nahe, dass die Menge auf der obersten Ebene Elemente enthält, die jeweils einen Output repräsentieren?3
Nach der eben zitierten Erklärung hat jeder Output drei Merkmale:
- Sie haben einen Output-Typ. Daraus folgt, dass in Flakes nicht nur Pakete definiert werden können. Wir erfahren von drei Typen:
packages
,devShells
undchecks
. Die beiden ersten kennen wir bereits. Was sindchecks
? Und welche weiteren Typen gibt es? - Sie haben eine Ziel-Architektur. Anwendungen werden für bestimmte CPU-Architekturen gebaut. Dazu gehören beispielsweise
x86_64-linux
,aarch64-linux
undx86_64-darwin
. - Outputs erhalten einen Namen. Wir haben oben bereits gesehen, dass sie darüber von Außen referenziert werden (wie bei
nix build
undnix run
).
Darüber hinaus ist von einer “spezifischen Hierarchie” die Rede. Wenn ich das richtig lese, dann sind die Typen ganz oben; in der Ebene darunter ist die Architektur; und auf der dritten Ebene sind die Namen. Wie wir sehen werden repräsentieren die Namen die Dinge selbst (das eigentliche Paket beispielsweise). Hierarchie heißt hier, dass Attributmengen ineinander verschachtelt werden.
Hier ein geringfügig komplexeres Beispiel, das diese Punkte illustriert:
{
description = "A very basic flake";
inputs = {
nixpkgs.url = "github:nixos/nixpkgs/22.05";
};
outputs = { self, nixpkgs }:
let
system = "x86_64-linux";
pkgs = import nixpkgs { inherit system; };
in {
packages.${system} = rec {
chord = pkgs.callPackage ./pkgs/chord {};
chord_custom_sg = pkgs.callPackage ./pkgs/chord { simgrid = custom_simgrid; };
custom_simgrid = pkgs.callPackage ./pkgs/simgrid/custom.nix {};
};
};
}
Als ich mir die Datei zum ersten Mal angeschaut habe, fiel mir die Definition der Architektur sofort ins Auge (system
). Ehrlich gesagt brauchte ich einen Moment länger, um die Hierarchie wiederzufinden, von der eben die Rede war. Das liegt vielleicht an der vereinfachten Syntax, bei der geschweifte Klammern vermieden werden. Bei mehr geschweiften Klammern wäre ich vielleicht sofort auf die ineinander verschachtelten Attributmengen gestoßen.
Anders als ich auf den ersten Blick dachte wird hier keine Variable definiert. Die Funktion gibt eine Attributmenge zurück. Die Attributmenge hat nur ein einziges Element, ein Attribut mit dem Namen packages
. Der Wert des Attributs ist eine weitere Attributmenge, die wiederum auch nur ein Element hat. Der Name des Attributs wurde in einer Variable definiert; es handelt sich um die CPU-Architektur. Der Wert des System-Attributs ist wiederum eine Attributmenge, dieses Mal mit drei Attributen:
rec {
chord = pkgs.callPackage ./pkgs/chord {};
chord_custom_sg = pkgs.callPackage ./pkgs/chord { simgrid = custom_simgrid; };
custom_simgrid = pkgs.callPackage ./pkgs/simgrid/custom.nix {};
};
Der Wert jedes dieser Attribute dürfte ein Paket repärsentieren. Der Paketname scheint dabei der Attributname zu sein.
Der Wert der Pakete ergibt sich daraus, dass (mit callPackage
) Derivations aufgerufen werden, die an anderer Stelle definiert wurden. Dieses Design-Pattern kennen wir prinzipiell bereits aus einem früheren Beitrag.
Dort wurde jedoch ein Mini-Paketrepo verwendet, das wir (oder die inria-Mitwirkenden) selbst definiert haben. Hier scheinen Derivation-Definitionen aus dem Nixpkgs-Repo verwendet zu werden. pkgs
wird folgendermaßen definiert:
pkgs = import nixpkgs { inherit system; };
Wir haben nun drei Quellen: inputs
zum Flake, Argumente zur outputs
-Funktion und Importe (deren Rückgabe im let
-Teil einer Variable zugewiesen werden kann). Leider wird nicht erläutert, unter welchen Umständen wir auf welche dieser Komponenten zugreifen würden.
Aus der Verwendung des packages
-Attribut folgt, dass wir ein oder mehr Pakete definieren wollen (Typ der Outputs). Hier finden wir die im obigen Zitat angedeutet Hierarchie:
packages.${system} = rec {
chord = pkgs.callPackage ./pkgs/chord {};
chord_custom_sg = pkgs.callPackage ./pkgs/chord { simgrid = custom_simgrid; };
custom_simgrid = pkgs.callPackage ./pkgs/simgrid/custom.nix {};
};
Sicherlich könnten wir im selben Flake auch Pakete für weitere Architekturen definieren. Und wir könnten neben Paketen noch devShells
oder andere Dinge definieren.
outputs = { self, nixpkgs }: {
packages = {
x86_64-linux {
paket_1_fuer_architektur_1 = ...;
paket_2_fuer_architektur_1 = ...;
}
x86_64-darwin {
paket_1_fuer_architektur_2 = ...;
paket_2_fuer_architektur_2 = ...;
}
};
devShells = ...
};
Abwärtskompatibilität
Es wird darauf hingewiesen, dass nix-build
nicht ohne Weiteres verwendet werden kann, um Flakes zu bauen. Es gibt jedoch eine Möglichkeit, um Abwärtskompatibilität zum “alten” Ansatz herzustellen.
Dazu muss dem Flake-Verzeichnis eine default.nix
-Datei mit dem folgenden Inhalt hinzugefügt werden:
(import (
fetchTarball {
url = "https://github.com/edolstra/flake-compat/archive/12c64ca55c1014cdc1b16ed5a804aa8576601ff2.tar.gz";
sha256 = "0jm6nzb83wa6ai17ly9fzpqc40wg1viib8klq8lby54agpl213w5"; }
) {
src = ./.;
}).defaultNix
Es wird nicht weiter auf den Inhalt der Datei oder darauf eingeangen, warum genau durch sie Abwärtskombailität hergestellt wird. Ist vielleicht auch nicht so wichtig. Allgemein frage ich mich, unter welchen Umständen wir auf den alten Befehl zurückgreifen wollen würden.
Tatsächlich wird angemerkt, dass wir nun noch immer nicht völlig wie bisher mit den definierten Derivations interagieren können. Wenn wir ein Paket mit nix-build
bauen möchten, das in einer Flake-Datei definiert wurde, dann müssen wir den genauen Attributpfad angeben.
Für das hello
-Paket im ersten Beispiel wäre das etwa:
nix-build -A packages.x86_64-linux.hello
Bei einer “gewöhnlichen” default.nix
, wie wir sie in den vorausgegangen Beiträgn kennengelernt haben, reichte ein weitaus kürzerer Befehl (nix-build -A hello
).
Flake Registries
Wir erfahren etwas über Flake Registries. Dabei handelt es sich um ein Feature, das mit Kanälen verglichen wird.
Bilden sie die Grundlage für Updates unserer Inputs? Leider erfahren wir nahezu gar nichts darüber, was sie wirklich sind oder wofür sie verwendet werden. Stattdessen werden nur die Befehle aufgeführt, um sie aufzulisten, ihren Inhalt anzuzeigen und um neue Flakes dem Register hinzuzufügen. Das sind die Unterbefehle:
nix registry list
nix flake show <Flake-Name>
nix registry add <Name> <URL>
Im Beitrag wurde das Mini-Paketrepo hinzugefügt, das in vorausgegangenen Beiträgen definiert wurde (es erhält den Namen mypkgs
).
Das sagt das offizielle Bedienungshandbuch dazu:
Flake registries are a convenience feature that allows you to refer to flakes using symbolic identifiers such as
nixpkgs
, rather than full URLs such asgit://github.com/NixOS/nixpkgs
. You can use these identifiers on the command line (e.g. when you donix run nixpkgs#hello
) or in flake input specifications inflake.nix
files. The latter are automatically resolved to full URLs and recorded in the flake’sflake.lock
file.
Das klingt nicht so richtig wie Kanäle, oder? Eher wie NIX_PATH
. Im Bedienungshandbuch wird auch gesagt, dass es drei verschiedene Register gibt (global, systemweit, für einzelne Benutzer).
Entwicklungsumgebungen in Flakes
Im letzten Abschnitt der Serie wird ein Flake definiert, das eine Entwicklungsumgebung für Experimente bereitstellt. In der Umgebung ist das chord
-Paket verfügbar; sie erhält deshalb den Namen chordShell
. Wie im “alten” Ansatz wird die Umgebung mit mkShell
definiert.
Das zeigt, dass Shell-Umgebungen wie Pakete im Rahmen von Flakes einen Namen erhalten. Über diesen können sie wieder auf der Kommandozeile referenziert werden. Ebenso wie bei Paketen werden sie für eine bestimmte Ziel-Architektur definiert. Der Output-Typ heißt devShells
.
Hier das Beispiel:
{
description = "My Experiments repo";
inputs = {
nixpkgs.url = "github:nixos/nixpkgs/22.05";
mypkgs.url = "git+https://gitlab.inria.fr/qguillot/mypkgs_example";
};
outputs = { self, nixpkgs, mypkgs }:
let
system = "x86_64-linux";
pkgs = import nixpkgs { inherit system; };
in {
devShells.${system} = {
chordShell = pkgs.mkShell {
buildInputs = [
mypkgs.packages.${system}.chord
];
};
};
};
}
Das Flake hat zwei Inputs, Nixpkgs (nixpkgs
) und das selbst definierte Mini-Repo (mypkgs
). Wie bei den anderen Beispielen scheinen die Flake-Inputs als Argumente für die outputs
-Funktion übergeben zu werden. Über pkgs
erhalten wir Zugriff auf die mkShell
-Funktion und über mypkgs
haben wir Zugriff das das chord
-Paket. Um das Paket innerhalb des Flake zu identifizieren, muss der genaue Attributpfad angegeben werden (packages.${system}.chord
).
Einleitend heißt es: “Let us create a flake for an experiments repository that will create a shell with the chord
package available.” Ich fand es zunächst überraschend, das von einem Repository gesprochenn wird. In meinem Kopf war ein Repository im Kontext von Nix primär eine Quelle von Paketen.
Aber natürlich sind Flakes (unter anderem) genau das. Tatsächlich haben wir ja sogar gesagt, dass Flake-Verzeichnisse notwendig ein Git-Repo umfassen. Ja… ich weiß nicht, warum ich diesen Hinweis zunächst überraschend fand. Jedes Flake scheint demnach potenziell wie das nixpkgs
-Repo zu fungieren.
Es gibt einen neuen Unterbefehl, um die in einem Flake definieren Entwicklungsumgebungen zu betreten:
nix develop <Verzeichnis>#<Shell-Name>
Für das obige Beispiel wäre das: nix develop .#chordShell
.
Offene Fragen
- Welche Einstellungen können in der Nix-Konfigurationsdatei vorgenommen werden?
- Wie können wir eigene Flake-Templates definieren?
- Wie legen wir ein Default-Template für Flakes fest?
- Reicht es,
flake.nix
der Git Staging Area hinzuzufügen oder müssen weitere Flake-Dateien eingecheckt werden? - Was sind die Unterschiede und Gemeinsamkeiten zwischen
nix-build
undnix build
? - Unter welchen Bedingungen und zu welchen Zeitpunkten wird die
flake.lock
-Datei aktualisiert? - Was genau passiert bei einem Update der Flake-Inputs?
- Was sind
checks
(einer der Output-Typen)? - Welche weiteren Output-Typen gibt es (neben
packages
,devShells
undchecks
)? - Was ist die Beziehung zwischen Flake-Inputs, den möglichen Argumenten der Outputs-Funktion und den Werten von
import
?
-
Das stimmt vielleicht nicht ganz. Zuvor haben wir Dateien geschrieben, die eine Funktion definierten. Nun schreiben wir Dateien, die eine Attributmenge definieren. Dazu unten mehr. ↩︎
-
Ich denke prinzipiell kann jedes der drei Attribute ausgelassen werden. ↩︎
-
Das stimmt nicht. Flakes sind hierarchisch und auf der obersten Ebene nach Output-Typ gegliedert. Wir werden gleich sehen, dass Pakete (Plural) erst auf der dritten Ebene auftauchen. ↩︎