2018-11-11 08:41:11 +00:00
|
|
|
{ system ? builtins.currentSystem,
|
|
|
|
config ? {},
|
2022-04-17 22:24:13 +01:00
|
|
|
pkgs ? import ../.. { inherit system config; },
|
|
|
|
systemdStage1 ? false
|
2018-11-11 08:41:11 +00:00
|
|
|
}:
|
2010-01-06 13:36:21 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with import ../lib/testing-python.nix { inherit system pkgs; };
|
2014-04-14 13:02:44 +01:00
|
|
|
with pkgs.lib;
|
2010-01-10 01:26:01 +00:00
|
|
|
|
2010-01-07 13:19:38 +00:00
|
|
|
let
|
2010-01-06 13:36:21 +00:00
|
|
|
|
2010-01-06 20:52:05 +00:00
|
|
|
# The configuration to install.
|
2022-04-04 17:22:03 +01:00
|
|
|
makeConfig = { bootLoader, grubDevice, grubIdentifier, grubUseEfi
|
2023-07-07 17:53:19 +01:00
|
|
|
, extraConfig, forceGrubReinstallCount ? 0, flake ? false
|
2023-11-18 19:38:08 +00:00
|
|
|
, clevisTest
|
2015-05-21 11:37:14 +01:00
|
|
|
}:
|
2014-08-31 17:18:13 +01:00
|
|
|
pkgs.writeText "configuration.nix" ''
|
2014-12-05 17:19:27 +00:00
|
|
|
{ config, lib, pkgs, modulesPath, ... }:
|
2010-01-06 20:52:05 +00:00
|
|
|
|
2013-09-04 12:05:09 +01:00
|
|
|
{ imports =
|
2013-10-13 16:24:11 +01:00
|
|
|
[ ./hardware-configuration.nix
|
2023-07-07 17:53:19 +01:00
|
|
|
${if flake
|
|
|
|
then "" # Still included, but via installer/flake.nix
|
|
|
|
else "<nixpkgs/nixos/modules/testing/test-instrumentation.nix>"}
|
2010-01-06 20:52:05 +00:00
|
|
|
];
|
|
|
|
|
2023-07-07 15:41:40 +01:00
|
|
|
networking.hostName = "thatworked";
|
|
|
|
|
2023-01-15 15:06:07 +00:00
|
|
|
documentation.enable = false;
|
|
|
|
|
2017-08-23 23:54:31 +01:00
|
|
|
# To ensure that we can rebuild the grub configuration on the nixos-rebuild
|
|
|
|
system.extraDependencies = with pkgs; [ stdenvNoCC ];
|
|
|
|
|
2022-04-17 22:24:13 +01:00
|
|
|
${optionalString systemdStage1 "boot.initrd.systemd.enable = true;"}
|
|
|
|
|
2016-04-23 12:20:34 +01:00
|
|
|
${optionalString (bootLoader == "grub") ''
|
2020-04-23 01:30:19 +01:00
|
|
|
boot.loader.grub.extraConfig = "serial; terminal_output serial";
|
2017-02-15 22:59:40 +00:00
|
|
|
${if grubUseEfi then ''
|
|
|
|
boot.loader.grub.device = "nodev";
|
|
|
|
boot.loader.grub.efiSupport = true;
|
|
|
|
boot.loader.grub.efiInstallAsRemovable = true; # XXX: needed for OVMF?
|
|
|
|
'' else ''
|
|
|
|
boot.loader.grub.device = "${grubDevice}";
|
|
|
|
boot.loader.grub.fsIdentifier = "${grubIdentifier}";
|
|
|
|
''}
|
2016-04-23 12:20:34 +01:00
|
|
|
|
|
|
|
boot.loader.grub.configurationLimit = 100 + ${toString forceGrubReinstallCount};
|
2012-04-11 10:42:53 +01:00
|
|
|
''}
|
2010-01-10 01:26:01 +00:00
|
|
|
|
2016-06-01 11:51:33 +01:00
|
|
|
${optionalString (bootLoader == "systemd-boot") ''
|
|
|
|
boot.loader.systemd-boot.enable = true;
|
2016-04-23 12:20:34 +01:00
|
|
|
''}
|
2014-09-04 18:30:25 +01:00
|
|
|
|
2023-01-19 19:50:59 +00:00
|
|
|
boot.initrd.secrets."/etc/secret" = ./secret;
|
2023-01-01 16:20:37 +00:00
|
|
|
|
2023-11-18 19:38:08 +00:00
|
|
|
${optionalString clevisTest ''
|
|
|
|
boot.kernelParams = [ "console=tty0" "ip=192.168.1.1:::255.255.255.0::eth1:none" ];
|
|
|
|
boot.initrd = {
|
|
|
|
availableKernelModules = [ "tpm_tis" ];
|
|
|
|
clevis = { enable = true; useTang = true; };
|
|
|
|
network.enable = true;
|
|
|
|
};
|
|
|
|
''}
|
|
|
|
|
2018-06-30 00:55:42 +01:00
|
|
|
users.users.alice = {
|
2017-02-20 19:57:16 +00:00
|
|
|
isNormalUser = true;
|
|
|
|
home = "/home/alice";
|
|
|
|
description = "Alice Foobar";
|
|
|
|
};
|
|
|
|
|
2015-06-10 12:14:40 +01:00
|
|
|
hardware.enableAllFirmware = lib.mkForce false;
|
2014-12-09 12:27:00 +00:00
|
|
|
|
2022-12-12 01:36:03 +00:00
|
|
|
${replaceStrings ["\n"] ["\n "] extraConfig}
|
2010-01-06 20:52:05 +00:00
|
|
|
}
|
|
|
|
'';
|
|
|
|
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2015-06-10 12:14:40 +01:00
|
|
|
# The test script boots a NixOS VM, installs NixOS on an empty hard
|
2010-01-07 14:19:19 +00:00
|
|
|
# disk, and then reboot from the hard disk. It's parameterized with
|
|
|
|
# a test script fragment `createPartitions', which must create
|
2014-04-30 08:40:18 +01:00
|
|
|
# partitions and filesystems.
|
2023-10-23 19:20:10 +01:00
|
|
|
testScriptFun = { bootLoader, createPartitions, grubDevice, grubUseEfi, grubIdentifier
|
|
|
|
, postInstallCommands, preBootCommands, postBootCommands, extraConfig
|
2023-11-18 19:38:08 +00:00
|
|
|
, testSpecialisationConfig, testFlakeSwitch, clevisTest, clevisFallbackTest
|
2015-05-21 11:37:14 +01:00
|
|
|
}:
|
2022-04-04 17:22:03 +01:00
|
|
|
let iface = "virtio";
|
2020-01-28 07:54:05 +00:00
|
|
|
isEfi = bootLoader == "systemd-boot" || (bootLoader == "grub" && grubUseEfi);
|
|
|
|
bios = if pkgs.stdenv.isAarch64 then "QEMU_EFI.fd" else "OVMF.fd";
|
2023-02-14 04:12:11 +00:00
|
|
|
in if !isEfi && !pkgs.stdenv.hostPlatform.isx86 then ''
|
|
|
|
machine.succeed("true")
|
|
|
|
'' else ''
|
2023-11-18 19:38:08 +00:00
|
|
|
import subprocess
|
|
|
|
tpm_folder = os.environ['NIX_BUILD_TOP']
|
2020-01-28 07:54:05 +00:00
|
|
|
def assemble_qemu_flags():
|
2020-08-22 05:33:59 +01:00
|
|
|
flags = "-cpu max"
|
2021-05-03 02:44:52 +01:00
|
|
|
${if (system == "x86_64-linux" || system == "i686-linux")
|
2021-11-24 00:33:53 +00:00
|
|
|
then ''flags += " -m 1024"''
|
|
|
|
else ''flags += " -m 768 -enable-kvm -machine virt,gic-version=host"''
|
2020-01-28 07:54:05 +00:00
|
|
|
}
|
2023-11-18 19:38:08 +00:00
|
|
|
${optionalString clevisTest ''flags += f" -chardev socket,id=chrtpm,path={tpm_folder}/swtpm-sock -tpmdev emulator,id=tpm0,chardev=chrtpm -device tpm-tis,tpmdev=tpm0"''}
|
|
|
|
${optionalString clevisTest ''flags += " -device virtio-net-pci,netdev=vlan1,mac=52:54:00:12:11:02 -netdev vde,id=vlan1,sock=\"$QEMU_VDE_SOCKET_1\""''}
|
2020-01-28 07:54:05 +00:00
|
|
|
return flags
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2010-01-06 13:36:21 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
qemu_flags = {"qemuFlags": assemble_qemu_flags()}
|
2010-01-06 20:52:05 +00:00
|
|
|
|
2023-07-07 16:32:03 +01:00
|
|
|
import os
|
|
|
|
|
|
|
|
image_dir = machine.state_dir
|
|
|
|
disk_image = os.path.join(image_dir, "machine.qcow2")
|
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
hd_flags = {
|
|
|
|
"hdaInterface": "${iface}",
|
2023-07-07 16:32:03 +01:00
|
|
|
"hda": disk_image,
|
2020-01-28 07:54:05 +00:00
|
|
|
}
|
|
|
|
${optionalString isEfi ''
|
|
|
|
hd_flags.update(
|
|
|
|
bios="${pkgs.OVMF.fd}/FV/${bios}"
|
|
|
|
)''
|
|
|
|
}
|
|
|
|
default_flags = {**hd_flags, **qemu_flags}
|
2014-05-30 20:22:25 +01:00
|
|
|
|
2010-01-06 14:37:23 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
def create_machine_named(name):
|
2020-05-11 23:41:18 +01:00
|
|
|
return create_machine({**default_flags, "name": name})
|
2010-01-06 16:46:21 +00:00
|
|
|
|
2023-11-18 19:38:08 +00:00
|
|
|
class Tpm:
|
|
|
|
def __init__(self):
|
|
|
|
self.start()
|
|
|
|
|
|
|
|
def start(self):
|
|
|
|
self.proc = subprocess.Popen(["${pkgs.swtpm}/bin/swtpm",
|
|
|
|
"socket",
|
|
|
|
"--tpmstate", f"dir={tpm_folder}/swtpm",
|
|
|
|
"--ctrl", f"type=unixio,path={tpm_folder}/swtpm-sock",
|
|
|
|
"--tpm2"
|
|
|
|
])
|
|
|
|
|
|
|
|
# Check whether starting swtpm failed
|
|
|
|
try:
|
|
|
|
exit_code = self.proc.wait(timeout=0.2)
|
|
|
|
if exit_code is not None and exit_code != 0:
|
|
|
|
raise Exception("failed to start swtpm")
|
|
|
|
except subprocess.TimeoutExpired:
|
|
|
|
pass
|
|
|
|
|
|
|
|
"""Check whether the swtpm process exited due to an error"""
|
|
|
|
def check(self):
|
|
|
|
exit_code = self.proc.poll()
|
|
|
|
if exit_code is not None and exit_code != 0:
|
|
|
|
raise Exception("swtpm process died")
|
|
|
|
|
|
|
|
|
|
|
|
os.mkdir(f"{tpm_folder}/swtpm")
|
|
|
|
tpm = Tpm()
|
|
|
|
tpm.check()
|
|
|
|
|
|
|
|
start_all()
|
|
|
|
${optionalString clevisTest ''
|
|
|
|
tang.wait_for_unit("sockets.target")
|
|
|
|
tang.wait_for_unit("network-online.target")
|
|
|
|
machine.wait_for_unit("network-online.target")
|
|
|
|
''}
|
|
|
|
machine.wait_for_unit("multi-user.target")
|
2010-01-06 20:52:05 +00:00
|
|
|
|
2010-01-06 16:46:21 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("Assert readiness of login prompt"):
|
|
|
|
machine.succeed("echo hello")
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("Wait for hard disks to appear in /dev"):
|
|
|
|
machine.succeed("udevadm settle")
|
2012-05-25 00:39:03 +01:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
${createPartitions}
|
2014-04-30 13:51:27 +01:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("Create the NixOS configuration"):
|
|
|
|
machine.succeed("nixos-generate-config --root /mnt")
|
|
|
|
machine.succeed("cat /mnt/etc/nixos/hardware-configuration.nix >&2")
|
|
|
|
machine.copy_from_host(
|
|
|
|
"${ makeConfig {
|
2022-04-04 17:22:03 +01:00
|
|
|
inherit bootLoader grubDevice grubIdentifier
|
2023-11-18 19:38:08 +00:00
|
|
|
grubUseEfi extraConfig clevisTest;
|
2020-01-28 07:54:05 +00:00
|
|
|
}
|
|
|
|
}",
|
|
|
|
"/mnt/etc/nixos/configuration.nix",
|
|
|
|
)
|
2023-01-01 16:20:37 +00:00
|
|
|
machine.copy_from_host("${pkgs.writeText "secret" "secret"}", "/mnt/etc/nixos/secret")
|
2020-01-28 07:54:05 +00:00
|
|
|
|
2023-11-18 19:38:08 +00:00
|
|
|
${optionalString clevisTest ''
|
|
|
|
with subtest("Create the Clevis secret with Tang"):
|
|
|
|
machine.wait_for_unit("network-online.target")
|
|
|
|
machine.succeed('echo -n password | clevis encrypt sss \'{"t": 2, "pins": {"tpm2": {}, "tang": {"url": "http://192.168.1.2"}}}\' -y > /mnt/etc/nixos/clevis-secret.jwe')''}
|
|
|
|
|
|
|
|
${optionalString clevisFallbackTest ''
|
|
|
|
with subtest("Shutdown Tang to check fallback to interactive prompt"):
|
|
|
|
tang.shutdown()
|
|
|
|
''}
|
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("Perform the installation"):
|
|
|
|
machine.succeed("nixos-install < /dev/null >&2")
|
|
|
|
|
|
|
|
with subtest("Do it again to make sure it's idempotent"):
|
|
|
|
machine.succeed("nixos-install < /dev/null >&2")
|
|
|
|
|
2023-01-01 16:20:37 +00:00
|
|
|
with subtest("Check that we can build things in nixos-enter"):
|
|
|
|
machine.succeed(
|
|
|
|
"""
|
|
|
|
nixos-enter -- nix-build --option substitute false -E 'derivation {
|
|
|
|
name = "t";
|
|
|
|
builder = "/bin/sh";
|
|
|
|
args = ["-c" "echo nixos-enter build > $out"];
|
|
|
|
system = builtins.currentSystem;
|
|
|
|
preferLocalBuild = true;
|
|
|
|
}'
|
|
|
|
"""
|
|
|
|
)
|
|
|
|
|
2023-10-23 19:20:10 +01:00
|
|
|
${postInstallCommands}
|
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("Shutdown system after installation"):
|
2023-01-15 02:25:17 +00:00
|
|
|
machine.succeed("umount -R /mnt")
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.succeed("sync")
|
|
|
|
machine.shutdown()
|
2010-01-06 15:14:26 +00:00
|
|
|
|
|
|
|
# Now see if we can boot the installation.
|
2020-01-28 07:54:05 +00:00
|
|
|
machine = create_machine_named("boot-after-install")
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2015-06-10 12:14:40 +01:00
|
|
|
# For example to enter LUKS passphrase.
|
2015-05-21 11:40:07 +01:00
|
|
|
${preBootCommands}
|
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("Assert that /boot get mounted"):
|
|
|
|
machine.wait_for_unit("local-fs.target")
|
|
|
|
${if bootLoader == "grub"
|
|
|
|
then ''machine.succeed("test -e /boot/grub")''
|
|
|
|
else ''machine.succeed("test -e /boot/loader/loader.conf")''
|
|
|
|
}
|
|
|
|
|
|
|
|
with subtest("Check whether /root has correct permissions"):
|
|
|
|
assert "700" in machine.succeed("stat -c '%a' /root")
|
|
|
|
|
|
|
|
with subtest("Assert swap device got activated"):
|
|
|
|
# uncomment once https://bugs.freedesktop.org/show_bug.cgi?id=86930 is resolved
|
|
|
|
machine.wait_for_unit("swap.target")
|
|
|
|
machine.succeed("cat /proc/swaps | grep -q /dev")
|
|
|
|
|
|
|
|
with subtest("Check that the store is in good shape"):
|
|
|
|
machine.succeed("nix-store --verify --check-contents >&2")
|
|
|
|
|
|
|
|
with subtest("Check whether the channel works"):
|
|
|
|
machine.succeed("nix-env -iA nixos.procps >&2")
|
|
|
|
assert ".nix-profile" in machine.succeed("type -tP ps | tee /dev/stderr")
|
|
|
|
|
|
|
|
with subtest(
|
|
|
|
"Check that the daemon works, and that non-root users can run builds "
|
|
|
|
"(this will build a new profile generation through the daemon)"
|
|
|
|
):
|
|
|
|
machine.succeed("su alice -l -c 'nix-env -iA nixos.procps' >&2")
|
|
|
|
|
|
|
|
with subtest("Configure system with writable Nix store on next boot"):
|
|
|
|
# we're not using copy_from_host here because the installer image
|
|
|
|
# doesn't know about the host-guest sharing mechanism.
|
|
|
|
machine.copy_from_host_via_shell(
|
|
|
|
"${ makeConfig {
|
2022-04-04 17:22:03 +01:00
|
|
|
inherit bootLoader grubDevice grubIdentifier
|
2023-11-18 19:38:08 +00:00
|
|
|
grubUseEfi extraConfig clevisTest;
|
2020-01-28 07:54:05 +00:00
|
|
|
forceGrubReinstallCount = 1;
|
|
|
|
}
|
|
|
|
}",
|
|
|
|
"/etc/nixos/configuration.nix",
|
|
|
|
)
|
|
|
|
|
|
|
|
with subtest("Check whether nixos-rebuild works"):
|
|
|
|
machine.succeed("nixos-rebuild switch >&2")
|
|
|
|
|
2021-11-10 23:57:18 +00:00
|
|
|
# FIXME: Nix 2.4 broke nixos-option, someone has to fix it.
|
|
|
|
# with subtest("Test nixos-option"):
|
|
|
|
# kernel_modules = machine.succeed("nixos-option boot.initrd.kernelModules")
|
|
|
|
# assert "virtio_console" in kernel_modules
|
|
|
|
# assert "List of modules" in kernel_modules
|
|
|
|
# assert "qemu-guest.nix" in kernel_modules
|
2020-01-28 07:54:05 +00:00
|
|
|
|
|
|
|
machine.shutdown()
|
2010-01-06 22:53:27 +00:00
|
|
|
|
2014-09-04 18:30:25 +01:00
|
|
|
# Check whether a writable store build works
|
2020-01-28 07:54:05 +00:00
|
|
|
machine = create_machine_named("rebuild-switch")
|
2015-05-21 11:40:07 +01:00
|
|
|
${preBootCommands}
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.wait_for_unit("multi-user.target")
|
|
|
|
|
|
|
|
# we're not using copy_from_host here because the installer image
|
|
|
|
# doesn't know about the host-guest sharing mechanism.
|
|
|
|
machine.copy_from_host_via_shell(
|
|
|
|
"${ makeConfig {
|
2022-04-04 17:22:03 +01:00
|
|
|
inherit bootLoader grubDevice grubIdentifier
|
2023-11-18 19:38:08 +00:00
|
|
|
grubUseEfi extraConfig clevisTest;
|
2020-01-28 07:54:05 +00:00
|
|
|
forceGrubReinstallCount = 2;
|
|
|
|
}
|
|
|
|
}",
|
|
|
|
"/etc/nixos/configuration.nix",
|
|
|
|
)
|
|
|
|
machine.succeed("nixos-rebuild boot >&2")
|
|
|
|
machine.shutdown()
|
2014-09-04 18:30:25 +01:00
|
|
|
|
2010-01-06 22:53:27 +00:00
|
|
|
# And just to be sure, check that the machine still boots after
|
|
|
|
# "nixos-rebuild switch".
|
2020-01-28 07:54:05 +00:00
|
|
|
machine = create_machine_named("boot-after-rebuild-switch")
|
2015-05-21 11:40:07 +01:00
|
|
|
${preBootCommands}
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.wait_for_unit("network.target")
|
2023-07-07 15:41:40 +01:00
|
|
|
|
|
|
|
# Sanity check, is it the configuration.nix we generated?
|
|
|
|
hostname = machine.succeed("hostname").strip()
|
|
|
|
assert hostname == "thatworked"
|
|
|
|
|
2020-05-25 03:05:55 +01:00
|
|
|
${postBootCommands}
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.shutdown()
|
2019-03-07 09:38:28 +00:00
|
|
|
|
|
|
|
# Tests for validating clone configuration entries in grub menu
|
2020-01-28 07:54:05 +00:00
|
|
|
''
|
2020-03-05 22:07:20 +00:00
|
|
|
+ optionalString testSpecialisationConfig ''
|
2020-01-28 07:54:05 +00:00
|
|
|
# Reboot Machine
|
|
|
|
machine = create_machine_named("clone-default-config")
|
|
|
|
${preBootCommands}
|
|
|
|
machine.wait_for_unit("multi-user.target")
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("Booted configuration name should be 'Home'"):
|
|
|
|
# This is not the name that shows in the grub menu.
|
|
|
|
# The default configuration is always shown as "Default"
|
|
|
|
machine.succeed("cat /run/booted-system/configuration-name >&2")
|
|
|
|
assert "Home" in machine.succeed("cat /run/booted-system/configuration-name")
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("We should **not** find a file named /etc/gitconfig"):
|
|
|
|
machine.fail("test -e /etc/gitconfig")
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("Set grub to boot the second configuration"):
|
|
|
|
machine.succeed("grub-reboot 1")
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2020-05-25 03:05:55 +01:00
|
|
|
${postBootCommands}
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.shutdown()
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
# Reboot Machine
|
|
|
|
machine = create_machine_named("clone-alternate-config")
|
|
|
|
${preBootCommands}
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.wait_for_unit("multi-user.target")
|
|
|
|
with subtest("Booted configuration name should be Work"):
|
|
|
|
machine.succeed("cat /run/booted-system/configuration-name >&2")
|
|
|
|
assert "Work" in machine.succeed("cat /run/booted-system/configuration-name")
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
with subtest("We should find a file named /etc/gitconfig"):
|
|
|
|
machine.succeed("test -e /etc/gitconfig")
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2020-05-25 03:05:55 +01:00
|
|
|
${postBootCommands}
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.shutdown()
|
2023-07-07 17:53:19 +01:00
|
|
|
''
|
|
|
|
+ optionalString testFlakeSwitch ''
|
|
|
|
${preBootCommands}
|
|
|
|
machine.start()
|
|
|
|
|
|
|
|
with subtest("Configure system with flake"):
|
|
|
|
# TODO: evaluate as user?
|
|
|
|
machine.succeed("""
|
|
|
|
mkdir /root/my-config
|
|
|
|
mv /etc/nixos/hardware-configuration.nix /root/my-config/
|
|
|
|
mv /etc/nixos/secret /root/my-config/
|
|
|
|
rm /etc/nixos/configuration.nix
|
|
|
|
""")
|
|
|
|
machine.copy_from_host_via_shell(
|
2023-07-08 19:49:37 +01:00
|
|
|
"${makeConfig {
|
2023-11-18 19:38:08 +00:00
|
|
|
inherit bootLoader grubDevice grubIdentifier grubUseEfi extraConfig clevisTest;
|
2023-07-08 19:49:37 +01:00
|
|
|
forceGrubReinstallCount = 1;
|
|
|
|
flake = true;
|
|
|
|
}}",
|
2023-07-07 17:53:19 +01:00
|
|
|
"/root/my-config/configuration.nix",
|
|
|
|
)
|
|
|
|
machine.copy_from_host_via_shell(
|
|
|
|
"${./installer/flake.nix}",
|
|
|
|
"/root/my-config/flake.nix",
|
|
|
|
)
|
|
|
|
machine.succeed("""
|
|
|
|
# for some reason the image does not have `pkgs.path`, so
|
|
|
|
# we use readlink to find a Nixpkgs source.
|
|
|
|
pkgs=$(readlink -f /nix/var/nix/profiles/per-user/root/channels)/nixos
|
|
|
|
if ! [[ -e $pkgs/pkgs/top-level/default.nix ]]; then
|
|
|
|
echo 1>&2 "$pkgs does not seem to be a nixpkgs source. Please fix the test so that pkgs points to a nixpkgs source.";
|
|
|
|
exit 1;
|
|
|
|
fi
|
|
|
|
sed -e s^@nixpkgs@^$pkgs^ -i /root/my-config/flake.nix
|
|
|
|
""")
|
|
|
|
|
|
|
|
with subtest("Switch to flake based config"):
|
|
|
|
machine.succeed("nixos-rebuild switch --flake /root/my-config#xyz")
|
|
|
|
|
|
|
|
${postBootCommands}
|
|
|
|
machine.shutdown()
|
|
|
|
|
|
|
|
${preBootCommands}
|
|
|
|
machine.start()
|
|
|
|
|
|
|
|
machine.wait_for_unit("multi-user.target")
|
|
|
|
|
|
|
|
with subtest("nix-channel command is not available anymore"):
|
|
|
|
machine.succeed("! which nix-channel")
|
|
|
|
|
2023-07-07 22:12:39 +01:00
|
|
|
# Note that the channel profile is still present on disk, but configured
|
|
|
|
# not to be used.
|
|
|
|
with subtest("builtins.nixPath is now empty"):
|
|
|
|
machine.succeed("""
|
|
|
|
[[ "[ ]" == "$(nix-instantiate builtins.nixPath --eval --expr)" ]]
|
|
|
|
""")
|
|
|
|
|
|
|
|
with subtest("<nixpkgs> does not resolve"):
|
|
|
|
machine.succeed("""
|
|
|
|
! nix-instantiate '<nixpkgs>' --eval --expr
|
|
|
|
""")
|
|
|
|
|
2023-07-07 17:53:19 +01:00
|
|
|
with subtest("Evaluate flake config in fresh env without nix-channel"):
|
|
|
|
machine.succeed("nixos-rebuild switch --flake /root/my-config#xyz")
|
|
|
|
|
|
|
|
with subtest("Evaluate flake config in fresh env without channel profiles"):
|
|
|
|
machine.succeed("""
|
|
|
|
(
|
|
|
|
exec 1>&2
|
|
|
|
rm -v /root/.nix-channels
|
|
|
|
rm -vrf ~/.nix-defexpr
|
|
|
|
rm -vrf /nix/var/nix/profiles/per-user/root/channels*
|
|
|
|
)
|
|
|
|
""")
|
|
|
|
machine.succeed("nixos-rebuild switch --flake /root/my-config#xyz")
|
|
|
|
|
|
|
|
${postBootCommands}
|
|
|
|
machine.shutdown()
|
2010-01-06 15:14:26 +00:00
|
|
|
'';
|
2010-01-06 16:46:21 +00:00
|
|
|
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2014-06-28 15:04:49 +01:00
|
|
|
makeInstallerTest = name:
|
2023-10-23 19:20:10 +01:00
|
|
|
{ createPartitions
|
|
|
|
, postInstallCommands ? "", preBootCommands ? "", postBootCommands ? ""
|
|
|
|
, extraConfig ? ""
|
2017-06-28 00:35:55 +01:00
|
|
|
, extraInstallerConfig ? {}
|
2016-06-01 11:51:33 +01:00
|
|
|
, bootLoader ? "grub" # either "grub" or "systemd-boot"
|
2022-04-04 17:22:03 +01:00
|
|
|
, grubDevice ? "/dev/vda", grubIdentifier ? "uuid", grubUseEfi ? false
|
2016-04-23 12:20:34 +01:00
|
|
|
, enableOCR ? false, meta ? {}
|
2020-03-05 22:07:20 +00:00
|
|
|
, testSpecialisationConfig ? false
|
2023-07-07 17:53:19 +01:00
|
|
|
, testFlakeSwitch ? false
|
2023-11-18 19:38:08 +00:00
|
|
|
, clevisTest ? false
|
|
|
|
, clevisFallbackTest ? false
|
2015-05-21 11:37:14 +01:00
|
|
|
}:
|
2014-04-14 15:24:08 +01:00
|
|
|
makeTest {
|
2015-05-22 06:14:00 +01:00
|
|
|
inherit enableOCR;
|
2015-06-10 12:14:40 +01:00
|
|
|
name = "installer-" + name;
|
2023-06-19 17:31:06 +01:00
|
|
|
meta = {
|
2015-07-12 11:09:40 +01:00
|
|
|
# put global maintainers here, individuals go into makeInstallerTest fkt call
|
2019-01-26 10:01:09 +00:00
|
|
|
maintainers = (meta.maintainers or []);
|
2015-07-12 11:09:40 +01:00
|
|
|
};
|
2015-06-10 12:14:40 +01:00
|
|
|
nodes = {
|
|
|
|
|
2018-02-07 18:32:41 +00:00
|
|
|
# The configuration of the machine used to run "nixos-install".
|
2020-01-28 07:54:05 +00:00
|
|
|
machine = { pkgs, ... }: {
|
|
|
|
imports = [
|
|
|
|
../modules/profiles/installation-device.nix
|
|
|
|
../modules/profiles/base.nix
|
|
|
|
extraInstallerConfig
|
nixos/qemu-vm: use persistent block device names
This change removes the bespoke logic around identifying block devices.
Instead of trying to find the right device by iterating over
`qemu.drives` and guessing the right partition number (e.g.
/dev/vda{1,2}), devices are now identified by persistent names provided
by udev in /dev/disk/by-*.
Before this change, the root device was formatted on demand in the
initrd. However, this makes it impossible to use filesystem identifiers
to identify devices. Now, the formatting step is performed before the VM
is started. Because some tests, however, rely on this behaviour, a
utility function to replace this behaviour in added in
/nixos/tests/common/auto-format-root-device.nix.
Devices that contain neither a partition table nor a filesystem are
identified by their hardware serial number which is injecetd via QEMU
(and is thus persistent and predictable). PCI paths are not a reliably
way to identify devices because their availability and numbering depends
on the QEMU machine type.
This change makes the module more robust against changes in QEMU and the
kernel (non-persistent device naming) and by decoupling abstractions
(i.e. rootDevice, bootPartition, and bootLoaderDevice) enables further
improvement down the line.
2023-06-08 01:34:10 +01:00
|
|
|
./common/auto-format-root-device.nix
|
2020-01-28 07:54:05 +00:00
|
|
|
];
|
|
|
|
|
nixos/qemu-vm: use persistent block device names
This change removes the bespoke logic around identifying block devices.
Instead of trying to find the right device by iterating over
`qemu.drives` and guessing the right partition number (e.g.
/dev/vda{1,2}), devices are now identified by persistent names provided
by udev in /dev/disk/by-*.
Before this change, the root device was formatted on demand in the
initrd. However, this makes it impossible to use filesystem identifiers
to identify devices. Now, the formatting step is performed before the VM
is started. Because some tests, however, rely on this behaviour, a
utility function to replace this behaviour in added in
/nixos/tests/common/auto-format-root-device.nix.
Devices that contain neither a partition table nor a filesystem are
identified by their hardware serial number which is injecetd via QEMU
(and is thus persistent and predictable). PCI paths are not a reliably
way to identify devices because their availability and numbering depends
on the QEMU machine type.
This change makes the module more robust against changes in QEMU and the
kernel (non-persistent device naming) and by decoupling abstractions
(i.e. rootDevice, bootPartition, and bootLoaderDevice) enables further
improvement down the line.
2023-06-08 01:34:10 +01:00
|
|
|
# In systemdStage1, also automatically format the device backing the
|
|
|
|
# root filesystem.
|
|
|
|
virtualisation.fileSystems."/".autoFormat = systemdStage1;
|
|
|
|
|
2021-01-10 12:41:09 +00:00
|
|
|
# builds stuff in the VM, needs more juice
|
2020-01-28 07:54:05 +00:00
|
|
|
virtualisation.diskSize = 8 * 1024;
|
2021-01-10 12:41:09 +00:00
|
|
|
virtualisation.cores = 8;
|
2023-01-18 10:45:48 +00:00
|
|
|
virtualisation.memorySize = 1536;
|
2020-01-28 07:54:05 +00:00
|
|
|
|
2022-04-17 22:24:13 +01:00
|
|
|
boot.initrd.systemd.enable = systemdStage1;
|
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
# Use a small /dev/vdb as the root disk for the
|
|
|
|
# installer. This ensures the target disk (/dev/vda) is
|
|
|
|
# the same during and after installation.
|
|
|
|
virtualisation.emptyDiskImages = [ 512 ];
|
2022-04-04 17:22:03 +01:00
|
|
|
virtualisation.rootDevice = "/dev/vdb";
|
2022-10-15 23:18:03 +01:00
|
|
|
virtualisation.bootLoaderDevice = "/dev/vda";
|
2022-04-04 17:22:03 +01:00
|
|
|
virtualisation.qemu.diskInterface = "virtio";
|
2023-11-18 19:38:08 +00:00
|
|
|
virtualisation.qemu.options = mkIf (clevisTest) [
|
|
|
|
"-chardev socket,id=chrtpm,path=$NIX_BUILD_TOP/swtpm-sock"
|
|
|
|
"-tpmdev emulator,id=tpm0,chardev=chrtpm"
|
|
|
|
"-device tpm-tis,tpmdev=tpm0"
|
|
|
|
];
|
|
|
|
# We don't want to have any networking in the guest apart from the clevis tests.
|
|
|
|
virtualisation.vlans = mkIf (!clevisTest) [];
|
2022-04-22 20:27:03 +01:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
boot.loader.systemd-boot.enable = mkIf (bootLoader == "systemd-boot") true;
|
|
|
|
|
|
|
|
hardware.enableAllFirmware = mkForce false;
|
|
|
|
|
|
|
|
# The test cannot access the network, so any packages we
|
|
|
|
# need must be included in the VM.
|
|
|
|
system.extraDependencies = with pkgs; [
|
2023-07-21 08:36:29 +01:00
|
|
|
bintools
|
2022-01-22 20:44:44 +00:00
|
|
|
brotli
|
|
|
|
brotli.dev
|
|
|
|
brotli.lib
|
2020-01-28 07:54:05 +00:00
|
|
|
desktop-file-utils
|
|
|
|
docbook5
|
|
|
|
docbook_xsl_ns
|
2023-05-04 07:29:04 +01:00
|
|
|
kbd.dev
|
2022-04-05 17:54:45 +01:00
|
|
|
kmod.dev
|
2022-04-22 19:03:46 +01:00
|
|
|
libarchive.dev
|
2020-01-28 07:54:05 +00:00
|
|
|
libxml2.bin
|
|
|
|
libxslt.bin
|
|
|
|
nixos-artwork.wallpapers.simple-dark-gray-bottom
|
|
|
|
ntp
|
|
|
|
perlPackages.ListCompare
|
|
|
|
perlPackages.XMLLibXML
|
2022-09-18 12:59:14 +01:00
|
|
|
# make-options-doc/default.nix
|
2024-01-12 14:22:49 +00:00
|
|
|
(python3.withPackages (p: [ p.mistune ]))
|
2020-01-28 07:54:05 +00:00
|
|
|
shared-mime-info
|
|
|
|
sudo
|
|
|
|
texinfo
|
|
|
|
unionfs-fuse
|
|
|
|
xorg.lndir
|
|
|
|
|
|
|
|
# add curl so that rather than seeing the test attempt to download
|
|
|
|
# curl's tarball, we see what it's trying to download
|
|
|
|
curl
|
|
|
|
]
|
2022-04-04 17:22:03 +01:00
|
|
|
++ optionals (bootLoader == "grub") (let
|
2021-02-13 03:33:46 +00:00
|
|
|
zfsSupport = lib.any (x: x == "zfs")
|
|
|
|
(extraInstallerConfig.boot.supportedFilesystems or []);
|
|
|
|
in [
|
|
|
|
(pkgs.grub2.override { inherit zfsSupport; })
|
|
|
|
(pkgs.grub2_efi.override { inherit zfsSupport; })
|
2023-11-18 19:38:08 +00:00
|
|
|
]) ++ optionals clevisTest [ pkgs.klibc ];
|
2020-01-28 07:54:05 +00:00
|
|
|
|
2021-11-19 22:36:26 +00:00
|
|
|
nix.settings = {
|
|
|
|
substituters = mkForce [];
|
|
|
|
hashed-mirrors = null;
|
|
|
|
connect-timeout = 1;
|
|
|
|
};
|
2020-01-28 07:54:05 +00:00
|
|
|
};
|
2015-06-10 12:14:40 +01:00
|
|
|
|
2023-11-18 19:38:08 +00:00
|
|
|
} // optionalAttrs clevisTest {
|
|
|
|
tang = {
|
|
|
|
services.tang = {
|
|
|
|
enable = true;
|
|
|
|
listenStream = [ "80" ];
|
|
|
|
ipAddressAllow = [ "192.168.1.0/24" ];
|
|
|
|
};
|
|
|
|
networking.firewall.allowedTCPPorts = [ 80 ];
|
|
|
|
};
|
2015-06-10 12:14:40 +01:00
|
|
|
};
|
|
|
|
|
2010-01-10 01:26:01 +00:00
|
|
|
testScript = testScriptFun {
|
2023-10-23 19:20:10 +01:00
|
|
|
inherit bootLoader createPartitions postInstallCommands preBootCommands postBootCommands
|
2022-04-04 17:22:03 +01:00
|
|
|
grubDevice grubIdentifier grubUseEfi extraConfig
|
2023-11-18 19:38:08 +00:00
|
|
|
testSpecialisationConfig testFlakeSwitch clevisTest clevisFallbackTest;
|
2010-01-10 01:26:01 +00:00
|
|
|
};
|
2014-04-14 15:24:08 +01:00
|
|
|
};
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
makeLuksRootTest = name: luksFormatOpts: makeInstallerTest name {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
2019-03-01 22:47:19 +00:00
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
2021-11-30 01:53:14 +00:00
|
|
|
+ " mkpart primary ext2 1M 100MB" # /boot
|
|
|
|
+ " mkpart primary linux-swap 100M 1024M"
|
2020-01-28 07:54:05 +00:00
|
|
|
+ " mkpart primary 1024M -1s", # LUKS
|
2019-03-01 22:47:19 +00:00
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"modprobe dm_mod dm_crypt",
|
|
|
|
"echo -n supersecret | cryptsetup luksFormat ${luksFormatOpts} -q /dev/vda3 -",
|
|
|
|
"echo -n supersecret | cryptsetup luksOpen --key-file - /dev/vda3 cryptroot",
|
|
|
|
"mkfs.ext3 -L nixos /dev/mapper/cryptroot",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount LABEL=boot /mnt/boot",
|
2020-01-28 07:54:05 +00:00
|
|
|
)
|
|
|
|
'';
|
|
|
|
extraConfig = ''
|
|
|
|
boot.kernelParams = lib.mkAfter [ "console=tty0" ];
|
|
|
|
'';
|
|
|
|
enableOCR = true;
|
|
|
|
preBootCommands = ''
|
|
|
|
machine.start()
|
2023-10-23 22:40:34 +01:00
|
|
|
machine.wait_for_text("[Pp]assphrase for")
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.send_chars("supersecret\n")
|
|
|
|
'';
|
|
|
|
};
|
2019-03-01 22:47:19 +00:00
|
|
|
|
2019-03-07 09:30:11 +00:00
|
|
|
# The (almost) simplest partitioning scheme: a swap partition and
|
|
|
|
# one big filesystem partition.
|
2020-01-28 07:54:05 +00:00
|
|
|
simple-test-config = {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
|
|
|
+ " mkpart primary linux-swap 1M 1024M"
|
|
|
|
+ " mkpart primary ext2 1024M -1s",
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda1 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.ext3 -L nixos /dev/vda2",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2023-07-07 17:53:19 +01:00
|
|
|
simple-test-config-flake = simple-test-config // {
|
|
|
|
testFlakeSwitch = true;
|
|
|
|
};
|
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
simple-uefi-grub-config = {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel gpt"
|
2021-02-18 09:58:25 +00:00
|
|
|
+ " mkpart ESP fat32 1M 100MiB" # /boot
|
2020-01-28 07:54:05 +00:00
|
|
|
+ " set 1 boot on"
|
2021-02-18 09:58:25 +00:00
|
|
|
+ " mkpart primary linux-swap 100MiB 1024MiB"
|
2020-01-28 07:54:05 +00:00
|
|
|
+ " mkpart primary ext2 1024MiB -1MiB", # /
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.ext3 -L nixos /dev/vda3",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkfs.vfat -n BOOT /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount LABEL=BOOT /mnt/boot",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
bootLoader = "grub";
|
|
|
|
grubUseEfi = true;
|
|
|
|
};
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2020-03-05 22:07:20 +00:00
|
|
|
specialisation-test-extraconfig = {
|
2020-01-28 07:54:05 +00:00
|
|
|
extraConfig = ''
|
|
|
|
environment.systemPackages = [ pkgs.grub2 ];
|
|
|
|
boot.loader.grub.configurationName = "Home";
|
2020-03-05 22:07:20 +00:00
|
|
|
specialisation.work.configuration = {
|
2020-01-28 07:54:05 +00:00
|
|
|
boot.loader.grub.configurationName = lib.mkForce "Work";
|
|
|
|
|
|
|
|
environment.etc = {
|
|
|
|
"gitconfig".text = "
|
|
|
|
[core]
|
|
|
|
gitproxy = none for work.com
|
|
|
|
";
|
|
|
|
};
|
2020-03-05 22:07:20 +00:00
|
|
|
};
|
2020-01-28 07:54:05 +00:00
|
|
|
'';
|
2020-03-05 22:07:20 +00:00
|
|
|
testSpecialisationConfig = true;
|
2019-03-07 09:38:28 +00:00
|
|
|
};
|
2023-01-01 18:52:05 +00:00
|
|
|
# disable zfs so we can support latest kernel if needed
|
|
|
|
no-zfs-module = {
|
|
|
|
nixpkgs.overlays = [(final: super: {
|
|
|
|
zfs = super.zfs.overrideAttrs(_: {meta.platforms = [];});}
|
|
|
|
)];
|
|
|
|
};
|
2023-11-18 19:38:08 +00:00
|
|
|
|
|
|
|
mkClevisBcachefsTest = { fallback ? false }: makeInstallerTest "clevis-bcachefs${optionalString fallback "-fallback"}" {
|
|
|
|
clevisTest = true;
|
|
|
|
clevisFallbackTest = fallback;
|
|
|
|
enableOCR = fallback;
|
|
|
|
extraInstallerConfig = {
|
|
|
|
imports = [ no-zfs-module ];
|
|
|
|
boot.supportedFilesystems = [ "bcachefs" ];
|
|
|
|
environment.systemPackages = with pkgs; [ keyutils clevis ];
|
|
|
|
};
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
|
|
|
+ " mkpart primary ext2 1M 100MB"
|
|
|
|
+ " mkpart primary linux-swap 100M 1024M"
|
|
|
|
+ " mkpart primary 1024M -1s",
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"keyctl link @u @s",
|
|
|
|
"echo -n password | mkfs.bcachefs -L root --encrypted /dev/vda3",
|
|
|
|
"echo -n password | bcachefs unlock /dev/vda3",
|
|
|
|
"echo -n password | mount -t bcachefs /dev/vda3 /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount LABEL=boot /mnt/boot",
|
|
|
|
"udevadm settle")
|
|
|
|
'';
|
|
|
|
extraConfig = ''
|
|
|
|
boot.initrd.clevis.devices."/dev/vda3".secretFile = "/etc/nixos/clevis-secret.jwe";
|
|
|
|
|
|
|
|
# We override what nixos-generate-config has generated because we do
|
|
|
|
# not know the UUID in advance.
|
|
|
|
fileSystems."/" = lib.mkForce { device = "/dev/vda3"; fsType = "bcachefs"; };
|
|
|
|
'';
|
|
|
|
preBootCommands = ''
|
|
|
|
tpm = Tpm()
|
|
|
|
tpm.check()
|
|
|
|
'' + optionalString fallback ''
|
|
|
|
machine.start()
|
|
|
|
machine.wait_for_text("enter passphrase for")
|
|
|
|
machine.send_chars("password\n")
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
mkClevisLuksTest = { fallback ? false }: makeInstallerTest "clevis-luks${optionalString fallback "-fallback"}" {
|
|
|
|
clevisTest = true;
|
|
|
|
clevisFallbackTest = fallback;
|
|
|
|
enableOCR = fallback;
|
|
|
|
extraInstallerConfig = {
|
|
|
|
environment.systemPackages = with pkgs; [ clevis ];
|
|
|
|
};
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
|
|
|
+ " mkpart primary ext2 1M 100MB"
|
|
|
|
+ " mkpart primary linux-swap 100M 1024M"
|
|
|
|
+ " mkpart primary 1024M -1s",
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"modprobe dm_mod dm_crypt",
|
|
|
|
"echo -n password | cryptsetup luksFormat -q /dev/vda3 -",
|
|
|
|
"echo -n password | cryptsetup luksOpen --key-file - /dev/vda3 crypt-root",
|
|
|
|
"mkfs.ext3 -L nixos /dev/mapper/crypt-root",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount LABEL=boot /mnt/boot",
|
|
|
|
"udevadm settle")
|
|
|
|
'';
|
|
|
|
extraConfig = ''
|
|
|
|
boot.initrd.clevis.devices."crypt-root".secretFile = "/etc/nixos/clevis-secret.jwe";
|
|
|
|
'';
|
|
|
|
preBootCommands = ''
|
|
|
|
tpm = Tpm()
|
|
|
|
tpm.check()
|
|
|
|
'' + optionalString fallback ''
|
|
|
|
machine.start()
|
|
|
|
${if systemdStage1 then ''
|
|
|
|
machine.wait_for_text("Please enter")
|
|
|
|
'' else ''
|
|
|
|
machine.wait_for_text("Passphrase for")
|
|
|
|
''}
|
|
|
|
machine.send_chars("password\n")
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
mkClevisZfsTest = { fallback ? false }: makeInstallerTest "clevis-zfs${optionalString fallback "-fallback"}" {
|
|
|
|
clevisTest = true;
|
|
|
|
clevisFallbackTest = fallback;
|
|
|
|
enableOCR = fallback;
|
|
|
|
extraInstallerConfig = {
|
|
|
|
boot.supportedFilesystems = [ "zfs" ];
|
|
|
|
environment.systemPackages = with pkgs; [ clevis ];
|
|
|
|
};
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
|
|
|
+ " mkpart primary ext2 1M 100MB"
|
|
|
|
+ " mkpart primary linux-swap 100M 1024M"
|
|
|
|
+ " mkpart primary 1024M -1s",
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"zpool create -O mountpoint=legacy rpool /dev/vda3",
|
|
|
|
"echo -n password | zfs create"
|
|
|
|
+ " -o encryption=aes-256-gcm -o keyformat=passphrase rpool/root",
|
|
|
|
"mount -t zfs rpool/root /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount LABEL=boot /mnt/boot",
|
|
|
|
"udevadm settle")
|
|
|
|
'';
|
|
|
|
extraConfig = ''
|
|
|
|
boot.initrd.clevis.devices."rpool/root".secretFile = "/etc/nixos/clevis-secret.jwe";
|
|
|
|
boot.zfs.requestEncryptionCredentials = true;
|
|
|
|
|
|
|
|
|
|
|
|
# Using by-uuid overrides the default of by-id, and is unique
|
|
|
|
# to the qemu disks, as they don't produce by-id paths for
|
|
|
|
# some reason.
|
|
|
|
boot.zfs.devNodes = "/dev/disk/by-uuid/";
|
|
|
|
networking.hostId = "00000000";
|
|
|
|
'';
|
|
|
|
preBootCommands = ''
|
|
|
|
tpm = Tpm()
|
|
|
|
tpm.check()
|
|
|
|
'' + optionalString fallback ''
|
|
|
|
machine.start()
|
|
|
|
${if systemdStage1 then ''
|
|
|
|
machine.wait_for_text("Enter key for rpool/root")
|
|
|
|
'' else ''
|
|
|
|
machine.wait_for_text("Key load error")
|
|
|
|
''}
|
|
|
|
machine.send_chars("password\n")
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2019-03-07 09:36:40 +00:00
|
|
|
in {
|
|
|
|
|
|
|
|
# !!! `parted mkpart' seems to silently create overlapping partitions.
|
|
|
|
|
|
|
|
|
|
|
|
# The (almost) simplest partitioning scheme: a swap partition and
|
|
|
|
# one big filesystem partition.
|
|
|
|
simple = makeInstallerTest "simple" simple-test-config;
|
2016-04-23 12:20:34 +01:00
|
|
|
|
2023-07-07 17:53:19 +01:00
|
|
|
switchToFlake = makeInstallerTest "switch-to-flake" simple-test-config-flake;
|
|
|
|
|
2019-03-07 09:38:28 +00:00
|
|
|
# Test cloned configurations with the simple grub configuration
|
2020-03-05 22:07:20 +00:00
|
|
|
simpleSpecialised = makeInstallerTest "simpleSpecialised" (simple-test-config // specialisation-test-extraconfig);
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2019-03-07 09:36:40 +00:00
|
|
|
# Simple GPT/UEFI configuration using systemd-boot with 3 partitions: ESP, swap & root filesystem
|
2020-01-28 07:54:05 +00:00
|
|
|
simpleUefiSystemdBoot = makeInstallerTest "simpleUefiSystemdBoot" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel gpt"
|
2021-11-30 01:53:14 +00:00
|
|
|
+ " mkpart ESP fat32 1M 100MiB" # /boot
|
2020-01-28 07:54:05 +00:00
|
|
|
+ " set 1 boot on"
|
2021-11-30 01:53:14 +00:00
|
|
|
+ " mkpart primary linux-swap 100MiB 1024MiB"
|
2020-01-28 07:54:05 +00:00
|
|
|
+ " mkpart primary ext2 1024MiB -1MiB", # /
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.ext3 -L nixos /dev/vda3",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkfs.vfat -n BOOT /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount LABEL=BOOT /mnt/boot",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
bootLoader = "systemd-boot";
|
|
|
|
};
|
2017-02-15 22:59:40 +00:00
|
|
|
|
2019-03-07 09:36:40 +00:00
|
|
|
simpleUefiGrub = makeInstallerTest "simpleUefiGrub" simple-uefi-grub-config;
|
|
|
|
|
2019-03-07 09:38:28 +00:00
|
|
|
# Test cloned configurations with the uefi grub configuration
|
2020-03-05 22:07:20 +00:00
|
|
|
simpleUefiGrubSpecialisation = makeInstallerTest "simpleUefiGrubSpecialisation" (simple-uefi-grub-config // specialisation-test-extraconfig);
|
2019-03-07 09:38:28 +00:00
|
|
|
|
2010-01-07 13:19:38 +00:00
|
|
|
# Same as the previous, but now with a separate /boot partition.
|
2020-01-28 07:54:05 +00:00
|
|
|
separateBoot = makeInstallerTest "separateBoot" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
2021-11-30 01:53:14 +00:00
|
|
|
+ " mkpart primary ext2 1M 100MB" # /boot
|
|
|
|
+ " mkpart primary linux-swap 100MB 1024M"
|
2020-01-28 07:54:05 +00:00
|
|
|
+ " mkpart primary ext2 1024M -1s", # /
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.ext3 -L nixos /dev/vda3",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount LABEL=boot /mnt/boot",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
2011-09-14 19:20:50 +01:00
|
|
|
|
2015-07-08 20:05:31 +01:00
|
|
|
# Same as the previous, but with fat32 /boot.
|
2020-01-28 07:54:05 +00:00
|
|
|
separateBootFat = makeInstallerTest "separateBootFat" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
2021-11-30 01:53:14 +00:00
|
|
|
+ " mkpart primary ext2 1M 100MB" # /boot
|
|
|
|
+ " mkpart primary linux-swap 100MB 1024M"
|
2020-01-28 07:54:05 +00:00
|
|
|
+ " mkpart primary ext2 1024M -1s", # /
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.ext3 -L nixos /dev/vda3",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkfs.vfat -n BOOT /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount LABEL=BOOT /mnt/boot",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
2015-07-08 20:05:31 +01:00
|
|
|
|
2023-03-24 18:16:24 +00:00
|
|
|
# Same as the previous, but with ZFS /boot.
|
|
|
|
separateBootZfs = makeInstallerTest "separateBootZfs" {
|
|
|
|
extraInstallerConfig = {
|
|
|
|
boot.supportedFilesystems = [ "zfs" ];
|
|
|
|
};
|
|
|
|
|
|
|
|
extraConfig = ''
|
|
|
|
# Using by-uuid overrides the default of by-id, and is unique
|
|
|
|
# to the qemu disks, as they don't produce by-id paths for
|
|
|
|
# some reason.
|
|
|
|
boot.zfs.devNodes = "/dev/disk/by-uuid/";
|
|
|
|
networking.hostId = "00000000";
|
|
|
|
'';
|
|
|
|
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
|
|
|
+ " mkpart primary ext2 1M 256MB" # /boot
|
|
|
|
+ " mkpart primary linux-swap 256MB 1280M"
|
|
|
|
+ " mkpart primary ext2 1280M -1s", # /
|
|
|
|
"udevadm settle",
|
|
|
|
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
|
|
|
|
"mkfs.ext4 -L nixos /dev/vda3",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
|
|
|
|
# Use as many ZFS features as possible to verify that GRUB can handle them
|
|
|
|
"zpool create"
|
|
|
|
" -o compatibility=grub2"
|
|
|
|
" -O utf8only=on"
|
|
|
|
" -O normalization=formD"
|
|
|
|
" -O compression=lz4" # Activate the lz4_compress feature
|
|
|
|
" -O xattr=sa"
|
|
|
|
" -O acltype=posixacl"
|
|
|
|
" bpool /dev/vda1",
|
|
|
|
"zfs create"
|
|
|
|
" -o recordsize=1M" # Prepare activating the large_blocks feature
|
|
|
|
" -o mountpoint=legacy"
|
|
|
|
" -o relatime=on"
|
|
|
|
" -o quota=1G"
|
|
|
|
" -o filesystem_limit=100" # Activate the filesystem_limits features
|
|
|
|
" bpool/boot",
|
|
|
|
|
|
|
|
# Snapshotting the top-level dataset would trigger a bug in GRUB2: https://github.com/openzfs/zfs/issues/13873
|
|
|
|
"zfs snapshot bpool/boot@snap-1", # Prepare activating the livelist and bookmarks features
|
|
|
|
"zfs clone bpool/boot@snap-1 bpool/test", # Activate the livelist feature
|
|
|
|
"zfs bookmark bpool/boot@snap-1 bpool/boot#bookmark", # Activate the bookmarks feature
|
|
|
|
"zpool checkpoint bpool", # Activate the zpool_checkpoint feature
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount -t zfs bpool/boot /mnt/boot",
|
|
|
|
"touch /mnt/boot/empty", # Activate zilsaxattr feature
|
|
|
|
"dd if=/dev/urandom of=/mnt/boot/test bs=1M count=1", # Activate the large_blocks feature
|
|
|
|
|
|
|
|
# Print out all enabled and active ZFS features (and some other stuff)
|
|
|
|
"sync /mnt/boot",
|
|
|
|
"zpool get all bpool >&2",
|
|
|
|
|
|
|
|
# Abort early if GRUB2 doesn't like the disks
|
|
|
|
"grub-probe --target=device /mnt/boot >&2",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
|
|
|
|
# umount & export bpool before shutdown
|
|
|
|
# this is a fix for "cannot import 'bpool': pool was previously in use from another system."
|
|
|
|
postInstallCommands = ''
|
|
|
|
machine.succeed("umount /mnt/boot")
|
|
|
|
machine.succeed("zpool export bpool")
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2017-06-28 00:35:55 +01:00
|
|
|
# zfs on / with swap
|
2020-01-28 07:54:05 +00:00
|
|
|
zfsroot = makeInstallerTest "zfs-root" {
|
|
|
|
extraInstallerConfig = {
|
|
|
|
boot.supportedFilesystems = [ "zfs" ];
|
|
|
|
};
|
2017-06-28 00:35:55 +01:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
extraConfig = ''
|
|
|
|
boot.supportedFilesystems = [ "zfs" ];
|
2017-06-28 00:35:55 +01:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
# Using by-uuid overrides the default of by-id, and is unique
|
|
|
|
# to the qemu disks, as they don't produce by-id paths for
|
|
|
|
# some reason.
|
|
|
|
boot.zfs.devNodes = "/dev/disk/by-uuid/";
|
|
|
|
networking.hostId = "00000000";
|
|
|
|
'';
|
2017-06-28 00:35:55 +01:00
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
2023-03-24 18:16:24 +00:00
|
|
|
+ " mkpart primary 1M 100MB" # /boot
|
2023-10-23 17:53:57 +01:00
|
|
|
+ " mkpart primary linux-swap 100M 1024M"
|
|
|
|
+ " mkpart primary 1024M -1s", # rpool
|
2020-01-28 07:54:05 +00:00
|
|
|
"udevadm settle",
|
2023-10-23 17:53:57 +01:00
|
|
|
"mkswap /dev/vda2 -L swap",
|
2020-01-28 07:54:05 +00:00
|
|
|
"swapon -L swap",
|
2023-10-23 17:53:57 +01:00
|
|
|
"zpool create rpool /dev/vda3",
|
2020-01-28 07:54:05 +00:00
|
|
|
"zfs create -o mountpoint=legacy rpool/root",
|
|
|
|
"mount -t zfs rpool/root /mnt",
|
2023-08-02 19:56:25 +01:00
|
|
|
"zfs create -o mountpoint=legacy rpool/root/usr",
|
|
|
|
"mkdir /mnt/usr",
|
|
|
|
"mount -t zfs rpool/root/usr /mnt/usr",
|
2023-03-24 18:16:24 +00:00
|
|
|
"mkfs.vfat -n BOOT /dev/vda1",
|
2023-10-23 17:53:57 +01:00
|
|
|
"mkdir /mnt/boot",
|
2023-03-24 18:16:24 +00:00
|
|
|
"mount LABEL=BOOT /mnt/boot",
|
2020-01-28 07:54:05 +00:00
|
|
|
"udevadm settle",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
2017-06-28 00:35:55 +01:00
|
|
|
|
2010-01-07 14:19:19 +00:00
|
|
|
# Create two physical LVM partitions combined into one volume group
|
2012-08-01 20:40:58 +01:00
|
|
|
# that contains the logical swap and root partitions.
|
2020-01-28 07:54:05 +00:00
|
|
|
lvm = makeInstallerTest "lvm" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
|
|
|
+ " mkpart primary 1M 2048M" # PV1
|
|
|
|
+ " set 1 lvm on"
|
|
|
|
+ " mkpart primary 2048M -1s" # PV2
|
|
|
|
+ " set 2 lvm on",
|
|
|
|
"udevadm settle",
|
|
|
|
"pvcreate /dev/vda1 /dev/vda2",
|
|
|
|
"vgcreate MyVolGroup /dev/vda1 /dev/vda2",
|
|
|
|
"lvcreate --size 1G --name swap MyVolGroup",
|
2022-03-24 00:12:24 +00:00
|
|
|
"lvcreate --size 6G --name nixos MyVolGroup",
|
2020-01-28 07:54:05 +00:00
|
|
|
"mkswap -f /dev/MyVolGroup/swap -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.xfs -L nixos /dev/MyVolGroup/nixos",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
2010-05-16 21:44:45 +01:00
|
|
|
|
2019-03-01 22:47:19 +00:00
|
|
|
# Boot off an encrypted root partition with the default LUKS header format
|
|
|
|
luksroot = makeLuksRootTest "luksroot-format1" "";
|
|
|
|
|
|
|
|
# Boot off an encrypted root partition with LUKS1 format
|
|
|
|
luksroot-format1 = makeLuksRootTest "luksroot-format1" "--type=LUKS1";
|
|
|
|
|
|
|
|
# Boot off an encrypted root partition with LUKS2 format
|
|
|
|
luksroot-format2 = makeLuksRootTest "luksroot-format2" "--type=LUKS2";
|
nixos/tests/installer: Add test for LUKS rootfs.
This serves as a regression test for #7859.
It's pretty straightforward, except from the fact that nixos-generate-
config doesn't detect LUKS devices and the "sleep 60".
As for the former, I have tried to add support for LUKS devices for
nixos-generate-config, but it's not so easy as it sounds, because we
need to create a device tree across all possible mappers and/or LVM up
to the "real" device and then decide whether it is relevant to what is
currently mounted. So I guess this is something for the nixpart branch
(see #2079).
And the latter isn't very trivial as well, because the LUKS passphrase
prompt is issued on /dev/console, which is the last "console=..." kernel
parameter (thus the `mkAfter`). So we can't simply grep the log, because
the prompt ends up being on one terminal only (tty0) and using select()
on $machine->{socket} doesn't work very well, because the FD is always
"ready for read". If we would read the FD, we would conflict with
$machine->connect and end up having an inconsistent state. Another idea
would be to use multithreading to do $machine->connect while feeding the
passphrase prompt in a loop and stop the thread once $machine->connect
is done. Turns out that this is not so easy as well, because the threads
need to share the $machine object and of course need to do properly
locking.
In the end I decided to use the "blindly hope that 60 seconds is enough"
approach for now and come up with a better solution later. Other VM
tests surely use sleep as well, but it's $machine->sleep, which is bound
to the clock of the VM, so if the build machine is on high load, a
$machine->sleep gets properly delayed but the timer outside the VM won't
get that delay, so the test is not deterministic.
Tested against the following revisions:
5e3fe39: Before the libgcrypt cleanup (a71f78a) that broke cryptsetup.
69a6848: While cryptsetup was broken (obviously the test failed).
15faa43: After cryptsetup has been switched to OpenSSL (fd588f9).
Signed-off-by: aszlig <aszlig@redmoonstudios.org>
2015-05-21 11:42:00 +01:00
|
|
|
|
2017-09-12 12:57:37 +01:00
|
|
|
# Test whether opening encrypted filesystem with keyfile
|
|
|
|
# Checks for regression of missing cryptsetup, when no luks device without
|
|
|
|
# keyfile is configured
|
2020-01-28 07:54:05 +00:00
|
|
|
encryptedFSWithKeyfile = makeInstallerTest "encryptedFSWithKeyfile" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
2018-09-22 11:22:17 +01:00
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
2021-11-30 01:53:14 +00:00
|
|
|
+ " mkpart primary ext2 1M 100MB" # /boot
|
|
|
|
+ " mkpart primary linux-swap 100M 1024M"
|
2020-01-28 07:54:05 +00:00
|
|
|
+ " mkpart primary 1024M 1280M" # LUKS with keyfile
|
|
|
|
+ " mkpart primary 1280M -1s",
|
2017-09-12 12:57:37 +01:00
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.ext3 -L nixos /dev/vda4",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount LABEL=boot /mnt/boot",
|
|
|
|
"modprobe dm_mod dm_crypt",
|
|
|
|
"echo -n supersecret > /mnt/keyfile",
|
|
|
|
"cryptsetup luksFormat -q /dev/vda3 --key-file /mnt/keyfile",
|
|
|
|
"cryptsetup luksOpen --key-file /mnt/keyfile /dev/vda3 crypt",
|
|
|
|
"mkfs.ext3 -L test /dev/mapper/crypt",
|
|
|
|
"cryptsetup luksClose crypt",
|
2020-01-28 07:54:05 +00:00
|
|
|
"mkdir -p /mnt/test",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
extraConfig = ''
|
|
|
|
fileSystems."/test" = {
|
|
|
|
device = "/dev/disk/by-label/test";
|
|
|
|
fsType = "ext3";
|
|
|
|
encrypted.enable = true;
|
|
|
|
encrypted.blkDev = "/dev/vda3";
|
|
|
|
encrypted.label = "crypt";
|
2023-10-23 22:40:34 +01:00
|
|
|
encrypted.keyFile = "/${if systemdStage1 then "sysroot" else "mnt-root"}/keyfile";
|
2020-01-28 07:54:05 +00:00
|
|
|
};
|
|
|
|
'';
|
|
|
|
};
|
2017-09-12 12:57:37 +01:00
|
|
|
|
2023-01-15 02:25:17 +00:00
|
|
|
# Full disk encryption (root, kernel and initrd encrypted) using GRUB, GPT/UEFI,
|
|
|
|
# LVM-on-LUKS and a keyfile in initrd.secrets to enter the passphrase once
|
|
|
|
fullDiskEncryption = makeInstallerTest "fullDiskEncryption" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel gpt"
|
|
|
|
+ " mkpart ESP fat32 1M 100MiB" # /boot/efi
|
|
|
|
+ " set 1 boot on"
|
|
|
|
+ " mkpart primary ext2 1024MiB -1MiB", # LUKS
|
|
|
|
"udevadm settle",
|
|
|
|
"modprobe dm_mod dm_crypt",
|
|
|
|
"dd if=/dev/random of=luks.key bs=256 count=1",
|
|
|
|
"echo -n supersecret | cryptsetup luksFormat -q --pbkdf-force-iterations 1000 --type luks1 /dev/vda2 -",
|
|
|
|
"echo -n supersecret | cryptsetup luksAddKey -q --pbkdf-force-iterations 1000 --key-file - /dev/vda2 luks.key",
|
|
|
|
"echo -n supersecret | cryptsetup luksOpen --key-file - /dev/vda2 crypt",
|
|
|
|
"pvcreate /dev/mapper/crypt",
|
|
|
|
"vgcreate crypt /dev/mapper/crypt",
|
|
|
|
"lvcreate -L 100M -n swap crypt",
|
|
|
|
"lvcreate -l '100%FREE' -n nixos crypt",
|
|
|
|
"mkfs.vfat -n efi /dev/vda1",
|
|
|
|
"mkfs.ext4 -L nixos /dev/crypt/nixos",
|
|
|
|
"mkswap -L swap /dev/crypt/swap",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkdir -p /mnt/{etc/nixos,boot/efi}",
|
|
|
|
"mount LABEL=efi /mnt/boot/efi",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mv luks.key /mnt/etc/nixos/"
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
bootLoader = "grub";
|
|
|
|
grubUseEfi = true;
|
|
|
|
extraConfig = ''
|
|
|
|
boot.loader.grub.enableCryptodisk = true;
|
|
|
|
boot.loader.efi.efiSysMountPoint = "/boot/efi";
|
|
|
|
|
|
|
|
boot.initrd.secrets."/luks.key" = ./luks.key;
|
|
|
|
boot.initrd.luks.devices.crypt =
|
|
|
|
{ device = "/dev/vda2";
|
|
|
|
keyFile = "/luks.key";
|
|
|
|
};
|
|
|
|
'';
|
|
|
|
enableOCR = true;
|
|
|
|
preBootCommands = ''
|
|
|
|
machine.start()
|
|
|
|
machine.wait_for_text("Enter passphrase for")
|
|
|
|
machine.send_chars("supersecret\n")
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2020-01-28 07:54:05 +00:00
|
|
|
swraid = makeInstallerTest "swraid" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda --"
|
|
|
|
+ " mklabel msdos"
|
|
|
|
+ " mkpart primary ext2 1M 100MB" # /boot
|
|
|
|
+ " mkpart extended 100M -1s"
|
2020-09-05 09:32:42 +01:00
|
|
|
+ " mkpart logical 102M 3102M" # md0 (root), first device
|
|
|
|
+ " mkpart logical 3103M 6103M" # md0 (root), second device
|
|
|
|
+ " mkpart logical 6104M 6360M" # md1 (swap), first device
|
|
|
|
+ " mkpart logical 6361M 6617M", # md1 (swap), second device
|
2020-01-28 07:54:05 +00:00
|
|
|
"udevadm settle",
|
|
|
|
"ls -l /dev/vda* >&2",
|
|
|
|
"cat /proc/partitions >&2",
|
|
|
|
"udevadm control --stop-exec-queue",
|
|
|
|
"mdadm --create --force /dev/md0 --metadata 1.2 --level=raid1 "
|
|
|
|
+ "--raid-devices=2 /dev/vda5 /dev/vda6",
|
|
|
|
"mdadm --create --force /dev/md1 --metadata 1.2 --level=raid1 "
|
|
|
|
+ "--raid-devices=2 /dev/vda7 /dev/vda8",
|
|
|
|
"udevadm control --start-exec-queue",
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap -f /dev/md1 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.ext3 -L nixos /dev/md0",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir /mnt/boot",
|
|
|
|
"mount LABEL=boot /mnt/boot",
|
|
|
|
"udevadm settle",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
preBootCommands = ''
|
|
|
|
machine.start()
|
|
|
|
machine.fail("dmesg | grep 'immediate safe mode'")
|
|
|
|
'';
|
|
|
|
};
|
2010-09-25 10:32:22 +01:00
|
|
|
|
2020-02-13 01:30:49 +00:00
|
|
|
bcache = makeInstallerTest "bcache" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda --"
|
|
|
|
+ " mklabel msdos"
|
2021-11-30 01:53:14 +00:00
|
|
|
+ " mkpart primary ext2 1M 100MB" # /boot
|
|
|
|
+ " mkpart primary 100MB 512MB " # swap
|
2020-02-13 01:30:49 +00:00
|
|
|
+ " mkpart primary 512MB 1024MB" # Cache (typically SSD)
|
|
|
|
+ " mkpart primary 1024MB -1s ", # Backing device (typically HDD)
|
|
|
|
"modprobe bcache",
|
|
|
|
"udevadm settle",
|
|
|
|
"make-bcache -B /dev/vda4 -C /dev/vda3",
|
|
|
|
"udevadm settle",
|
|
|
|
"mkfs.ext3 -L nixos /dev/bcache0",
|
|
|
|
"mount LABEL=nixos /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir /mnt/boot",
|
|
|
|
"mount LABEL=boot /mnt/boot",
|
|
|
|
"mkswap -f /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2022-05-05 01:26:43 +01:00
|
|
|
bcachefsSimple = makeInstallerTest "bcachefs-simple" {
|
|
|
|
extraInstallerConfig = {
|
|
|
|
boot.supportedFilesystems = [ "bcachefs" ];
|
2023-01-01 18:52:05 +00:00
|
|
|
imports = [ no-zfs-module ];
|
2022-05-05 01:26:43 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
|
|
|
+ " mkpart primary ext2 1M 100MB" # /boot
|
|
|
|
+ " mkpart primary linux-swap 100M 1024M" # swap
|
|
|
|
+ " mkpart primary 1024M -1s", # /
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.bcachefs -L root /dev/vda3",
|
|
|
|
"mount -t bcachefs /dev/vda3 /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount /dev/vda1 /mnt/boot",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
bcachefsEncrypted = makeInstallerTest "bcachefs-encrypted" {
|
|
|
|
extraInstallerConfig = {
|
|
|
|
boot.supportedFilesystems = [ "bcachefs" ];
|
2023-01-01 18:52:05 +00:00
|
|
|
|
|
|
|
# disable zfs so we can support latest kernel if needed
|
|
|
|
imports = [ no-zfs-module ];
|
|
|
|
|
2022-05-05 01:26:43 +01:00
|
|
|
environment.systemPackages = with pkgs; [ keyutils ];
|
|
|
|
};
|
|
|
|
|
|
|
|
extraConfig = ''
|
2023-01-01 18:53:11 +00:00
|
|
|
boot.kernelParams = lib.mkAfter [ "console=tty0" ];
|
|
|
|
'';
|
|
|
|
|
|
|
|
enableOCR = true;
|
|
|
|
preBootCommands = ''
|
|
|
|
machine.start()
|
2023-10-21 16:56:47 +01:00
|
|
|
# Enter it wrong once
|
|
|
|
machine.wait_for_text("enter passphrase for ")
|
|
|
|
machine.send_chars("wrong\n")
|
|
|
|
# Then enter it right.
|
2023-01-01 18:53:11 +00:00
|
|
|
machine.wait_for_text("enter passphrase for ")
|
|
|
|
machine.send_chars("password\n")
|
2022-05-05 01:26:43 +01:00
|
|
|
'';
|
|
|
|
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
|
|
|
+ " mkpart primary ext2 1M 100MB" # /boot
|
|
|
|
+ " mkpart primary linux-swap 100M 1024M" # swap
|
|
|
|
+ " mkpart primary 1024M -1s", # /
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"echo password | mkfs.bcachefs -L root --encrypted /dev/vda3",
|
2023-09-22 11:58:04 +01:00
|
|
|
"echo password | bcachefs unlock -k session /dev/vda3",
|
2023-06-15 18:25:37 +01:00
|
|
|
"echo password | mount -t bcachefs /dev/vda3 /mnt",
|
2022-05-05 01:26:43 +01:00
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount /dev/vda1 /mnt/boot",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
bcachefsMulti = makeInstallerTest "bcachefs-multi" {
|
|
|
|
extraInstallerConfig = {
|
|
|
|
boot.supportedFilesystems = [ "bcachefs" ];
|
2023-01-01 18:52:05 +00:00
|
|
|
|
|
|
|
# disable zfs so we can support latest kernel if needed
|
|
|
|
imports = [ no-zfs-module ];
|
2022-05-05 01:26:43 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"flock /dev/vda parted --script /dev/vda -- mklabel msdos"
|
|
|
|
+ " mkpart primary ext2 1M 100MB" # /boot
|
|
|
|
+ " mkpart primary linux-swap 100M 1024M" # swap
|
|
|
|
+ " mkpart primary 1024M 4096M" # /
|
|
|
|
+ " mkpart primary 4096M -1s", # /
|
|
|
|
"udevadm settle",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.bcachefs -L root --metadata_replicas 2 --foreground_target ssd --promote_target ssd --background_target hdd --label ssd /dev/vda3 --label hdd /dev/vda4",
|
|
|
|
"mount -t bcachefs /dev/vda3:/dev/vda4 /mnt",
|
|
|
|
"mkfs.ext3 -L boot /dev/vda1",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount /dev/vda1 /mnt/boot",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2014-05-02 05:11:49 +01:00
|
|
|
# Test using labels to identify volumes in grub
|
2014-08-29 10:48:00 +01:00
|
|
|
simpleLabels = makeInstallerTest "simpleLabels" {
|
2014-05-02 05:11:49 +01:00
|
|
|
createPartitions = ''
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.succeed(
|
|
|
|
"sgdisk -Z /dev/vda",
|
|
|
|
"sgdisk -n 1:0:+1M -n 2:0:+1G -N 3 -t 1:ef02 -t 2:8200 -t 3:8300 -c 3:root /dev/vda",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.ext4 -L root /dev/vda3",
|
|
|
|
"mount LABEL=root /mnt",
|
|
|
|
)
|
2014-05-02 05:11:49 +01:00
|
|
|
'';
|
|
|
|
grubIdentifier = "label";
|
|
|
|
};
|
|
|
|
|
|
|
|
# Test using the provided disk name within grub
|
2014-05-02 08:03:25 +01:00
|
|
|
# TODO: Fix udev so the symlinks are unneeded in /dev/disks
|
2014-08-29 10:48:00 +01:00
|
|
|
simpleProvided = makeInstallerTest "simpleProvided" {
|
2014-05-02 05:11:49 +01:00
|
|
|
createPartitions = ''
|
2020-01-28 07:54:05 +00:00
|
|
|
uuid = "$(blkid -s UUID -o value /dev/vda2)"
|
|
|
|
machine.succeed(
|
|
|
|
"sgdisk -Z /dev/vda",
|
|
|
|
"sgdisk -n 1:0:+1M -n 2:0:+100M -n 3:0:+1G -N 4 -t 1:ef02 -t 2:8300 "
|
|
|
|
+ "-t 3:8200 -t 4:8300 -c 2:boot -c 4:root /dev/vda",
|
|
|
|
"mkswap /dev/vda3 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.ext4 -L boot /dev/vda2",
|
|
|
|
"mkfs.ext4 -L root /dev/vda4",
|
|
|
|
)
|
|
|
|
machine.execute(f"ln -s ../../vda2 /dev/disk/by-uuid/{uuid}")
|
|
|
|
machine.execute("ln -s ../../vda4 /dev/disk/by-label/root")
|
|
|
|
machine.succeed(
|
|
|
|
"mount /dev/disk/by-label/root /mnt",
|
|
|
|
"mkdir /mnt/boot",
|
|
|
|
f"mount /dev/disk/by-uuid/{uuid} /mnt/boot",
|
|
|
|
)
|
2014-05-02 05:11:49 +01:00
|
|
|
'';
|
|
|
|
grubIdentifier = "provided";
|
|
|
|
};
|
|
|
|
|
2014-05-01 12:21:30 +01:00
|
|
|
# Simple btrfs grub testing
|
2014-08-29 10:48:00 +01:00
|
|
|
btrfsSimple = makeInstallerTest "btrfsSimple" {
|
2014-05-01 12:21:30 +01:00
|
|
|
createPartitions = ''
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.succeed(
|
|
|
|
"sgdisk -Z /dev/vda",
|
|
|
|
"sgdisk -n 1:0:+1M -n 2:0:+1G -N 3 -t 1:ef02 -t 2:8200 -t 3:8300 -c 3:root /dev/vda",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.btrfs -L root /dev/vda3",
|
|
|
|
"mount LABEL=root /mnt",
|
|
|
|
)
|
2014-05-01 12:21:30 +01:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
# Test to see if we can detect /boot and /nix on subvolumes
|
2014-08-29 10:48:00 +01:00
|
|
|
btrfsSubvols = makeInstallerTest "btrfsSubvols" {
|
2014-05-01 12:21:30 +01:00
|
|
|
createPartitions = ''
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.succeed(
|
|
|
|
"sgdisk -Z /dev/vda",
|
|
|
|
"sgdisk -n 1:0:+1M -n 2:0:+1G -N 3 -t 1:ef02 -t 2:8200 -t 3:8300 -c 3:root /dev/vda",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.btrfs -L root /dev/vda3",
|
|
|
|
"btrfs device scan",
|
|
|
|
"mount LABEL=root /mnt",
|
|
|
|
"btrfs subvol create /mnt/boot",
|
|
|
|
"btrfs subvol create /mnt/nixos",
|
|
|
|
"btrfs subvol create /mnt/nixos/default",
|
|
|
|
"umount /mnt",
|
|
|
|
"mount -o defaults,subvol=nixos/default LABEL=root /mnt",
|
|
|
|
"mkdir /mnt/boot",
|
|
|
|
"mount -o defaults,subvol=boot LABEL=root /mnt/boot",
|
|
|
|
)
|
2014-05-01 12:21:30 +01:00
|
|
|
'';
|
|
|
|
};
|
2014-08-30 00:37:28 +01:00
|
|
|
|
|
|
|
# Test to see if we can detect default and aux subvolumes correctly
|
|
|
|
btrfsSubvolDefault = makeInstallerTest "btrfsSubvolDefault" {
|
|
|
|
createPartitions = ''
|
2020-01-28 07:54:05 +00:00
|
|
|
machine.succeed(
|
|
|
|
"sgdisk -Z /dev/vda",
|
|
|
|
"sgdisk -n 1:0:+1M -n 2:0:+1G -N 3 -t 1:ef02 -t 2:8200 -t 3:8300 -c 3:root /dev/vda",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.btrfs -L root /dev/vda3",
|
|
|
|
"btrfs device scan",
|
|
|
|
"mount LABEL=root /mnt",
|
|
|
|
"btrfs subvol create /mnt/badpath",
|
|
|
|
"btrfs subvol create /mnt/badpath/boot",
|
|
|
|
"btrfs subvol create /mnt/nixos",
|
|
|
|
"btrfs subvol set-default "
|
2020-08-23 15:10:02 +01:00
|
|
|
+ "$(btrfs subvol list /mnt | grep 'nixos' | awk '{print $2}') /mnt",
|
2020-01-28 07:54:05 +00:00
|
|
|
"umount /mnt",
|
|
|
|
"mount -o defaults LABEL=root /mnt",
|
|
|
|
"mkdir -p /mnt/badpath/boot", # Help ensure the detection mechanism
|
|
|
|
# is actually looking up subvolumes
|
|
|
|
"mkdir /mnt/boot",
|
|
|
|
"mount -o defaults,subvol=badpath/boot LABEL=root /mnt/boot",
|
|
|
|
)
|
2014-08-30 00:37:28 +01:00
|
|
|
'';
|
|
|
|
};
|
2022-10-31 14:31:10 +00:00
|
|
|
|
|
|
|
# Test to see if we can deal with subvols that need to be escaped in fstab
|
|
|
|
btrfsSubvolEscape = makeInstallerTest "btrfsSubvolEscape" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
|
|
|
"sgdisk -Z /dev/vda",
|
|
|
|
"sgdisk -n 1:0:+1M -n 2:0:+1G -N 3 -t 1:ef02 -t 2:8200 -t 3:8300 -c 3:root /dev/vda",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"mkfs.btrfs -L root /dev/vda3",
|
|
|
|
"btrfs device scan",
|
|
|
|
"mount LABEL=root /mnt",
|
|
|
|
"btrfs subvol create '/mnt/nixos in space'",
|
|
|
|
"btrfs subvol create /mnt/boot",
|
|
|
|
"umount /mnt",
|
|
|
|
"mount -o 'defaults,subvol=nixos in space' LABEL=root /mnt",
|
|
|
|
"mkdir /mnt/boot",
|
|
|
|
"mount -o defaults,subvol=boot LABEL=root /mnt/boot",
|
|
|
|
)
|
|
|
|
'';
|
|
|
|
};
|
2023-11-18 19:38:08 +00:00
|
|
|
} // {
|
|
|
|
clevisBcachefs = mkClevisBcachefsTest { };
|
|
|
|
clevisBcachefsFallback = mkClevisBcachefsTest { fallback = true; };
|
|
|
|
clevisLuks = mkClevisLuksTest { };
|
|
|
|
clevisLuksFallback = mkClevisLuksTest { fallback = true; };
|
|
|
|
clevisZfs = mkClevisZfsTest { };
|
|
|
|
clevisZfsFallback = mkClevisZfsTest { fallback = true; };
|
2023-05-07 17:47:44 +01:00
|
|
|
} // optionalAttrs systemdStage1 {
|
|
|
|
stratisRoot = makeInstallerTest "stratisRoot" {
|
|
|
|
createPartitions = ''
|
|
|
|
machine.succeed(
|
2023-05-08 22:18:55 +01:00
|
|
|
"sgdisk --zap-all /dev/vda",
|
|
|
|
"sgdisk --new=1:0:+100M --typecode=0:ef00 /dev/vda", # /boot
|
|
|
|
"sgdisk --new=2:0:+1G --typecode=0:8200 /dev/vda", # swap
|
|
|
|
"sgdisk --new=3:0:+5G --typecode=0:8300 /dev/vda", # /
|
|
|
|
"udevadm settle",
|
2023-05-07 17:47:44 +01:00
|
|
|
|
2023-05-08 22:18:55 +01:00
|
|
|
"mkfs.vfat /dev/vda1",
|
|
|
|
"mkswap /dev/vda2 -L swap",
|
|
|
|
"swapon -L swap",
|
|
|
|
"stratis pool create my-pool /dev/vda3",
|
|
|
|
"stratis filesystem create my-pool nixos",
|
|
|
|
"udevadm settle",
|
2023-05-07 17:47:44 +01:00
|
|
|
|
2023-05-08 22:18:55 +01:00
|
|
|
"mount /dev/stratis/my-pool/nixos /mnt",
|
|
|
|
"mkdir -p /mnt/boot",
|
|
|
|
"mount /dev/vda1 /mnt/boot"
|
2023-05-07 17:47:44 +01:00
|
|
|
)
|
|
|
|
'';
|
|
|
|
bootLoader = "systemd-boot";
|
|
|
|
extraInstallerConfig = { modulesPath, ...}: {
|
|
|
|
config = {
|
|
|
|
services.stratis.enable = true;
|
|
|
|
environment.systemPackages = [
|
|
|
|
pkgs.stratis-cli
|
|
|
|
pkgs.thin-provisioning-tools
|
|
|
|
pkgs.lvm2.bin
|
|
|
|
pkgs.stratisd.initrd
|
|
|
|
];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
2010-01-06 13:36:21 +00:00
|
|
|
}
|