2017-10-22 16:01:59 +00:00
|
|
|
|
---
|
2024-12-28 12:06:33 +00:00
|
|
|
|
filename: learnnix.nix
|
2017-10-22 16:01:59 +00:00
|
|
|
|
contributors:
|
|
|
|
|
- ["Chris Martin", "http://chris-martin.org/"]
|
|
|
|
|
translators:
|
|
|
|
|
- ["Dennis Keller", "https://github.com/denniskeller"]
|
|
|
|
|
---
|
|
|
|
|
|
2019-10-23 13:34:13 +00:00
|
|
|
|
Nix ist eine simple funktionale Programmiersprache, die für den
|
2017-10-22 16:01:59 +00:00
|
|
|
|
[Nix package manager](https://nixos.org/nix/) und
|
|
|
|
|
[NixOS](https://nixos.org/) entwickelt wurde.
|
|
|
|
|
|
2019-10-23 13:34:13 +00:00
|
|
|
|
Du kannst Nix Ausdrücke evaluieren mithilfe von
|
2017-10-22 16:01:59 +00:00
|
|
|
|
[nix-instantiate](https://nixos.org/nix/manual/#sec-nix-instantiate)
|
|
|
|
|
oder [`nix-repl`](https://github.com/edolstra/nix-repl).
|
|
|
|
|
|
2024-04-07 10:41:09 +00:00
|
|
|
|
```nix
|
2017-10-22 16:01:59 +00:00
|
|
|
|
with builtins; [
|
|
|
|
|
|
|
|
|
|
# Kommentare
|
|
|
|
|
#=========================================
|
|
|
|
|
|
|
|
|
|
# Inline Kommentare sehen so aus.
|
|
|
|
|
|
2019-10-23 13:34:13 +00:00
|
|
|
|
/* Multizeilen Kommentare
|
2017-10-22 16:01:59 +00:00
|
|
|
|
sehen so aus. */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Booleans
|
|
|
|
|
#=========================================
|
|
|
|
|
|
|
|
|
|
(true && false) # Und
|
|
|
|
|
#=> false
|
|
|
|
|
|
|
|
|
|
(true || false) # Oder
|
|
|
|
|
#=> true
|
|
|
|
|
|
|
|
|
|
(if 3 < 4 then "a" else "b") # Bedingungen
|
|
|
|
|
#=> "a"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Integers
|
|
|
|
|
#=========================================
|
|
|
|
|
|
|
|
|
|
# Integers sind die einzigen numerischen Typen.
|
|
|
|
|
|
|
|
|
|
1 0 42 (-3) # Einige integers
|
|
|
|
|
|
|
|
|
|
(4 + 6 + 12 - 2) # Addition
|
|
|
|
|
#=> 20
|
|
|
|
|
|
|
|
|
|
(7 / 2) # Division
|
|
|
|
|
#=> 3
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Strings
|
|
|
|
|
#=========================================
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
"String Literale sind in Anführungszeichen."
|
2017-10-22 16:01:59 +00:00
|
|
|
|
|
|
|
|
|
"
|
2019-10-23 13:34:13 +00:00
|
|
|
|
String Literale können mehrere
|
2017-10-22 16:01:59 +00:00
|
|
|
|
Zeilen umspannen.
|
|
|
|
|
"
|
|
|
|
|
|
|
|
|
|
''
|
|
|
|
|
Dies wird als Literal mit eingerückten String bezeichnet.
|
|
|
|
|
Es entfernt intelligent führende Leerzeichen.
|
|
|
|
|
''
|
|
|
|
|
|
|
|
|
|
''
|
|
|
|
|
a
|
|
|
|
|
b
|
|
|
|
|
''
|
|
|
|
|
#=> "a\n b"
|
|
|
|
|
|
|
|
|
|
("ab" + "cd") # String Konkatenation
|
|
|
|
|
#=> "abcd"
|
|
|
|
|
|
|
|
|
|
# Mit Antiquotation kannst du Werte in Strings einbetten.
|
|
|
|
|
("Dein Homeverzeichnis ist ${getEnv "HOME"}")
|
|
|
|
|
#=> "Dein Homeverzeichnis ist /home/alice"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Paths
|
|
|
|
|
#=========================================
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
# Nix besitzt einen primitiven Datentyp für Pfade
|
2017-10-22 16:01:59 +00:00
|
|
|
|
/tmp/tutorials/learn.nix
|
|
|
|
|
|
|
|
|
|
# Ein relativer Pfad wird beim Parsing zu einem absoluten Pfad aufgelöst,
|
|
|
|
|
# relativ zu der Datei in der es auftritt.
|
|
|
|
|
tutorials/learn.nix
|
|
|
|
|
#=> /the-base-path/tutorials/learn.nix
|
|
|
|
|
|
2019-10-23 13:34:13 +00:00
|
|
|
|
# Ein Pfad muss mindestens einen Schrägstrich enthalten. Ein Pfad für eine
|
2017-10-22 16:01:59 +00:00
|
|
|
|
# Datei im selben Verzeichnis benötigt ein ./ Präfix.
|
|
|
|
|
./learn.nix
|
|
|
|
|
#=> /the-base-path/learn.nix
|
|
|
|
|
|
|
|
|
|
# Der / Operator muss von Leerraum umgeben sein wenn du dividieren möchtest.
|
|
|
|
|
7/2 # Das ist ein Pfadliteral
|
|
|
|
|
(7 / 2) # Das ist ein Integerliteral
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Importe
|
|
|
|
|
#=========================================
|
|
|
|
|
|
|
|
|
|
# Eine nix Datei besitzt einen einzelnen top-level Ausdruck mit keinen freien Variablen.
|
|
|
|
|
# Ein Import-Ausdruck wird zum Wert der Datei, die importiert wird, ausgewertet.
|
|
|
|
|
(import /tmp/foo.nix)
|
|
|
|
|
|
|
|
|
|
# Importe können ebenso mit Strings spezifiziert werden.
|
|
|
|
|
(import "/tmp/foo.nix")
|
|
|
|
|
|
|
|
|
|
# Import Pfade müssen absolut sein. Pfadliterale
|
|
|
|
|
# sind automatisch aufgelöst, das ist ein Ordnung.
|
|
|
|
|
(import ./foo.nix)
|
|
|
|
|
|
|
|
|
|
# Jedoch passiert dies nicht mit Strings.
|
|
|
|
|
(import "./foo.nix")
|
|
|
|
|
#=> error: string ‘foo.nix’ doesn't represent an absolute path
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Let
|
|
|
|
|
#=========================================
|
|
|
|
|
|
|
|
|
|
# `let` Blöcke erlauben es uns Werte zu Variablen zu binden.
|
|
|
|
|
(let x = "a"; in
|
|
|
|
|
x + x + x)
|
|
|
|
|
#=> "aaa"
|
|
|
|
|
|
|
|
|
|
# Bindungen können auf sich gegenseitig verweisen. Die Reihenfolge spielt
|
|
|
|
|
# keine Rolle.
|
|
|
|
|
(let y = x + "b";
|
|
|
|
|
x = "a"; in
|
|
|
|
|
y + "c")
|
|
|
|
|
#=> "abc"
|
|
|
|
|
|
|
|
|
|
# Innere Bindungen überschatten Äußere.
|
|
|
|
|
(let a = 1; in
|
|
|
|
|
let a = 2; in
|
|
|
|
|
a)
|
|
|
|
|
#=> 2
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Funktionen
|
|
|
|
|
#=========================================
|
|
|
|
|
|
|
|
|
|
(n: n + 1) # Funktion, die 1 addiert
|
|
|
|
|
|
|
|
|
|
((n: n + 1) 5) # Dieselbe Funktion angewendet auf 5.
|
|
|
|
|
#=> 6
|
|
|
|
|
|
|
|
|
|
# Es gibt keine spezielle Syntax für benannte Funktionen, aber sie
|
|
|
|
|
# können mit `let` Blöcken, wie jeder andere Wert auch, gebunden werden.
|
|
|
|
|
(let succ = (n: n + 1); in succ 5)
|
|
|
|
|
#=> 6
|
|
|
|
|
|
|
|
|
|
# Eine Funktion hat genau ein Argument.
|
|
|
|
|
# Mehrere Argumente können erreicht werden mithilfe von Currying.
|
|
|
|
|
((x: y: x + "-" + y) "a" "b")
|
|
|
|
|
#=> "a-b"
|
|
|
|
|
|
|
|
|
|
# Benannte Funktionsargumente gibt es auch. Diese werden wir einführen, nachdem wir uns Sets
|
|
|
|
|
# angeschaut haben.
|
|
|
|
|
|
|
|
|
|
# Listen
|
|
|
|
|
#=========================================
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
# Listen werden durch eckige Klammern gekennzeichnet.
|
2017-10-22 16:01:59 +00:00
|
|
|
|
|
|
|
|
|
(length [1 2 3 "x"])
|
|
|
|
|
#=> 4
|
|
|
|
|
|
|
|
|
|
([1 2 3] ++ [4 5])
|
|
|
|
|
#=> [1 2 3 4 5]
|
|
|
|
|
|
|
|
|
|
(concatLists [[1 2] [3 4] [5]])
|
|
|
|
|
#=> [1 2 3 4 5]
|
|
|
|
|
|
|
|
|
|
(head [1 2 3])
|
|
|
|
|
#=> 1
|
|
|
|
|
(tail [1 2 3])
|
|
|
|
|
#=> [2 3]
|
|
|
|
|
|
|
|
|
|
(elemAt ["a" "b" "c" "d"] 2)
|
|
|
|
|
#=> "c"
|
|
|
|
|
|
|
|
|
|
(elem 2 [1 2 3])
|
|
|
|
|
#=> true
|
|
|
|
|
(elem 5 [1 2 3])
|
|
|
|
|
#=> false
|
|
|
|
|
|
|
|
|
|
(filter (n: n < 3) [1 2 3 4])
|
|
|
|
|
#=> [ 1 2 ]
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Sets
|
|
|
|
|
#=========================================
|
|
|
|
|
|
|
|
|
|
# Ein "Set" ist eine ungeordnete Zuordnung mit Stringschlüsseln.
|
|
|
|
|
{ foo = [1 2]; bar = "x"; }
|
|
|
|
|
|
|
|
|
|
# Der . Operator nimmt einen Wert aus dem Set.
|
|
|
|
|
{ a = 1; b = 2; }.a
|
|
|
|
|
#=> 1
|
|
|
|
|
|
|
|
|
|
# Der ? Operator testet, ob der Schlüssel in dem Set vorhanden ist.
|
|
|
|
|
({ a = 1; b = 2; } ? a)
|
|
|
|
|
#=> true
|
|
|
|
|
({ a = 1; b = 2; } ? c)
|
|
|
|
|
#=> false
|
|
|
|
|
|
|
|
|
|
# Der // Operator mergt zwei Sets.
|
|
|
|
|
({ a = 1; } // { b = 2; })
|
|
|
|
|
#=> { a = 1; b = 2; }
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
# Werte auf der rechten Seite überschreiben die Werte auf der linken Seite.
|
2017-10-22 16:01:59 +00:00
|
|
|
|
({ a = 1; b = 2; } // { a = 3; c = 4; })
|
|
|
|
|
#=> { a = 3; b = 2; c = 4; }
|
|
|
|
|
|
2024-08-25 20:19:56 +00:00
|
|
|
|
# Das Schlüsselwort rec bezeichnet ein "rekursives Set", in dem sich Attribute
|
2017-10-22 16:01:59 +00:00
|
|
|
|
# aufeinander beziehen können.
|
|
|
|
|
(let a = 1; in { a = 2; b = a; }.b)
|
|
|
|
|
#=> 1
|
|
|
|
|
(let a = 1; in rec { a = 2; b = a; }.b)
|
|
|
|
|
#=> 2
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
# Verschachtelte Sets können stückweise definiert werden.
|
2017-10-22 16:01:59 +00:00
|
|
|
|
{
|
|
|
|
|
a.b = 1;
|
|
|
|
|
a.c.d = 2;
|
|
|
|
|
a.c.e = 3;
|
|
|
|
|
}.a.c
|
|
|
|
|
#=> { d = 2; e = 3; }
|
|
|
|
|
|
|
|
|
|
# Die Nachkommen eines Attributs können in diesem Feld nicht zugeordnet werden, wenn
|
2019-10-23 13:34:13 +00:00
|
|
|
|
# das Attribut selbst nicht zugewiesen wurde.
|
2017-10-22 16:01:59 +00:00
|
|
|
|
{
|
|
|
|
|
a = { b = 1; };
|
|
|
|
|
a.c = 2;
|
|
|
|
|
}
|
|
|
|
|
#=> error: attribute ‘a’ already defined
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# With
|
|
|
|
|
#=========================================
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
# Der Körper eines Sets Blocks wird mit der Zuordnung eines Satzes an die Variablen gebunden.
|
2017-10-22 16:01:59 +00:00
|
|
|
|
(with { a = 1; b = 2; };
|
|
|
|
|
a + b)
|
|
|
|
|
# => 3
|
|
|
|
|
|
|
|
|
|
# Innere Bindungen überschatten äußere Bindungen.
|
|
|
|
|
(with { a = 1; b = 2; };
|
|
|
|
|
(with { a = 5; };
|
|
|
|
|
a + b))
|
|
|
|
|
#=> 7
|
|
|
|
|
|
|
|
|
|
# Die erste Linie diese Tutorials startet mit "with builtins;",
|
2019-10-23 13:34:13 +00:00
|
|
|
|
# weil builtins ein Set mit allen eingebauten
|
2017-10-22 16:01:59 +00:00
|
|
|
|
# Funktionen (length, head, tail, filter, etc.) umfasst.
|
2019-10-23 13:34:13 +00:00
|
|
|
|
# Das erspart uns beispielsweise "builtins.length" zu schreiben,
|
2017-10-22 16:01:59 +00:00
|
|
|
|
# anstatt nur "length".
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Set patterns
|
|
|
|
|
#=========================================
|
|
|
|
|
|
|
|
|
|
# Sets sind nützlich, wenn du mehrere Werte einer Funktion
|
|
|
|
|
# übergeben musst.
|
|
|
|
|
(args: args.x + "-" + args.y) { x = "a"; y = "b"; }
|
|
|
|
|
#=> "a-b"
|
|
|
|
|
|
|
|
|
|
# Dies kann mit Hilfe von Set patterns deutlicher geschrieben werden.
|
|
|
|
|
({x, y}: x + "-" + y) { x = "a"; y = "b"; }
|
|
|
|
|
#=> "a-b"
|
|
|
|
|
|
|
|
|
|
# Standardmäßig schlägt das Muster bei Sets mit zusätzlichen Schlüsseln fehl.
|
|
|
|
|
({x, y}: x + "-" + y) { x = "a"; y = "b"; z = "c"; }
|
|
|
|
|
#=> error: anonymous function called with unexpected argument ‘z’
|
|
|
|
|
|
|
|
|
|
# Durch Hinzufügen von ", ..." können zusätzliche Schlüssel ignoriert werden.
|
|
|
|
|
({x, y, ...}: x + "-" + y) { x = "a"; y = "b"; z = "c"; }
|
|
|
|
|
#=> "a-b"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Errors
|
|
|
|
|
#=========================================
|
|
|
|
|
|
|
|
|
|
# `throw` bewirkt, dass die Auswertung mit einer Fehlermeldung abgebrochen wird.
|
|
|
|
|
(2 + (throw "foo"))
|
|
|
|
|
#=> error: foo
|
|
|
|
|
|
|
|
|
|
# `tryEval` fängt geworfene Fehler.
|
|
|
|
|
(tryEval 42)
|
|
|
|
|
#=> { success = true; value = 42; }
|
|
|
|
|
(tryEval (2 + (throw "foo")))
|
|
|
|
|
#=> { success = false; value = false; }
|
|
|
|
|
|
|
|
|
|
# `abort` ist ähnlich wie throw, aber es ist fatal. Es kann nicht gefangen werden.
|
|
|
|
|
(tryEval (abort "foo"))
|
|
|
|
|
#=> error: evaluation aborted with the following error message: ‘foo’
|
|
|
|
|
|
2019-10-23 13:34:13 +00:00
|
|
|
|
# `assert` evaluiert zu dem gegebenen Wert, wenn die Bedingung wahr ist, sonst
|
2017-10-22 16:01:59 +00:00
|
|
|
|
# löst es eine abfangbare Exception aus.
|
|
|
|
|
(assert 1 < 2; 42)
|
|
|
|
|
#=> 42
|
|
|
|
|
(assert 1 > 2; 42)
|
|
|
|
|
#=> error: assertion failed at (string):1:1
|
|
|
|
|
(tryEval (assert 1 > 2; 42))
|
|
|
|
|
#=> { success = false; value = false; }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# Impurity
|
|
|
|
|
#=========================================
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
# Da die Wiederholbarkeit von Builds für den Nix Packetmanager entscheidend ist,
|
2019-10-23 13:34:13 +00:00
|
|
|
|
# werden in der Nix Sprache reine funktionale Elemente betont. Es gibt aber ein paar
|
2017-10-22 16:01:59 +00:00
|
|
|
|
# unreine Elemente.
|
2017-10-30 04:43:23 +00:00
|
|
|
|
# Du kannst auf Umgebungsvariablen verweisen.
|
2017-10-22 16:01:59 +00:00
|
|
|
|
(getEnv "HOME")
|
|
|
|
|
#=> "/home/alice"
|
|
|
|
|
|
|
|
|
|
# Die trace Funktion wird zum Debugging verwendet. Sie gibt das erste Argument zu stderr aus
|
|
|
|
|
# und evaluiert das zweite Argument.
|
|
|
|
|
(trace 1 2)
|
|
|
|
|
#=> trace: 1
|
|
|
|
|
#=> 2
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
# Du kannst Dateien in den Nix Store schreiben. Obwohl unrein, kannst du dir relativ sicher sein,
|
2017-10-22 16:01:59 +00:00
|
|
|
|
# dass es sicher ist, da der Dateiname aus dem Hash des Inhalts abgeleitet wird.
|
|
|
|
|
# Du kannst Dateien von überall lesen. In diesem Beispiel schreiben wir Dateien in den Store
|
|
|
|
|
# und lesen wieder davon.
|
|
|
|
|
(let filename = toFile "foo.txt" "hello!"; in
|
|
|
|
|
[filename (builtins.readFile filename)])
|
|
|
|
|
#=> [ "/nix/store/ayh05aay2anx135prqp0cy34h891247x-foo.txt" "hello!" ]
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
# Außerdem können wir Dateien in den Nix Store herunterladen.
|
2017-10-22 16:01:59 +00:00
|
|
|
|
(fetchurl "https://example.com/package-1.2.3.tgz")
|
|
|
|
|
#=> "/nix/store/2drvlh8r57f19s9il42zg89rdr33m2rm-package-1.2.3.tgz"
|
|
|
|
|
|
|
|
|
|
]
|
|
|
|
|
```
|
|
|
|
|
|
2017-10-30 04:43:23 +00:00
|
|
|
|
### Weitere Ressourcen
|
2017-10-22 16:01:59 +00:00
|
|
|
|
|
2024-04-20 09:58:56 +00:00
|
|
|
|
* [Nix Manual - Nix expression language](https://nixos.org/nix/manual/#ch-expression-language)
|
|
|
|
|
* [James Fisher - Nix by example - Part 1: The Nix expression language](https://medium.com/@MrJamesFisher/nix-by-example-a0063a1a4c55)
|
|
|
|
|
* [Susan Potter - Nix Cookbook - Nix By Example](https://ops.functionalalgebra.com/nix-by-example/)
|
|
|
|
|
* [Zero to Nix - Nix Tutorial](https://zero-to-nix.com/)
|
|
|
|
|
* [Rommel Martinez - A Gentle Introduction to the Nix Family](https://web.archive.org/web/20210121042658/https://ebzzry.io/en/nix/#nix)
|