Improve NixOS custom user module

Now, it allows for fine-grained configuration for specific users. I also
managed to fix the infinite recursion error by directly assigning the
values to the keys instead of creating a merged module value in
`config`.
This commit is contained in:
Gabriel Arazas 2021-12-19 17:33:33 +08:00
parent 3a431615c4
commit 00e5c13cff
2 changed files with 69 additions and 40 deletions

View File

@ -28,7 +28,7 @@
}; };
editors.neovim.enable = true; editors.neovim.enable = true;
themes.a-happy-gnome.enable = true; themes.a-happy-gnome.enable = true;
users.users = [ "foo-dogsquared" ]; users.users.foo-dogsquared = {};
hardware-setup.backup-archive.enable = true; hardware-setup.backup-archive.enable = true;
}; };

View File

@ -1,60 +1,89 @@
# This enables home-manager specific configs and an easier modularization for user-specific configurations. # This enables home-manager specific configs and an easier modularization for
# user-specific configurations. This is specifically for creating a convenient
# way to create users from `users/home-manager`.
#
# If you're looking to create users from `users/nixos`, you can just import
# them directly.
{ inputs, config, options, lib, ... }: { inputs, config, options, lib, ... }:
let let
cfg = config.modules.users; cfg = config.modules.users;
userModules = lib.getUsers "home-manager" cfg.users; users = lib.attrNames cfg.users;
homeManagerUserModules = lib.getUsers "home-manager" users;
homeManagerModules = lib.filesToAttr ../home-manager; homeManagerModules = lib.filesToAttr ../home-manager;
users = lib.attrNames userModules; homeManagerUsers = lib.attrNames homeManagerUserModules;
nonexistentUsers = lib.filter (name: !lib.elem name users) cfg.users; nonexistentUsers = lib.filter (name: !lib.elem name homeManagerUsers) users;
mkUser = user: modulePath: userOption = { name, config, ... }: {
let options = {
defaultConfig = { settings = lib.mkOption {
home.username = user; type = lib.types.attrs;
home.homeDirectory = "/home/${user}"; description =
"Configuration to be merged in <literal>users.users.<name></literal> from NixOS configuration.";
xdg.enable = true; default = { };
example = {
uid = 1234;
description = "John Doe";
extraGroups = [ "wheel" "adbusers" "audio" ];
};
}; };
in {
users.users.${user} = {
isNormalUser = true;
extraGroups = [ "wheel" ];
};
home-manager.users.${user} = import modulePath;
};
in {
options.modules.users = {
users = lib.mkOption {
default = [ ];
type = with lib.types; listOf str;
description =
"A list of users from the `./users` directory to be included in the NixOS config.";
}; };
}; };
# FIXME: Recursion error when using `lib.getUsers cfg.users`. mapUsers = f: lib.mapAttrs f cfg.users;
# Time to study how Nix modules really work. in {
# The assertion is basically enough for this case. options.modules.users = {
imports = [ users = lib.mkOption {
# home-manager to enable user-specific config. default = { };
inputs.home-manager.nixosModules.home-manager description =
"A set of users from the `./users/home-manager` directory to be included in the NixOS config.
This will also create the appropriate user settings in <literal>users.users</literal> in the NixOS configuration.";
example = {
foo-dogsquared.settings = {
extraGroups = [ "wheel" "audio" "libvirtd" ];
};
alice = { };
bob = { };
};
type = with lib.types; attrsOf (submodule userOption);
};
};
# The global configuration for the home-manager module. imports = [ inputs.home-manager.nixosModules.home-manager ];
{
home-manager.useUserPackages = true;
home-manager.useGlobalPkgs = true;
home-manager.sharedModules = lib.modulesToList homeManagerModules;
}
] ++ (lib.mapAttrsToList mkUser userModules);
config = { config = {
assertions = [{ assertions = [{
assertion = (builtins.length nonexistentUsers) < 1; assertion = (builtins.length nonexistentUsers) < 1;
message = "${ message = "${
lib.concatMapStringsSep ", " (u: "'${u}'") nonexistentUsers lib.concatMapStringsSep ", " (u: "'${u}'") nonexistentUsers
} is not found in the `./users` directory."; } is not found in the `./users/home-manager` directory.";
}]; }];
# The global configuration for the home-manager module.
home-manager.useUserPackages = true;
home-manager.useGlobalPkgs = true;
home-manager.sharedModules = lib.modulesToList homeManagerModules;
# Mapping each users to the respective user configuration.
# Setting users for home-manager.
home-manager.users = mapUsers (user: _:
let
homeManagerUserModulePath = lib.getAttr user homeManagerUserModules;
homeManagerUserConfig = import homeManagerUserModulePath;
in homeManagerUserConfig);
# NixOS users.
users.users = mapUsers (user: opts:
let
defaultUserConfig = {
extraGroups = [ "wheel" ];
createHome = true;
home = "/home/${user}";
};
# TODO: Effectively override the option.
# We assume all users set with this module are normal users.
absoluteOverrides = { isNormalUser = true; };
in defaultUserConfig // opts.settings // absoluteOverrides);
}; };
} }