mirror of
https://github.com/ryan4yin/nixos-and-flakes-book.git
synced 2025-01-18 20:08:29 +01:00
172 lines
7.5 KiB
Markdown
172 lines
7.5 KiB
Markdown
# `flake.nix` 配置详解 {#flake-nix-configuration-explained}
|
||
|
||
上面我们创建了一个 `flake.nix` 文件并通过它来管理系统配置,但你对它的结构还是一头雾水,下
|
||
面我们来详细解释一下这个文件的内容。
|
||
|
||
### 1. flake inputs
|
||
|
||
首先看看其中的 `inputs` 属性,它是一个 attribute set,其中定义了这个 flake 的所有依赖项,
|
||
这些依赖项会在被拉取后,作为参数传递给 `outputs` 函数:
|
||
|
||
```nix{2-5,7}
|
||
{
|
||
inputs = {
|
||
# NixOS 官方软件源,这里使用 nixos-23.11 分支
|
||
nixpkgs.url = "github:NixOS/nixpkgs/nixos-23.11";
|
||
};
|
||
|
||
outputs = { self, nixpkgs, ... }@inputs: {
|
||
# 省略掉前面的配置......
|
||
};
|
||
}
|
||
```
|
||
|
||
`inputs` 中的每一项依赖有许多类型与定义方式,可以是另一个 flake,也可以是一个普通的 Git 仓
|
||
库,又或者一个本地路径。
|
||
[Flakes 的其他玩法 - Flake 的 inputs](../other-usage-of-flakes/inputs.md) 中详细介绍了常见
|
||
的依赖项类型与定义方式。
|
||
|
||
这里我们只定义了 `nixpkgs` 这一个依赖项,使用的是 flake 中最常见的引用方式,即
|
||
`github:owner/name/reference`,这里的 `reference` 可以是分支名、commit-id 或 tag。
|
||
|
||
`nixpkgs` 在 `inputs` 中被定义后,就可以在后面的 `outputs` 函数的参数中使用此依赖项中的内
|
||
容了,我们的示例中正是这么干的。
|
||
|
||
### 2. flake outputs
|
||
|
||
再来看看 `outputs`,它是一个以 `inputs` 中的依赖项为参数的函数,函数的返回值是一个
|
||
attribute set,这个返回的 attribute set 即为该 flake 的构建结果:
|
||
|
||
```nix{10-18}
|
||
{
|
||
description = "A simple NixOS flake";
|
||
|
||
inputs = {
|
||
# NixOS 官方软件源,这里使用 nixos-23.11 分支
|
||
nixpkgs.url = "github:NixOS/nixpkgs/nixos-23.11";
|
||
};
|
||
|
||
outputs = { self, nixpkgs, ... }@inputs: {
|
||
# hostname 为 my-nixos 的主机会使用这个配置
|
||
nixosConfigurations.my-nixos = nixpkgs.lib.nixosSystem {
|
||
system = "x86_64-linux";
|
||
modules = [
|
||
./configuration.nix
|
||
];
|
||
};
|
||
};
|
||
}
|
||
```
|
||
|
||
flake 有很多的用途,也可以有很多不同类型的
|
||
outputs,[Flake 的 outputs](../other-usage-of-flakes/outputs.md) 一节有更详细的介绍。这里
|
||
我们只用到了 `nixosConfigurations` 这一类型的 outputs,它用于配置 NixOS 系统。
|
||
|
||
在我们运行 `sudo nixos-rebuild switch` 命令时,它会从 `/etc/nixos/flake.nix` 的 `outputs`
|
||
函数返回值中查找 `nixosConfigurations.my-nixos` (其中的 `my-nixos` 将会是你当前系统的
|
||
hostname)这一属性,并使用其中的定义来配置你的 NixOS 系统。
|
||
|
||
实际我们也可以自定义 flake 的位置与 NixOS 配置的名称,而不是使用默认值。只需要在
|
||
`nixos-rebuild` 命令后面添加 `--flake` 参数即可,一个例子:
|
||
|
||
```nix
|
||
sudo nixos-rebuild switch --flake /path/to/your/flake#your-hostname
|
||
```
|
||
|
||
上述命令中的 `--flake /path/to/your/flake#your-hostname` 参数简要说明如下:
|
||
|
||
1. `/path/to/your/flake` 为目标 flake 的位置,默认会使用 `/etc/nixos/` 这个路径。
|
||
2. `#` 是一个分隔符,其后的 `your-hostname` 则是 NixOS 配置的名称。`nixos-rebuild` 默认会
|
||
以你当前系统的 hostname 为配置名称进行查找。
|
||
|
||
你甚至能直接引用一个远程的 GitHub 仓库作为你的 flake 来源,示例如下:
|
||
|
||
```nix
|
||
sudo nixos-rebuild switch --flake github:owner/repo#your-hostname
|
||
```
|
||
|
||
### 3. `outputs` 函数的特殊参数 `self` {#special-parameter-self-of-outputs-function}
|
||
|
||
虽然我们前面并未提到,但是前面的所有示例代码中,`outputs` 函数都还有一个特殊的参数
|
||
`self`,这里我们简单介绍一下它的作用。
|
||
|
||
[nix flake - Nix Manual] 对其的描述是:
|
||
|
||
> The special input named `self` refers to the outputs and source tree of this flake.
|
||
|
||
所以说 `self` 是当前 flake 的 `outputs` 函数的返回值,同时也是当前 flake 源码的文件夹路径
|
||
(source tree)。
|
||
|
||
这里我们并未使用到 `self` 这个参数,在后面一些更复杂的例子(或者你网上搜
|
||
到的一些配置)中,我们会看到 `self` 的用法。
|
||
|
||
> 注意:你可能会在一些代码中看到,有人会使用 `self.outputs` 来引用当前 flake 的输出,这
|
||
> 确实是可行的,但 Nix Manual 并未对其做任何说明,属于是 flake 的内部实现细节,不建议在
|
||
> 你自己的代码中使用!
|
||
|
||
### 4. `nixpkgs.lib.nixosSystem` 函数的简单介绍 {#simple-introduction-to-nixpkgs-lib-nixos-system}
|
||
|
||
**一个 Flake 可以依赖其他 Flakes,从而使用它们提供的功能**。
|
||
|
||
默认情况下,一个 flake 会在其每个依赖项(即 `inputs` 中的每一项)的根目录下寻找
|
||
`flake.nix` 文件并**懒惰求值**(lazy evaluation)它们的 `outputs` 函数,接着将这些函数返回
|
||
的 attribute sets 作为参数传递给它自身的 `outputs` 函数,这样我们就能在当前 flake 中使用它
|
||
所依赖的其他 flakes 提供的功能了。
|
||
|
||
更精确地说,对每个依赖项的 `outputs` 函数的求值都是懒惰(lazy)的,也就是说,一个 flake 的
|
||
`outputs` 函数只有在被真正使用到的时候才会被求值,这样就能避免不必要的计算,从而提高效率。
|
||
|
||
上面的描述可能有点绕,我们还是结合本节中使用的 `flake.nix` 示例来看看这个过程。我们的
|
||
`flake.nix` 声明了 `inputs.nixpkgs` 这个依赖项,因此 [nixpkgs/flake.nix] 会在我们执行
|
||
`sudo nixos-rebuild switch` 这个命令时被求值。从 Nixpkgs 仓库的源码中能看到它的 flake
|
||
outputs 定义中有返回 `lib` 这个属性,我们的例子中就使用了 `lib` 属性中的 `nixosSystem` 这
|
||
个函数来配置我们的 NixOS 系统:
|
||
|
||
```nix{8-13}
|
||
{
|
||
inputs = {
|
||
# NixOS 官方软件源,这里使用 nixos-23.11 分支
|
||
nixpkgs.url = "github:NixOS/nixpkgs/nixos-23.11";
|
||
};
|
||
|
||
outputs = { self, nixpkgs, ... }@inputs: {
|
||
nixosConfigurations.my-nixos = nixpkgs.lib.nixosSystem {
|
||
system = "x86_64-linux";
|
||
modules = [
|
||
./configuration.nix
|
||
];
|
||
};
|
||
};
|
||
}
|
||
```
|
||
|
||
`nixpkgs.lib.nixosSystem` 后面跟的 attribute set 就是该函数的参数,我们这里只设置了两个参
|
||
数:
|
||
|
||
1. `system`: 这个很好懂,就是系统架构参数。
|
||
2. `modules`: 此函数是一个 modules 的列表,NixOS 的实际系统配置都定义在这些 modules 中。
|
||
|
||
`/etc/nixos/configuration.nix` 这个配置文件本身就是一个 Nixpkgs Module,因此可以直接将其添
|
||
加到 `modules` 列表中使用。
|
||
|
||
新手阶段了解这些就足够了,探究 `nixpkgs.lib.nixosSystem` 函数的具体实现需要对 Nixpkgs 的模
|
||
块系统有一定的了解。读者可以在学习了
|
||
[模块化 NixOS 配置](./modularize-the-configuration.md) 一节后,再回过头来从
|
||
[nixpkgs/flake.nix] 中找到 `nixpkgs.lib.nixosSystem` 的定义,跟踪它的源码,研究其实现方
|
||
式。
|
||
|
||
|
||
[nix flake - Nix Manual]:
|
||
https://nixos.org/manual/nix/stable/command-ref/new-cli/nix3-flake#flake-inputs
|
||
[nixpkgs/flake.nix]: https://github.com/NixOS/nixpkgs/tree/nixos-23.11/flake.nix
|
||
[nixpkgs/nixos/lib/eval-config.nix]:
|
||
https://github.com/NixOS/nixpkgs/tree/nixos-23.11/nixos/lib/eval-config.nix
|
||
[Module System - Nixpkgs]:
|
||
https://github.com/NixOS/nixpkgs/blob/23.11/doc/module-system/module-system.chapter.md
|
||
[nixpkgs/nixos-23.11/lib/modules.nix - _module.args]:
|
||
https://github.com/NixOS/nixpkgs/blob/nixos-23.11/lib/modules.nix#L122-L184
|
||
[nixpkgs/nixos-23.11/nixos/doc/manual/development/option-types.section.md#L237-L244]:
|
||
https://github.com/NixOS/nixpkgs/blob/nixos-23.11/nixos/doc/manual/development/option-types.section.md?plain=1#L237-L244
|
||
|
||
|