What’s a good easy way to conditionally import NixOS modules?
In my configuration I’ve got lots of little fairly self-contained NixOS modules which do something like:
- install and configure several system tools
- configure a postgresql server for development use
- setup a basic desktop environment
- … et cetera.
I want to enable and disable these modules on various machines at various times.
The standard way to do this would be to have each machine’s NixOS configuration import just the modules that it needs.
But at this point, I’ve got so many modules that I also want to make groups of modules which can be enabled together (while still retaining the ability to control individual modules). It’s getting messy.
So the standard way to solve this is to define my own NixOS options which enable individual groups and sub-groups of configuration. I have done this a little bit and it helps.
It’s getting tedious, however. Many modules would take the form:
# ./modules/this.nix
{ lib, config, pkgs, ... }: {
options.me.install.this = lib.mkEnableOption "install and configure this";
config = lib.mkIf config.me.install.this {
# ...
};
}
I really just want something like:
# ./modules/wrong.nix
{ config, lib, ... }: {
options.me = with lib; {
install = {
system-tools = mkEnableOption "install and configure basic system tools";
dev-db = mkEnableOption "setup postgresql for development";
desktop = mkEnableOption "desktop environment configured how i like it";
# etc
};
tasks = {
dev = mkEnableOption "dev stuff relevant to me";
# etc
};
};
config = lib.mkIf config.me.tasks.dev {
me.install.dev-db = lib.mkDefault true;
me.install.compiler-and-tools = lib.mkDefault true;
me.install.emacs = lib.mkDefault true;
};
imports = [ ./modules/base.nix ]
++ lib.optional config.me.install.system-tools ./modules/system-tools.nix
++ lib.optional config.me.install.dev-db ./modules/dev-db.nix
++ lib.optional config.me.install.desktop ./modules/desktop.nix;
}
But as we all know, module imports
cannot depend on config
, because that would cause infinite recursion.
What I’m trying now is to generate modules which wrap other modules with a mkEnableOption
conditional. It seems to work, but I haven’t tested it extensively.
Here is an example. It will import all modules ./fragments/*.nix
and create me.fragments.*.enable
options for them.
# flake.nix
{
description = "Ad-hoc config snippets";
outputs = { self, nixpkgs }: let
inherit (nixpkgs) lib;
fragmentModules = attrPath: dir: extraSpecialArgs:
lib.mapAttrsToList
(makeFragmentModule attrPath extraSpecialArgs)
(dirAttrs dir);
makeFragmentModule = attrPath: extraSpecialArgs: name: path: let
enableAttrPath = attrPath ++ [ name "enable" ];
in args@{ config, ... }: {
_file = path;
key = path;
options = lib.setAttrByPath enableAttrPath (lib.mkEnableOption "${name} fragment");
config = lib.mkIf
(lib.getAttrFromPath enableAttrPath config)
(lib.applyModuleArgsIfFunction "fixme" (import path) (args // extraSpecialArgs));
};
dirAttrs = path: lib.mapAttrs'
(name: _: nixFileAttrInPath path name)
(lib.filterAttrs
(name: type: lib.hasSuffix ".nix" name && type == "regular")
(builtins.readDir path));
nixFileAttrInPath = path: name: {
name = nixFileAttrName name;
value = path + "/${name}";
};
nixFileAttrName = path: lib.removeSuffix ".nix" (builtins.baseNameOf path);
in {
nixosModules.default = {
imports = fragmentModules ["me" "fragments"] ./fragments { flake = self; };
options.me.tasks = with lib; {
dev = mkEnableOption "dev stuff relevant to me";
laptop = mkEnableOption "it's a computer with a gui";
};
config = lib.mkMerge [
(lib.mkIf config.me.tasks.dev {
me.fragments.dev-db.enable = lib.mkDefault true;
me.fragments.compiler-and-tools.enable = lib.mkDefault true;
me.fragments.emacs.enable = lib.mkDefault true;
})
(lib.mkIf config.me.tasks.laptop {
me.fragments.system-tools.enable = lib.mkDefault true;
me.fragments.desktop.enable = lib.mkDefault true;
})
];
};
nixosModules.tethys = {
me.tasks.laptop = true;
me.tasks.dev = true;
};
nixosModules.server = {
me.tasks.dev = true;
me.tasks.web = true;
me.fragments.system-tools.enable = true;
me.fragments.dev-db.enable = false;
};
nixosConfigurations = {
tethys = nixpkgs.lib.nixosSystem {
system = "x86_64-linux";
modules = [
./tethys/hardware-configuration.nix
self.nixosModules.default
self.nixosModules.tethys
];
};
server = nixpkgs.lib.nixosSystem {
system = "x86_64-linux";
modules = [
./server/hardware-configuration.nix
self.nixosModules.default
self.nixosModules.server
];
};
};
};
}
Surely someone is doing something like this already. Maybe I have overlooked something in the NixOS documentation. Perhaps there is a nice little library I can import?