mirror of
https://github.com/RGBCube/hjem
synced 2025-10-13 13:12:16 +00:00
345 lines
11 KiB
Nix
345 lines
11 KiB
Nix
# The common module that contains Hjem's per-user options. To ensure Hjem remains
|
|
# somewhat compliant with cross-platform paradigms (e.g. NixOS or Darwin.) Platform
|
|
# specific options such as nixpkgs module system or nix-darwin module system should
|
|
# be avoided here.
|
|
{
|
|
config,
|
|
options,
|
|
pkgs,
|
|
lib,
|
|
...
|
|
}: let
|
|
inherit (lib.attrsets) mapAttrsToList;
|
|
inherit (lib.strings) concatLines concatMapStringsSep;
|
|
inherit (lib.modules) mkDefault mkDerivedConfig mkIf mkMerge;
|
|
inherit (lib.options) literalExpression mkEnableOption mkOption;
|
|
inherit (lib.strings) hasPrefix;
|
|
inherit (lib.types) addCheck anything attrsOf bool either functionTo int lines listOf nullOr package path str submodule oneOf;
|
|
inherit (builtins) isList;
|
|
|
|
cfg = config;
|
|
|
|
fileType = relativeTo:
|
|
submodule ({
|
|
name,
|
|
target,
|
|
config,
|
|
options,
|
|
...
|
|
}: {
|
|
options = {
|
|
enable =
|
|
mkEnableOption "creation of this file"
|
|
// {
|
|
default = true;
|
|
example = false;
|
|
};
|
|
|
|
target = mkOption {
|
|
type = str;
|
|
apply = p:
|
|
if hasPrefix "/" p
|
|
then throw "This option cannot handle absolute paths yet!"
|
|
else "${config.relativeTo}/${p}";
|
|
defaultText = "name";
|
|
description = ''
|
|
Path to target file relative to {option}`hjem.users.<name>.files.<file>.relativeTo`.
|
|
'';
|
|
};
|
|
|
|
text = mkOption {
|
|
default = null;
|
|
type = nullOr lines;
|
|
description = "Text of the file";
|
|
};
|
|
|
|
source = mkOption {
|
|
type = nullOr path;
|
|
default = null;
|
|
description = "Path of the source file or directory";
|
|
};
|
|
|
|
generator = lib.mkOption {
|
|
# functionTo doesn't actually check the return type, so do that ourselves
|
|
type = addCheck (nullOr (functionTo (either options.source.type options.text.type))) (x: let
|
|
generatedValue = x config.value;
|
|
generatesDrv = options.source.type.check generatedValue;
|
|
generatesStr = options.text.type.check generatedValue;
|
|
in
|
|
x != null -> (generatesDrv || generatesStr));
|
|
default = null;
|
|
description = ''
|
|
Function that when applied to `value` will create the `source` or `text` of the file.
|
|
|
|
Detection is automatic, as we check if the `generator` generates a derivation or a string after applying to `value`.
|
|
'';
|
|
example = literalExpression "lib.generators.toGitINI";
|
|
};
|
|
|
|
value = lib.mkOption {
|
|
type = nullOr (attrsOf anything);
|
|
default = null;
|
|
description = "Value passed to the `generator`.";
|
|
example = {
|
|
user.email = "me@example.com";
|
|
};
|
|
};
|
|
|
|
executable = mkOption {
|
|
type = bool;
|
|
default = false;
|
|
example = true;
|
|
description = ''
|
|
Whether to set the execute bit on the target file.
|
|
'';
|
|
};
|
|
|
|
clobber = mkOption {
|
|
type = bool;
|
|
default = cfg.clobberFiles;
|
|
defaultText = literalExpression "config.hjem.clobberByDefault";
|
|
description = ''
|
|
Whether to "clobber" existing target paths.
|
|
|
|
- If using the **systemd-tmpfiles** hook (Linux only), tmpfile rules
|
|
will be constructed with `L+` (*re*create) instead of `L`
|
|
(create) type while this is set to `true`.
|
|
'';
|
|
};
|
|
|
|
relativeTo = mkOption {
|
|
internal = true;
|
|
type = path;
|
|
default = relativeTo;
|
|
description = "Path to which symlinks will be relative to";
|
|
apply = x:
|
|
assert (hasPrefix "/" x || abort "Relative path ${x} cannot be used for files.<file>.relativeTo"); x;
|
|
};
|
|
};
|
|
|
|
config = let
|
|
generatedValue = config.generator config.value;
|
|
hasGenerator = config.generator != null;
|
|
generatesDrv = options.source.type.check generatedValue;
|
|
generatesStr = options.text.type.check generatedValue;
|
|
in
|
|
mkMerge [
|
|
{
|
|
target = mkDefault name;
|
|
source = mkIf (config.text != null) (mkDerivedConfig options.text (text:
|
|
pkgs.writeTextFile {
|
|
inherit name text;
|
|
inherit (config) executable;
|
|
}));
|
|
}
|
|
|
|
(lib.mkIf (hasGenerator && generatesDrv) {
|
|
source = mkDefault generatedValue;
|
|
})
|
|
|
|
(lib.mkIf (hasGenerator && generatesStr) {
|
|
text = mkDefault generatedValue;
|
|
})
|
|
];
|
|
});
|
|
in {
|
|
imports = [
|
|
# Makes "assertions" option available without having to duplicate the work
|
|
# already done in the Nixpkgs module.
|
|
(pkgs.path + "/nixos/modules/misc/assertions.nix")
|
|
];
|
|
|
|
options = {
|
|
enable =
|
|
mkEnableOption "home management for this user"
|
|
// {
|
|
default = true;
|
|
example = false;
|
|
};
|
|
|
|
user = mkOption {
|
|
type = str;
|
|
description = "The owner of a given home directory.";
|
|
};
|
|
|
|
directory = mkOption {
|
|
type = path;
|
|
description = ''
|
|
The home directory for the user, to which files configured in
|
|
{option}`hjem.users.<name>.files` will be relative to by default.
|
|
'';
|
|
};
|
|
|
|
clobberFiles = mkOption {
|
|
type = bool;
|
|
example = true;
|
|
description = ''
|
|
The default override behaviour for files managed by Hjem for a
|
|
particular user.
|
|
|
|
A top level option exists under the Hjem module option
|
|
{option}`hjem.clobberByDefault`. Per-file behaviour can be modified
|
|
with {option}`hjem.users.<name>.files.<file>.clobber`.
|
|
'';
|
|
};
|
|
|
|
files = mkOption {
|
|
default = {};
|
|
type = attrsOf (fileType cfg.directory);
|
|
example = {".config/foo.txt".source = "Hello World";};
|
|
description = "Files to be managed by Hjem";
|
|
};
|
|
|
|
xdg = {
|
|
cache = {
|
|
directory = mkOption {
|
|
type = path;
|
|
default = "${cfg.directory}/.cache";
|
|
defaultText = "$HOME/.cache";
|
|
description = ''
|
|
The XDG cache directory for the user, to which files configured in
|
|
{option}`hjem.users.<name>.xdg.cache.files` will be relative to by default.
|
|
|
|
Adds {env}`XDG_CACHE_HOME` to {option}`environment.sessionVariables` for
|
|
this user if changed.
|
|
'';
|
|
};
|
|
files = mkOption {
|
|
default = {};
|
|
type = attrsOf (fileType cfg.xdg.cache.directory);
|
|
example = {"foo.txt".source = "Hello World";};
|
|
description = "Cache files to be managed by Hjem";
|
|
};
|
|
};
|
|
|
|
config = {
|
|
directory = mkOption {
|
|
type = path;
|
|
default = "${cfg.directory}/.config";
|
|
defaultText = "$HOME/.config";
|
|
description = ''
|
|
The XDG config directory for the user, to which files configured in
|
|
{option}`hjem.users.<name>.xdg.config.files` will be relative to by default.
|
|
|
|
Adds {env}`XDG_CONFIG_HOME` to {option}`environment.sessionVariables` for
|
|
this user if changed.
|
|
'';
|
|
};
|
|
files = mkOption {
|
|
default = {};
|
|
type = attrsOf (fileType cfg.xdg.config.directory);
|
|
example = {"foo.txt".source = "Hello World";};
|
|
description = "Config files to be managed by Hjem";
|
|
};
|
|
};
|
|
|
|
data = {
|
|
directory = mkOption {
|
|
type = path;
|
|
default = "${cfg.directory}/.local/share";
|
|
defaultText = "$HOME/.local/share";
|
|
description = ''
|
|
The XDG data directory for the user, to which files configured in
|
|
{option}`hjem.users.<name>.xdg.data.files` will be relative to by default.
|
|
|
|
Adds {env}`XDG_DATA_HOME` to {option}`environment.sessionVariables` for
|
|
this user if changed.
|
|
'';
|
|
};
|
|
files = mkOption {
|
|
default = {};
|
|
type = attrsOf (fileType cfg.xdg.data.directory);
|
|
example = {"foo.txt".source = "Hello World";};
|
|
description = "data files to be managed by Hjem";
|
|
};
|
|
};
|
|
|
|
state = {
|
|
directory = mkOption {
|
|
type = path;
|
|
default = "${cfg.directory}/.local/state";
|
|
defaultText = "$HOME/.local/share";
|
|
description = ''
|
|
The XDG state directory for the user, to which files configured in
|
|
{option}`hjem.users.<name>.xdg.state.files` will be relative to by default.
|
|
|
|
Adds {env}`XDG_STATE_HOME` to {option}`environment.sessionVariables` for
|
|
this user if changed.
|
|
'';
|
|
};
|
|
files = mkOption {
|
|
default = {};
|
|
type = attrsOf (fileType cfg.xdg.state.directory);
|
|
example = {"foo.txt".source = "Hello World";};
|
|
description = "state files to be managed by Hjem";
|
|
};
|
|
};
|
|
};
|
|
|
|
packages = mkOption {
|
|
type = listOf package;
|
|
default = [];
|
|
example = literalExpression "[pkgs.hello]";
|
|
description = "Packages to install for this user";
|
|
};
|
|
|
|
environment = {
|
|
loadEnv = mkOption {
|
|
type = path;
|
|
readOnly = true;
|
|
description = ''
|
|
A POSIX compliant shell script containing the user session variables needed to bootstrap the session.
|
|
|
|
As there is no reliable and agnostic way of setting session variables, Hjem's
|
|
environment module does nothing by itself. Rather, it provides a POSIX compliant shell script
|
|
that needs to be sourced where needed.
|
|
'';
|
|
};
|
|
sessionVariables = mkOption {
|
|
type = attrsOf (oneOf [(listOf (oneOf [int str path])) int str path]);
|
|
default = {};
|
|
example = {
|
|
EDITOR = "nvim";
|
|
VISUAL = "nvim";
|
|
};
|
|
description = ''
|
|
A set of environment variables used in the user environment.
|
|
If a list of strings is used, they will be concatenated with colon
|
|
characters.
|
|
'';
|
|
};
|
|
};
|
|
};
|
|
|
|
config = {
|
|
environment = {
|
|
sessionVariables = {
|
|
XDG_CACHE_HOME = mkIf (cfg.xdg.cache.directory != options.xdg.cache.directory.default) cfg.xdg.cache.directory;
|
|
XDG_CONFIG_HOME = mkIf (cfg.xdg.config.directory != options.xdg.config.directory.default) cfg.xdg.config.directory;
|
|
XDG_DATA_HOME = mkIf (cfg.xdg.data.directory != options.xdg.data.directory.default) cfg.xdg.data.directory;
|
|
XDG_STATE_HOME = mkIf (cfg.xdg.state.directory != options.xdg.state.directory.default) cfg.xdg.state.directory;
|
|
};
|
|
loadEnv = let
|
|
toEnv = env:
|
|
if isList env
|
|
then concatMapStringsSep ":" toString env
|
|
else toString env;
|
|
in
|
|
lib.pipe cfg.environment.sessionVariables [
|
|
(mapAttrsToList (name: value: "export ${name}=\"${toEnv value}\""))
|
|
concatLines
|
|
(pkgs.writeShellScript "load-env")
|
|
];
|
|
};
|
|
assertions = [
|
|
{
|
|
assertion = cfg.user != "";
|
|
message = "A user must be configured in 'hjem.users.<user>.name'";
|
|
}
|
|
{
|
|
assertion = cfg.directory != "";
|
|
message = "A home directory must be configured in 'hjem.users.<user>.directory'";
|
|
}
|
|
];
|
|
};
|
|
}
|