nixos modules init (rEFInd & cache)
This commit is contained in:
parent
ccbee50a68
commit
522c08cf37
5 changed files with 367 additions and 16 deletions
16
flake.nix
16
flake.nix
|
@ -28,17 +28,6 @@
|
|||
)
|
||||
);
|
||||
|
||||
mkModule = {
|
||||
name ? "default",
|
||||
class,
|
||||
file,
|
||||
}: {
|
||||
_class = class;
|
||||
_file = "${self.outPath}/flake.nix#${class}Modules.${name}";
|
||||
|
||||
imports = [(import file {haipkgsSelf = self;})];
|
||||
};
|
||||
|
||||
loadPackages = pkgs: let
|
||||
packageNames = builtins.attrNames (builtins.readDir ./pkgs);
|
||||
in
|
||||
|
@ -98,10 +87,7 @@
|
|||
self.packages.${pkgs.stdenv.hostPlatform.system}
|
||||
);
|
||||
|
||||
nixosModules.default = mkModule {
|
||||
class = "nixos";
|
||||
file = ./modules/nixos;
|
||||
};
|
||||
nixosModules = import ./modules/nixos;
|
||||
|
||||
haiLib = import ./lib {inherit lib;};
|
||||
|
||||
|
|
|
@ -1 +1,7 @@
|
|||
{}
|
||||
let
|
||||
modules = {
|
||||
refind = import ./refind;
|
||||
haicache = import ./haicache.nix;
|
||||
};
|
||||
in
|
||||
modules // {default = modules;}
|
||||
|
|
29
modules/nixos/haicache.nix
Normal file
29
modules/nixos/haicache.nix
Normal file
|
@ -0,0 +1,29 @@
|
|||
{
|
||||
config,
|
||||
lib,
|
||||
...
|
||||
}: let
|
||||
inherit (lib.modules) mkIf mkBefore;
|
||||
inherit (lib.options) mkOption;
|
||||
inherit (lib.types) types;
|
||||
cfg = config.haipkgs.cache;
|
||||
in {
|
||||
options = {
|
||||
haipkgs.cache.enable = mkOption {
|
||||
default = true;
|
||||
example = false;
|
||||
type = types.bool;
|
||||
description = ''
|
||||
Whether to enable haipkgs binary cache.
|
||||
'';
|
||||
};
|
||||
};
|
||||
config = {
|
||||
nix.settings = mkIf cfg.enable {
|
||||
substituters = mkBefore ["https://haipkgs.cachix.org"];
|
||||
trusted-public-keys = mkBefore [
|
||||
"haipkgs.cachix.org-1:AcjMqIafTEQ7dw99RpeTJU2ywNUn1h8yIxz2+zjpK/A="
|
||||
];
|
||||
};
|
||||
};
|
||||
}
|
101
modules/nixos/refind/default.nix
Normal file
101
modules/nixos/refind/default.nix
Normal file
|
@ -0,0 +1,101 @@
|
|||
{
|
||||
config,
|
||||
lib,
|
||||
pkgs,
|
||||
...
|
||||
}:
|
||||
with lib; let
|
||||
cfg = config.boot.loader.refind;
|
||||
|
||||
efi = config.boot.loader.efi;
|
||||
|
||||
refindBuilder = pkgs.substituteAll {
|
||||
src = ./refind-builder.py;
|
||||
|
||||
isExecutable = true;
|
||||
|
||||
inherit (pkgs) python3;
|
||||
|
||||
nix = config.nix.package.out;
|
||||
|
||||
timeout =
|
||||
if config.boot.loader.timeout != null
|
||||
then config.boot.loader.timeout
|
||||
else "";
|
||||
|
||||
extraConfig = cfg.extraConfig;
|
||||
|
||||
maxEntries = cfg.maxGenerations;
|
||||
|
||||
extraIcons =
|
||||
if cfg.extraIcons != null
|
||||
then cfg.extraIcons
|
||||
else "";
|
||||
|
||||
themes =
|
||||
if cfg.themes != null
|
||||
then cfg.themes
|
||||
else "[]";
|
||||
|
||||
inherit (pkgs) refind efibootmgr coreutils gnugrep gnused gawk utillinux gptfdisk findutils;
|
||||
|
||||
inherit (efi) efiSysMountPoint canTouchEfiVariables;
|
||||
};
|
||||
in {
|
||||
options.boot.loader.refind = {
|
||||
enable = mkOption {
|
||||
default = false;
|
||||
type = types.bool;
|
||||
description = "Whether to enable the refind EFI boot manager";
|
||||
};
|
||||
|
||||
extraConfig = mkOption {
|
||||
type = types.lines;
|
||||
default = "";
|
||||
description = "Extra configuration text appended to refind.conf";
|
||||
};
|
||||
|
||||
maxGenerations = mkOption {
|
||||
type = types.int;
|
||||
default = 100;
|
||||
description = "Maximum number of generations in submenu. This is to avoid problems with refind or possible size problems with the config";
|
||||
};
|
||||
|
||||
extraIcons = mkOption {
|
||||
type = types.nullOr types.path;
|
||||
default = null;
|
||||
description = "A directory containing icons to be copied to 'extra-icons'";
|
||||
};
|
||||
|
||||
themes = mkOption {
|
||||
type = types.listOf types.path;
|
||||
default = [];
|
||||
description = "A list of theme paths to copy";
|
||||
};
|
||||
};
|
||||
|
||||
config = mkIf cfg.enable {
|
||||
environment.systemPackages = with pkgs; [refind gptfdisk findutils];
|
||||
assertions = [
|
||||
{
|
||||
assertion = (config.boot.kernelPackages.kernel.features or {efiBootStub = true;}) ? efiBootStub;
|
||||
|
||||
message = "This kernel does not support the EFI boot stub";
|
||||
}
|
||||
];
|
||||
|
||||
boot.loader.grub.enable = mkDefault false;
|
||||
|
||||
# boot.loader.supportsInitrdSecrets = false; # TODO what does this do ?
|
||||
|
||||
system = {
|
||||
build.installBootLoader = refindBuilder;
|
||||
|
||||
boot.loader.id = "refind";
|
||||
|
||||
requiredKernelConfig = with config.lib.kernelConfig; [
|
||||
(isYes "EFI_STUB")
|
||||
];
|
||||
};
|
||||
};
|
||||
}
|
229
modules/nixos/refind/refind-builder.py
Normal file
229
modules/nixos/refind/refind-builder.py
Normal file
|
@ -0,0 +1,229 @@
|
|||
#! @python3@/bin/python3 -B
|
||||
import argparse
|
||||
import os
|
||||
import os.path
|
||||
import sys
|
||||
import errno
|
||||
import subprocess
|
||||
import glob
|
||||
import datetime
|
||||
import shutil
|
||||
import ctypes
|
||||
libc = ctypes.CDLL("libc.so.6")
|
||||
|
||||
extra_config = '''
|
||||
@extraConfig@
|
||||
'''
|
||||
|
||||
|
||||
def mkdir_p(path):
|
||||
try:
|
||||
os.makedirs(path)
|
||||
except OSError as e:
|
||||
if e.errno != errno.EEXIST or not os.path.isdir(path):
|
||||
raise
|
||||
|
||||
|
||||
def system_dir(profile, generation):
|
||||
if profile:
|
||||
return "/nix/var/nix/profiles/system-profiles/%s-%d-link" % (profile, generation)
|
||||
return "/nix/var/nix/profiles/system-%d-link" % (generation)
|
||||
|
||||
|
||||
def copy_if_not_exists(source, dest):
|
||||
if not os.path.exists(dest):
|
||||
shutil.copyfile(source, dest)
|
||||
|
||||
|
||||
def get_generations(profile=None):
|
||||
gen_list = subprocess.check_output([
|
||||
"@nix@/bin/nix-env",
|
||||
"--list-generations",
|
||||
"-p",
|
||||
"/nix/var/nix/profiles/%s" % ("system-profiles/" + profile if profile else "system"),
|
||||
"--option", "build-users-group", ""
|
||||
], universal_newlines=True)
|
||||
gen_lines = gen_list.split('\n')
|
||||
gen_lines.pop()
|
||||
return [(profile, int(line.split()[0])) for line in gen_lines]
|
||||
|
||||
|
||||
def get_profiles():
|
||||
if os.path.isdir("/nix/var/nix/profiles/system-profiles/"):
|
||||
return [
|
||||
x for x in os.listdir("/nix/var/nix/profiles/system-profiles/")
|
||||
if not x.endswith("-link")
|
||||
]
|
||||
return []
|
||||
|
||||
|
||||
def profile_path(profile, generation, name):
|
||||
return os.readlink("%s/%s" % (system_dir(profile, generation), name))
|
||||
|
||||
|
||||
def copy_from_profile(profile, generation, name, dry_run=False):
|
||||
store_file_path = profile_path(profile, generation, name)
|
||||
suffix = os.path.basename(store_file_path)
|
||||
store_dir = os.path.basename(os.path.dirname(store_file_path))
|
||||
efi_file_path = "/EFI/nixos/%s-%s.efi" % (store_dir, suffix)
|
||||
if not dry_run:
|
||||
copy_if_not_exists(store_file_path, "@efiSysMountPoint@%s" % (efi_file_path))
|
||||
return efi_file_path
|
||||
|
||||
|
||||
def describe_generation(generation_dir):
|
||||
try:
|
||||
with open("%s/nixos-version" % generation_dir) as f:
|
||||
nixos_version = f.read()
|
||||
except IOError:
|
||||
nixos_version = "Unknown"
|
||||
|
||||
kernel_dir = os.path.dirname(os.path.realpath("%s/kernel" % generation_dir))
|
||||
module_dir = glob.glob("%s/lib/modules/*" % kernel_dir)[0]
|
||||
kernel_version = os.path.basename(module_dir)
|
||||
|
||||
build_time = int(os.path.getctime(generation_dir))
|
||||
build_date = datetime.datetime.fromtimestamp(build_time).strftime('%F')
|
||||
|
||||
description = "NixOS {}, Linux Kernel {}, Built on {}".format(
|
||||
nixos_version, kernel_version, build_date
|
||||
)
|
||||
|
||||
return description
|
||||
|
||||
|
||||
def generation_details(profile, generation):
|
||||
kernel = copy_from_profile(profile, generation, "kernel")
|
||||
initrd = copy_from_profile(profile, generation, "initrd")
|
||||
generation_dir = os.readlink(system_dir(profile, generation))
|
||||
kernel_params = "systemConfig=%s init=%s/init " % (generation_dir, generation_dir)
|
||||
with open("%s/kernel-params" % (generation_dir)) as params_file:
|
||||
kernel_params = kernel_params + params_file.read()
|
||||
description = describe_generation(generation_dir)
|
||||
return {
|
||||
"profile": profile,
|
||||
"generation": generation,
|
||||
"kernel": kernel,
|
||||
"initrd": initrd,
|
||||
"kernel_params": kernel_params,
|
||||
"description": description
|
||||
}
|
||||
|
||||
|
||||
MENU_ENTRY = """
|
||||
menuentry "NixOS" {{
|
||||
loader {kernel}
|
||||
initrd {initrd}
|
||||
options "{kernel_params}"
|
||||
{submenuentries}
|
||||
}}
|
||||
"""
|
||||
|
||||
SUBMENUENTRY = """
|
||||
submenuentry "Generation {generation} {description}" {{
|
||||
loader {kernel}
|
||||
initrd {initrd}
|
||||
options "{kernel_params}"
|
||||
}}
|
||||
"""
|
||||
|
||||
|
||||
def write_refind_config(path, default_generation, generations):
|
||||
with open(path, 'w') as f:
|
||||
if "@timeout@" != "":
|
||||
f.write("timeout @timeout@")
|
||||
f.write(extra_config)
|
||||
|
||||
rev_generations = sorted(generations, key=lambda x: x[1], reverse=True)
|
||||
submenuentries = []
|
||||
max_entries = @maxEntries@
|
||||
current_entry=1
|
||||
for generation in rev_generations:
|
||||
if current_entry>max_entries:
|
||||
break
|
||||
submenuentries.append(SUBMENUENTRY.format(
|
||||
**generation_details(*generation)
|
||||
))
|
||||
current_entry+=1
|
||||
|
||||
f.write(MENU_ENTRY.format(
|
||||
**generation_details(*default_generation),
|
||||
submenuentries="\n".join(submenuentries)
|
||||
))
|
||||
|
||||
|
||||
def main():
|
||||
parser = argparse.ArgumentParser(description='Update NixOS-related systemd-boot files')
|
||||
parser.add_argument('default_config', metavar='DEFAULT-CONFIG', help='The default NixOS config to boot')
|
||||
args = parser.parse_args()
|
||||
themes = "@themes@".split()
|
||||
|
||||
mkdir_p("@efiSysMountPoint@/EFI/refind")
|
||||
mkdir_p("@efiSysMountPoint@/EFI/nixos")
|
||||
|
||||
if os.getenv("NIXOS_INSTALL_BOOTLOADER") == "1":
|
||||
|
||||
if "@canTouchEfiVariables@" == "1":
|
||||
subprocess.check_call(
|
||||
["@refind@/bin/refind-install"],
|
||||
env={"PATH": ":".join([
|
||||
"@efibootmgr@/bin",
|
||||
"@coreutils@/bin",
|
||||
"@utillinux@/bin",
|
||||
"@gnugrep@/bin",
|
||||
"@gnused@/bin",
|
||||
"@gawk@/bin",
|
||||
"@gptfdisk@/bin",
|
||||
"@findutils@/bin"
|
||||
])}
|
||||
)
|
||||
os.remove("/boot/refind_linux.conf")
|
||||
else:
|
||||
print("DONT KNOW WHAT TO DO")
|
||||
|
||||
if "@extraIcons@" != "":
|
||||
icons_dir = "@efiSysMountPoint@/EFI/refind/extra-icons"
|
||||
if os.path.exists(icons_dir):
|
||||
if os.path.exists(icons_dir + "-backup"):
|
||||
shutil.rmtree(icons_dir + "-backup")
|
||||
os.rename(icons_dir, icons_dir + "-backup")
|
||||
print("Notice: Backed up existing extra-icons directory as extra-icons-backup.")
|
||||
shutil.copytree("@extraIcons@", icons_dir)
|
||||
|
||||
if themes != []:
|
||||
themes_dir = "@efiSysMountPoint@/EFI/refind/themes"
|
||||
if os.path.exists(themes_dir):
|
||||
if os.path.exists(themes_dir + "-backup"):
|
||||
shutil.rmtree(themes_dir + "-backup")
|
||||
os.rename(themes_dir, themes_dir + "-backup")
|
||||
for path in themes:
|
||||
name = path.split("-", 1)[-1]
|
||||
shutil.copytree(path, themes_dir+"/"+name)
|
||||
else:
|
||||
print(themes)
|
||||
|
||||
generations = get_generations()
|
||||
for profile in get_profiles():
|
||||
generations += get_generations(profile)
|
||||
default_generation = None
|
||||
for generation in generations:
|
||||
if os.readlink(system_dir(*generation)) == args.default_config:
|
||||
default_generation = generation
|
||||
|
||||
write_refind_config(
|
||||
"@efiSysMountPoint@/EFI/refind/refind.conf",
|
||||
default_generation,
|
||||
generations
|
||||
)
|
||||
|
||||
# Since fat32 provides little recovery facilities after a crash,
|
||||
# it can leave the system in an unbootable state, when a crash/outage
|
||||
# happens shortly after an update. To decrease the likelihood of this
|
||||
# event sync the efi filesystem after each update.
|
||||
rc = libc.syncfs(os.open("@efiSysMountPoint@", os.O_RDONLY))
|
||||
if rc != 0:
|
||||
print("could not sync @efiSysMountPoint@: {}".format(os.strerror(rc)), file=sys.stderr)
|
||||
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
Loading…
Add table
Reference in a new issue