2017-10-14 18:56:02 +00:00
|
|
|
{ config, lib, pkgs, utils, ... }:
|
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
|
|
|
|
cfg = config.home-manager;
|
|
|
|
|
2020-01-16 22:41:14 +00:00
|
|
|
extendedLib = import ../modules/lib/stdlib-extended.nix pkgs.lib;
|
|
|
|
|
|
|
|
hmModule = types.submoduleWith {
|
2020-07-09 23:47:16 +00:00
|
|
|
specialArgs = {
|
|
|
|
lib = extendedLib;
|
|
|
|
nixosConfig = config;
|
2021-10-06 14:32:13 +00:00
|
|
|
osConfig = config;
|
2021-09-29 07:14:05 +00:00
|
|
|
modulesPath = ../modules;
|
2021-02-21 06:02:25 +00:00
|
|
|
} // cfg.extraSpecialArgs;
|
2020-02-01 23:39:17 +00:00
|
|
|
modules = [
|
|
|
|
({ name, ... }: {
|
2020-01-16 22:41:14 +00:00
|
|
|
imports = import ../modules/modules.nix {
|
|
|
|
inherit pkgs;
|
|
|
|
lib = extendedLib;
|
2020-03-01 16:06:20 +00:00
|
|
|
useNixpkgsModule = !cfg.useGlobalPkgs;
|
2020-01-16 22:41:14 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
config = {
|
|
|
|
submoduleSupport.enable = true;
|
|
|
|
submoduleSupport.externalPackageInstall = cfg.useUserPackages;
|
|
|
|
|
|
|
|
# The per-user directory inside /etc/profiles is not known by
|
|
|
|
# fontconfig by default.
|
2020-02-01 23:39:17 +00:00
|
|
|
fonts.fontconfig.enable = cfg.useUserPackages
|
|
|
|
&& config.fonts.fontconfig.enable;
|
2020-01-16 22:41:14 +00:00
|
|
|
|
|
|
|
home.username = config.users.users.${name}.name;
|
|
|
|
home.homeDirectory = config.users.users.${name}.home;
|
2021-07-19 21:07:02 +00:00
|
|
|
|
|
|
|
# Make activation script use same version of Nix as system as a whole.
|
|
|
|
# This avoids problems with Nix not being in PATH.
|
|
|
|
home.extraActivationPath = [ config.nix.package ];
|
2020-01-16 22:41:14 +00:00
|
|
|
};
|
2020-02-01 23:39:17 +00:00
|
|
|
})
|
2021-02-21 06:02:25 +00:00
|
|
|
] ++ cfg.sharedModules;
|
2020-01-16 22:41:14 +00:00
|
|
|
};
|
2017-10-14 18:56:02 +00:00
|
|
|
|
2020-02-01 23:39:17 +00:00
|
|
|
serviceEnvironment = optionalAttrs (cfg.backupFileExtension != null) {
|
|
|
|
HOME_MANAGER_BACKUP_EXT = cfg.backupFileExtension;
|
|
|
|
} // optionalAttrs cfg.verbose { VERBOSE = "1"; };
|
2017-10-14 18:56:02 +00:00
|
|
|
|
2020-02-01 23:39:17 +00:00
|
|
|
in {
|
2017-10-14 18:56:02 +00:00
|
|
|
options = {
|
2017-12-19 14:43:40 +00:00
|
|
|
home-manager = {
|
|
|
|
useUserPackages = mkEnableOption ''
|
|
|
|
installation of user packages through the
|
2021-01-18 22:21:32 +00:00
|
|
|
<option>users.users.<name>.packages</option> option.
|
2017-10-14 18:56:02 +00:00
|
|
|
'';
|
2017-12-19 14:43:40 +00:00
|
|
|
|
2020-03-01 16:06:20 +00:00
|
|
|
useGlobalPkgs = mkEnableOption ''
|
|
|
|
using the system configuration's <literal>pkgs</literal>
|
|
|
|
argument in Home Manager. This disables the Home Manager
|
|
|
|
options <option>nixpkgs.*</option>
|
|
|
|
'';
|
|
|
|
|
2019-08-08 16:24:08 +00:00
|
|
|
backupFileExtension = mkOption {
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
default = null;
|
|
|
|
example = "backup";
|
|
|
|
description = ''
|
|
|
|
On activation move existing files by appending the given
|
|
|
|
file extension rather than exiting with an error.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2021-02-21 06:02:25 +00:00
|
|
|
extraSpecialArgs = mkOption {
|
|
|
|
type = types.attrs;
|
|
|
|
default = { };
|
|
|
|
description = ''
|
|
|
|
Extra <literal>specialArgs</literal> passed to Home Manager.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
sharedModules = mkOption {
|
2021-03-26 02:01:16 +00:00
|
|
|
type = with types;
|
|
|
|
listOf (anything // {
|
|
|
|
inherit (submodule { }) check;
|
|
|
|
description = "Home Manager modules";
|
|
|
|
});
|
2021-02-21 06:02:25 +00:00
|
|
|
default = [ ];
|
|
|
|
example = literalExample "[ { home.packages = [ nixpkgs-fmt ]; } ]";
|
|
|
|
description = ''
|
|
|
|
Extra modules added to all users.
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
2019-08-08 16:24:08 +00:00
|
|
|
verbose = mkEnableOption "verbose output on activation";
|
|
|
|
|
2017-12-19 14:43:40 +00:00
|
|
|
users = mkOption {
|
|
|
|
type = types.attrsOf hmModule;
|
2020-02-01 23:39:17 +00:00
|
|
|
default = { };
|
2021-01-18 22:21:32 +00:00
|
|
|
# Set as not visible to prevent the entire submodule being included in
|
|
|
|
# the documentation.
|
|
|
|
visible = false;
|
2017-12-19 14:43:40 +00:00
|
|
|
description = ''
|
|
|
|
Per-user Home Manager configuration.
|
|
|
|
'';
|
|
|
|
};
|
2017-10-14 18:56:02 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2020-02-01 23:39:17 +00:00
|
|
|
config = mkIf (cfg.users != { }) {
|
|
|
|
warnings = flatten (flip mapAttrsToList cfg.users (user: config:
|
|
|
|
flip map config.warnings (warning: "${user} profile: ${warning}")));
|
|
|
|
|
|
|
|
assertions = flatten (flip mapAttrsToList cfg.users (user: config:
|
|
|
|
flip map config.assertions (assertion: {
|
|
|
|
inherit (assertion) assertion;
|
|
|
|
message = "${user} profile: ${assertion.message}";
|
|
|
|
})));
|
|
|
|
|
|
|
|
users.users = mkIf cfg.useUserPackages
|
2020-06-18 22:21:12 +00:00
|
|
|
(mapAttrs (username: usercfg: { packages = [ usercfg.home.path ]; })
|
2020-02-01 23:39:17 +00:00
|
|
|
cfg.users);
|
2017-12-19 14:43:40 +00:00
|
|
|
|
2020-06-17 21:33:13 +00:00
|
|
|
environment.pathsToLink = mkIf cfg.useUserPackages [ "/etc/profile.d" ];
|
|
|
|
|
2019-05-24 07:17:25 +00:00
|
|
|
systemd.services = mapAttrs' (_: usercfg:
|
2020-02-01 23:39:17 +00:00
|
|
|
let username = usercfg.home.username;
|
|
|
|
in nameValuePair ("home-manager-${utils.escapeSystemdPath username}") {
|
|
|
|
description = "Home Manager environment for ${username}";
|
|
|
|
wantedBy = [ "multi-user.target" ];
|
|
|
|
wants = [ "nix-daemon.socket" ];
|
|
|
|
after = [ "nix-daemon.socket" ];
|
2021-08-22 02:14:40 +00:00
|
|
|
before = [ "systemd-user-sessions.service" ];
|
2020-02-01 23:39:17 +00:00
|
|
|
|
|
|
|
environment = serviceEnvironment;
|
|
|
|
|
2020-05-25 09:16:57 +00:00
|
|
|
unitConfig = { RequiresMountsFor = usercfg.home.homeDirectory; };
|
|
|
|
|
2021-06-14 17:49:38 +00:00
|
|
|
stopIfChanged = false;
|
|
|
|
|
2020-02-01 23:39:17 +00:00
|
|
|
serviceConfig = {
|
|
|
|
User = usercfg.home.username;
|
|
|
|
Type = "oneshot";
|
|
|
|
RemainAfterExit = "yes";
|
2021-09-02 03:08:55 +00:00
|
|
|
TimeoutStartSec = 90;
|
2020-02-01 23:39:17 +00:00
|
|
|
SyslogIdentifier = "hm-activate-${username}";
|
|
|
|
|
2021-06-18 09:52:20 +00:00
|
|
|
ExecStart = let
|
|
|
|
systemctl =
|
|
|
|
"XDG_RUNTIME_DIR=\${XDG_RUNTIME_DIR:-/run/user/$UID} systemctl";
|
|
|
|
|
|
|
|
sed = "${pkgs.gnused}/bin/sed";
|
|
|
|
|
|
|
|
exportedSystemdVariables = concatStringsSep "|" [
|
|
|
|
"DBUS_SESSION_BUS_ADDRESS"
|
|
|
|
"DISPLAY"
|
|
|
|
"WAYLAND_DISPLAY"
|
|
|
|
"XAUTHORITY"
|
|
|
|
"XDG_RUNTIME_DIR"
|
|
|
|
];
|
|
|
|
|
|
|
|
setupEnv = pkgs.writeScript "hm-setup-env" ''
|
|
|
|
#! ${pkgs.runtimeShell} -el
|
|
|
|
|
|
|
|
# The activation script is run by a login shell to make sure
|
|
|
|
# that the user is given a sane environment.
|
|
|
|
# If the user is logged in, import variables from their current
|
|
|
|
# session environment.
|
|
|
|
eval "$(
|
|
|
|
${systemctl} --user show-environment 2> /dev/null \
|
|
|
|
| ${sed} -En '/^(${exportedSystemdVariables})=/s/^/export /p'
|
|
|
|
)"
|
|
|
|
|
|
|
|
exec "$1/activate"
|
|
|
|
'';
|
|
|
|
in "${setupEnv} ${usercfg.home.activationPackage}";
|
2020-02-01 23:39:17 +00:00
|
|
|
};
|
|
|
|
}) cfg.users;
|
2017-10-14 18:56:02 +00:00
|
|
|
};
|
|
|
|
}
|