2013-01-01 14:42:43 +01:00
|
|
|
|
{ config, pkgs, utils, ... }:
|
2012-06-02 02:15:07 +02:00
|
|
|
|
|
|
|
|
|
with pkgs.lib;
|
2013-01-01 14:42:43 +01:00
|
|
|
|
with utils;
|
2012-08-06 17:45:59 +02:00
|
|
|
|
with import ./systemd-unit-options.nix { inherit config pkgs; };
|
2012-06-02 02:15:07 +02:00
|
|
|
|
|
|
|
|
|
let
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
cfg = config.systemd;
|
2012-06-18 05:31:21 +02:00
|
|
|
|
|
2013-01-16 13:17:57 +01:00
|
|
|
|
systemd = cfg.package;
|
2012-06-02 02:15:07 +02:00
|
|
|
|
|
2012-06-18 05:31:21 +02:00
|
|
|
|
makeUnit = name: unit:
|
2014-01-18 17:10:39 +01:00
|
|
|
|
pkgs.runCommand "unit" { preferLocalBuild = true; inherit (unit) text; }
|
|
|
|
|
((if !unit.enable then ''
|
2012-10-29 21:01:36 +01:00
|
|
|
|
mkdir -p $out
|
2014-01-15 12:58:16 +01:00
|
|
|
|
ln -s /dev/null $out/${name}
|
|
|
|
|
'' else if unit.linkTarget != null then ''
|
|
|
|
|
mkdir -p $out
|
|
|
|
|
ln -s ${unit.linkTarget} $out/${name}
|
2014-01-18 17:10:39 +01:00
|
|
|
|
'' else if unit.text != null then ''
|
2012-10-29 21:01:36 +01:00
|
|
|
|
mkdir -p $out
|
2014-01-15 12:58:16 +01:00
|
|
|
|
echo -n "$text" > $out/${name}
|
2014-01-18 17:10:39 +01:00
|
|
|
|
'' else "") + optionalString (unit.extraConfig != {}) ''
|
|
|
|
|
mkdir -p $out/${name}.d
|
|
|
|
|
${concatStringsSep "\n" (mapAttrsToList (n: v: "echo -n \"${v}\" > $out/${name}.d/${n}") unit.extraConfig)}
|
2012-10-29 21:01:36 +01:00
|
|
|
|
'');
|
2012-06-02 02:15:07 +02:00
|
|
|
|
|
|
|
|
|
upstreamUnits =
|
2012-06-15 00:44:56 +02:00
|
|
|
|
[ # Targets.
|
2012-06-02 02:15:07 +02:00
|
|
|
|
"basic.target"
|
2013-01-08 17:26:51 +01:00
|
|
|
|
"sysinit.target"
|
2012-06-02 02:15:07 +02:00
|
|
|
|
"sockets.target"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
"graphical.target"
|
|
|
|
|
"multi-user.target"
|
|
|
|
|
"getty.target"
|
|
|
|
|
"network.target"
|
2013-07-16 11:55:12 +02:00
|
|
|
|
"network-online.target"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
"nss-lookup.target"
|
|
|
|
|
"nss-user-lookup.target"
|
|
|
|
|
"time-sync.target"
|
2012-08-11 00:56:12 +02:00
|
|
|
|
#"cryptsetup.target"
|
2012-08-15 00:14:48 +02:00
|
|
|
|
"sigpwr.target"
|
2013-03-27 13:58:12 +01:00
|
|
|
|
"timers.target"
|
|
|
|
|
"paths.target"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
|
2013-01-21 21:01:48 +01:00
|
|
|
|
# Rescue mode.
|
2013-01-08 18:24:06 +01:00
|
|
|
|
"rescue.target"
|
|
|
|
|
"rescue.service"
|
|
|
|
|
|
2012-06-15 19:09:22 +02:00
|
|
|
|
# Udev.
|
2012-07-16 23:47:11 +02:00
|
|
|
|
"systemd-udevd-control.socket"
|
|
|
|
|
"systemd-udevd-kernel.socket"
|
|
|
|
|
"systemd-udevd.service"
|
2012-06-15 19:09:22 +02:00
|
|
|
|
"systemd-udev-settle.service"
|
|
|
|
|
"systemd-udev-trigger.service"
|
|
|
|
|
|
2012-08-15 00:14:48 +02:00
|
|
|
|
# Hardware (started by udev when a relevant device is plugged in).
|
|
|
|
|
"sound.target"
|
|
|
|
|
"bluetooth.target"
|
|
|
|
|
"printer.target"
|
|
|
|
|
"smartcard.target"
|
|
|
|
|
|
2012-06-15 00:44:56 +02:00
|
|
|
|
# Login stuff.
|
|
|
|
|
"systemd-logind.service"
|
|
|
|
|
"autovt@.service"
|
2012-10-18 17:54:07 +02:00
|
|
|
|
#"systemd-vconsole-setup.service"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
"systemd-user-sessions.service"
|
2012-06-15 19:09:22 +02:00
|
|
|
|
"dbus-org.freedesktop.login1.service"
|
2012-06-15 20:51:48 +02:00
|
|
|
|
"user@.service"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
|
|
|
|
|
# Journal.
|
|
|
|
|
"systemd-journald.socket"
|
|
|
|
|
"systemd-journald.service"
|
2012-07-20 18:02:42 +02:00
|
|
|
|
"systemd-journal-flush.service"
|
2012-07-16 23:47:11 +02:00
|
|
|
|
"syslog.socket"
|
2012-06-02 02:15:07 +02:00
|
|
|
|
|
2012-06-15 00:44:56 +02:00
|
|
|
|
# SysV init compatibility.
|
|
|
|
|
"systemd-initctl.socket"
|
|
|
|
|
"systemd-initctl.service"
|
|
|
|
|
|
2012-08-06 22:52:08 +02:00
|
|
|
|
# Kernel module loading.
|
2012-10-11 23:58:46 +02:00
|
|
|
|
#"systemd-modules-load.service"
|
2012-08-06 22:52:08 +02:00
|
|
|
|
|
2012-06-02 02:15:07 +02:00
|
|
|
|
# Filesystems.
|
2012-07-16 23:47:11 +02:00
|
|
|
|
"systemd-fsck@.service"
|
|
|
|
|
"systemd-fsck-root.service"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
"systemd-remount-fs.service"
|
2012-06-02 02:15:07 +02:00
|
|
|
|
"local-fs.target"
|
|
|
|
|
"local-fs-pre.target"
|
|
|
|
|
"remote-fs.target"
|
|
|
|
|
"remote-fs-pre.target"
|
|
|
|
|
"swap.target"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
"dev-hugepages.mount"
|
2012-06-02 02:15:07 +02:00
|
|
|
|
"dev-mqueue.mount"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
"sys-fs-fuse-connections.mount"
|
|
|
|
|
"sys-kernel-config.mount"
|
|
|
|
|
"sys-kernel-debug.mount"
|
|
|
|
|
|
|
|
|
|
# Hibernate / suspend.
|
|
|
|
|
"hibernate.target"
|
|
|
|
|
"suspend.target"
|
|
|
|
|
"sleep.target"
|
2013-03-27 13:58:12 +01:00
|
|
|
|
"hybrid-sleep.target"
|
2012-07-20 21:40:50 +02:00
|
|
|
|
"systemd-hibernate.service"
|
|
|
|
|
"systemd-suspend.service"
|
2013-03-27 13:58:12 +01:00
|
|
|
|
"systemd-hybrid-sleep.service"
|
2012-07-20 21:40:50 +02:00
|
|
|
|
"systemd-shutdownd.socket"
|
|
|
|
|
"systemd-shutdownd.service"
|
2012-06-02 02:15:07 +02:00
|
|
|
|
|
|
|
|
|
# Reboot stuff.
|
|
|
|
|
"reboot.target"
|
2012-07-16 23:47:11 +02:00
|
|
|
|
"systemd-reboot.service"
|
2012-06-02 02:15:07 +02:00
|
|
|
|
"poweroff.target"
|
2012-07-16 23:47:11 +02:00
|
|
|
|
"systemd-poweroff.service"
|
2012-06-02 02:15:07 +02:00
|
|
|
|
"halt.target"
|
2012-07-16 23:47:11 +02:00
|
|
|
|
"systemd-halt.service"
|
2012-06-02 02:15:07 +02:00
|
|
|
|
"ctrl-alt-del.target"
|
|
|
|
|
"shutdown.target"
|
|
|
|
|
"umount.target"
|
|
|
|
|
"final.target"
|
2012-08-15 00:14:48 +02:00
|
|
|
|
"kexec.target"
|
2013-09-16 17:15:42 +02:00
|
|
|
|
"systemd-kexec.service"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
|
2012-06-19 23:02:54 +02:00
|
|
|
|
# Password entry.
|
|
|
|
|
"systemd-ask-password-console.path"
|
|
|
|
|
"systemd-ask-password-console.service"
|
|
|
|
|
"systemd-ask-password-wall.path"
|
|
|
|
|
"systemd-ask-password-wall.service"
|
2013-01-21 21:01:48 +01:00
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
++ optionals cfg.enableEmergencyMode [
|
|
|
|
|
"emergency.target"
|
|
|
|
|
"emergency.service"
|
2012-06-02 02:15:07 +02:00
|
|
|
|
];
|
|
|
|
|
|
2012-06-15 00:44:56 +02:00
|
|
|
|
upstreamWants =
|
2013-03-27 13:58:12 +01:00
|
|
|
|
[ #"basic.target.wants"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
"sysinit.target.wants"
|
|
|
|
|
"sockets.target.wants"
|
|
|
|
|
"local-fs.target.wants"
|
|
|
|
|
"multi-user.target.wants"
|
|
|
|
|
"shutdown.target.wants"
|
2013-03-27 13:58:12 +01:00
|
|
|
|
"timers.target.wants"
|
2012-06-15 00:44:56 +02:00
|
|
|
|
];
|
|
|
|
|
|
2012-10-15 22:01:30 +02:00
|
|
|
|
makeJobScript = name: text:
|
|
|
|
|
let x = pkgs.writeTextFile { name = "unit-script"; executable = true; destination = "/bin/${name}"; inherit text; };
|
|
|
|
|
in "${x}/bin/${name}";
|
2012-07-19 23:41:42 +02:00
|
|
|
|
|
2012-10-09 21:14:15 +02:00
|
|
|
|
unitConfig = { name, config, ... }: {
|
|
|
|
|
config = {
|
|
|
|
|
unitConfig =
|
|
|
|
|
{ Requires = concatStringsSep " " config.requires;
|
|
|
|
|
Wants = concatStringsSep " " config.wants;
|
|
|
|
|
After = concatStringsSep " " config.after;
|
|
|
|
|
Before = concatStringsSep " " config.before;
|
2012-10-10 22:50:41 +02:00
|
|
|
|
BindsTo = concatStringsSep " " config.bindsTo;
|
2012-10-09 21:14:15 +02:00
|
|
|
|
PartOf = concatStringsSep " " config.partOf;
|
2013-09-22 21:04:54 +02:00
|
|
|
|
Conflicts = concatStringsSep " " config.conflicts;
|
2012-10-11 23:54:43 +02:00
|
|
|
|
"X-Restart-Triggers" = toString config.restartTriggers;
|
2012-10-09 21:14:15 +02:00
|
|
|
|
} // optionalAttrs (config.description != "") {
|
|
|
|
|
Description = config.description;
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
2012-08-06 17:45:59 +02:00
|
|
|
|
serviceConfig = { name, config, ... }: {
|
2013-11-18 15:45:24 +01:00
|
|
|
|
config = mkMerge
|
|
|
|
|
[ { # Default path for systemd services. Should be quite minimal.
|
|
|
|
|
path =
|
|
|
|
|
[ pkgs.coreutils
|
|
|
|
|
pkgs.findutils
|
|
|
|
|
pkgs.gnugrep
|
|
|
|
|
pkgs.gnused
|
|
|
|
|
systemd
|
|
|
|
|
];
|
|
|
|
|
environment.PATH = config.path;
|
|
|
|
|
}
|
|
|
|
|
(mkIf (config.preStart != "")
|
|
|
|
|
{ serviceConfig.ExecStartPre = makeJobScript "${name}-pre-start" ''
|
|
|
|
|
#! ${pkgs.stdenv.shell} -e
|
|
|
|
|
${config.preStart}
|
|
|
|
|
'';
|
|
|
|
|
})
|
|
|
|
|
(mkIf (config.script != "")
|
|
|
|
|
{ serviceConfig.ExecStart = makeJobScript "${name}-start" ''
|
|
|
|
|
#! ${pkgs.stdenv.shell} -e
|
|
|
|
|
${config.script}
|
|
|
|
|
'' + " " + config.scriptArgs;
|
|
|
|
|
})
|
|
|
|
|
(mkIf (config.postStart != "")
|
|
|
|
|
{ serviceConfig.ExecStartPost = makeJobScript "${name}-post-start" ''
|
|
|
|
|
#! ${pkgs.stdenv.shell} -e
|
|
|
|
|
${config.postStart}
|
|
|
|
|
'';
|
|
|
|
|
})
|
2013-11-26 18:24:55 +01:00
|
|
|
|
(mkIf (config.preStop != "")
|
|
|
|
|
{ serviceConfig.ExecStop = makeJobScript "${name}-pre-stop" ''
|
|
|
|
|
#! ${pkgs.stdenv.shell} -e
|
|
|
|
|
${config.preStop}
|
|
|
|
|
'';
|
|
|
|
|
})
|
2013-11-18 15:45:24 +01:00
|
|
|
|
(mkIf (config.postStop != "")
|
|
|
|
|
{ serviceConfig.ExecStopPost = makeJobScript "${name}-post-stop" ''
|
|
|
|
|
#! ${pkgs.stdenv.shell} -e
|
|
|
|
|
${config.postStop}
|
|
|
|
|
'';
|
|
|
|
|
})
|
|
|
|
|
];
|
2012-08-06 17:45:59 +02:00
|
|
|
|
};
|
|
|
|
|
|
2012-12-28 13:29:53 +01:00
|
|
|
|
mountConfig = { name, config, ... }: {
|
|
|
|
|
config = {
|
|
|
|
|
mountConfig =
|
|
|
|
|
{ What = config.what;
|
|
|
|
|
Where = config.where;
|
|
|
|
|
} // optionalAttrs (config.type != "") {
|
|
|
|
|
Type = config.type;
|
|
|
|
|
} // optionalAttrs (config.options != "") {
|
|
|
|
|
Options = config.options;
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
2013-09-23 22:56:05 +02:00
|
|
|
|
automountConfig = { name, config, ... }: {
|
|
|
|
|
config = {
|
|
|
|
|
automountConfig =
|
|
|
|
|
{ Where = config.where;
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
2012-10-01 22:27:42 +02:00
|
|
|
|
toOption = x:
|
|
|
|
|
if x == true then "true"
|
|
|
|
|
else if x == false then "false"
|
|
|
|
|
else toString x;
|
|
|
|
|
|
|
|
|
|
attrsToSection = as:
|
|
|
|
|
concatStrings (concatLists (mapAttrsToList (name: value:
|
|
|
|
|
map (x: ''
|
|
|
|
|
${name}=${toOption x}
|
|
|
|
|
'')
|
|
|
|
|
(if isList value then value else [value]))
|
|
|
|
|
as));
|
|
|
|
|
|
2012-10-02 00:58:11 +02:00
|
|
|
|
targetToUnit = name: def:
|
2013-04-19 22:28:00 +02:00
|
|
|
|
{ inherit (def) wantedBy requiredBy enable;
|
2012-10-02 00:58:11 +02:00
|
|
|
|
text =
|
|
|
|
|
''
|
|
|
|
|
[Unit]
|
|
|
|
|
${attrsToSection def.unitConfig}
|
|
|
|
|
'';
|
|
|
|
|
};
|
2012-06-18 21:28:31 +02:00
|
|
|
|
|
2012-10-02 00:58:11 +02:00
|
|
|
|
serviceToUnit = name: def:
|
2013-04-19 22:28:00 +02:00
|
|
|
|
{ inherit (def) wantedBy requiredBy enable;
|
2012-06-18 21:28:31 +02:00
|
|
|
|
text =
|
|
|
|
|
''
|
|
|
|
|
[Unit]
|
2012-10-01 22:27:42 +02:00
|
|
|
|
${attrsToSection def.unitConfig}
|
2012-06-18 21:28:31 +02:00
|
|
|
|
|
|
|
|
|
[Service]
|
2012-10-30 17:27:14 +01:00
|
|
|
|
${let env = cfg.globalEnvironment // def.environment;
|
2013-02-01 13:39:47 +01:00
|
|
|
|
in concatMapStrings (n: "Environment=\"${n}=${getAttr n env}\"\n") (attrNames env)}
|
2012-08-17 19:14:42 +02:00
|
|
|
|
${optionalString (!def.restartIfChanged) "X-RestartIfChanged=false"}
|
2013-01-05 01:05:25 +01:00
|
|
|
|
${optionalString (!def.stopIfChanged) "X-StopIfChanged=false"}
|
2012-10-01 22:27:42 +02:00
|
|
|
|
${attrsToSection def.serviceConfig}
|
2012-06-18 21:28:31 +02:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2012-10-02 00:58:11 +02:00
|
|
|
|
socketToUnit = name: def:
|
2013-04-19 22:28:00 +02:00
|
|
|
|
{ inherit (def) wantedBy requiredBy enable;
|
2012-10-02 00:58:11 +02:00
|
|
|
|
text =
|
|
|
|
|
''
|
|
|
|
|
[Unit]
|
|
|
|
|
${attrsToSection def.unitConfig}
|
|
|
|
|
|
|
|
|
|
[Socket]
|
|
|
|
|
${attrsToSection def.socketConfig}
|
2013-05-14 16:07:55 +02:00
|
|
|
|
${concatStringsSep "\n" (map (s: "ListenStream=${s}") def.listenStreams)}
|
2012-10-02 00:58:11 +02:00
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2013-03-02 01:03:13 +01:00
|
|
|
|
timerToUnit = name: def:
|
2013-04-19 22:28:00 +02:00
|
|
|
|
{ inherit (def) wantedBy requiredBy enable;
|
2013-03-02 01:03:13 +01:00
|
|
|
|
text =
|
|
|
|
|
''
|
|
|
|
|
[Unit]
|
|
|
|
|
${attrsToSection def.unitConfig}
|
|
|
|
|
|
|
|
|
|
[Timer]
|
|
|
|
|
${attrsToSection def.timerConfig}
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2012-12-28 13:29:53 +01:00
|
|
|
|
mountToUnit = name: def:
|
2013-04-19 22:28:00 +02:00
|
|
|
|
{ inherit (def) wantedBy requiredBy enable;
|
2012-12-28 13:29:53 +01:00
|
|
|
|
text =
|
|
|
|
|
''
|
|
|
|
|
[Unit]
|
|
|
|
|
${attrsToSection def.unitConfig}
|
|
|
|
|
|
|
|
|
|
[Mount]
|
|
|
|
|
${attrsToSection def.mountConfig}
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2013-09-23 22:56:05 +02:00
|
|
|
|
automountToUnit = name: def:
|
|
|
|
|
{ inherit (def) wantedBy requiredBy enable;
|
|
|
|
|
text =
|
|
|
|
|
''
|
|
|
|
|
[Unit]
|
|
|
|
|
${attrsToSection def.unitConfig}
|
|
|
|
|
|
|
|
|
|
[Automount]
|
|
|
|
|
${attrsToSection def.automountConfig}
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2012-06-16 06:19:43 +02:00
|
|
|
|
units = pkgs.runCommand "units" { preferLocalBuild = true; }
|
2012-06-02 02:15:07 +02:00
|
|
|
|
''
|
2012-07-19 23:32:50 +02:00
|
|
|
|
mkdir -p $out
|
2012-06-02 02:15:07 +02:00
|
|
|
|
for i in ${toString upstreamUnits}; do
|
|
|
|
|
fn=${systemd}/example/systemd/system/$i
|
2012-08-11 00:56:12 +02:00
|
|
|
|
if ! [ -e $fn ]; then echo "missing $fn"; false; fi
|
2012-06-15 00:44:56 +02:00
|
|
|
|
if [ -L $fn ]; then
|
2012-07-19 23:32:50 +02:00
|
|
|
|
cp -pd $fn $out/
|
2012-06-15 00:44:56 +02:00
|
|
|
|
else
|
2012-07-19 23:32:50 +02:00
|
|
|
|
ln -s $fn $out/
|
2012-06-15 00:44:56 +02:00
|
|
|
|
fi
|
2012-06-02 02:15:07 +02:00
|
|
|
|
done
|
2012-08-06 17:45:59 +02:00
|
|
|
|
|
2012-06-15 00:44:56 +02:00
|
|
|
|
for i in ${toString upstreamWants}; do
|
|
|
|
|
fn=${systemd}/example/systemd/system/$i
|
2012-08-11 00:56:12 +02:00
|
|
|
|
if ! [ -e $fn ]; then echo "missing $fn"; false; fi
|
2012-07-19 23:32:50 +02:00
|
|
|
|
x=$out/$(basename $fn)
|
2012-06-15 00:44:56 +02:00
|
|
|
|
mkdir $x
|
|
|
|
|
for i in $fn/*; do
|
|
|
|
|
y=$x/$(basename $i)
|
|
|
|
|
cp -pd $i $y
|
|
|
|
|
if ! [ -e $y ]; then rm -v $y; fi
|
|
|
|
|
done
|
|
|
|
|
done
|
2012-08-06 17:45:59 +02:00
|
|
|
|
|
2013-11-18 13:18:58 +01:00
|
|
|
|
for i in ${toString (mapAttrsToList (n: v: v.unit) cfg.units)}; do
|
2014-01-15 12:58:16 +01:00
|
|
|
|
ln -fs $i/* $out/
|
2012-06-02 02:15:07 +02:00
|
|
|
|
done
|
2012-06-18 05:31:21 +02:00
|
|
|
|
|
2012-08-21 17:28:47 +02:00
|
|
|
|
for i in ${toString cfg.packages}; do
|
|
|
|
|
ln -s $i/etc/systemd/system/* $out/
|
|
|
|
|
done
|
|
|
|
|
|
2012-06-18 05:31:21 +02:00
|
|
|
|
${concatStrings (mapAttrsToList (name: unit:
|
|
|
|
|
concatMapStrings (name2: ''
|
2013-11-12 16:58:36 +01:00
|
|
|
|
mkdir -p $out/'${name2}.wants'
|
|
|
|
|
ln -sfn '../${name}' $out/'${name2}.wants'/
|
2012-06-18 05:31:21 +02:00
|
|
|
|
'') unit.wantedBy) cfg.units)}
|
|
|
|
|
|
2013-04-19 22:28:00 +02:00
|
|
|
|
${concatStrings (mapAttrsToList (name: unit:
|
|
|
|
|
concatMapStrings (name2: ''
|
2013-11-12 16:58:36 +01:00
|
|
|
|
mkdir -p $out/'${name2}.requires'
|
|
|
|
|
ln -sfn '../${name}' $out/'${name2}.requires'/
|
2013-04-19 22:28:00 +02:00
|
|
|
|
'') unit.requiredBy) cfg.units)}
|
|
|
|
|
|
2012-07-19 23:32:50 +02:00
|
|
|
|
ln -s ${cfg.defaultUnit} $out/default.target
|
2012-07-21 00:32:24 +02:00
|
|
|
|
|
2013-01-08 18:24:06 +01:00
|
|
|
|
ln -s rescue.target $out/kbrequest.target
|
|
|
|
|
|
2013-01-24 14:55:55 +01:00
|
|
|
|
mkdir -p $out/getty.target.wants/
|
2014-01-15 12:58:16 +01:00
|
|
|
|
ln -s ../autovt@tty1.service $out/getty.target.wants/
|
2013-01-24 14:55:55 +01:00
|
|
|
|
|
2012-12-27 10:04:05 +01:00
|
|
|
|
ln -s ../local-fs.target ../remote-fs.target ../network.target ../nss-lookup.target \
|
|
|
|
|
../nss-user-lookup.target ../swap.target $out/multi-user.target.wants/
|
2012-06-02 02:15:07 +02:00
|
|
|
|
''; # */
|
2012-08-06 17:45:59 +02:00
|
|
|
|
|
2012-06-02 02:15:07 +02:00
|
|
|
|
in
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
2012-06-15 00:44:56 +02:00
|
|
|
|
###### interface
|
|
|
|
|
|
|
|
|
|
options = {
|
|
|
|
|
|
2013-01-16 13:17:57 +01:00
|
|
|
|
systemd.package = mkOption {
|
|
|
|
|
default = pkgs.systemd;
|
|
|
|
|
type = types.package;
|
|
|
|
|
description = "The systemd package.";
|
|
|
|
|
};
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.units = mkOption {
|
2012-06-18 21:28:31 +02:00
|
|
|
|
description = "Definition of systemd units.";
|
2012-06-18 05:31:21 +02:00
|
|
|
|
default = {};
|
|
|
|
|
type = types.attrsOf types.optionSet;
|
2013-11-18 13:18:58 +01:00
|
|
|
|
options = { name, config, ... }:
|
|
|
|
|
{ options = {
|
|
|
|
|
text = mkOption {
|
2014-01-18 17:10:39 +01:00
|
|
|
|
type = types.nullOr types.str;
|
|
|
|
|
default = null;
|
2013-11-18 13:18:58 +01:00
|
|
|
|
description = "Text of this systemd unit.";
|
|
|
|
|
};
|
|
|
|
|
enable = mkOption {
|
|
|
|
|
default = true;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
|
|
|
|
If set to false, this unit will be a symlink to
|
|
|
|
|
/dev/null. This is primarily useful to prevent specific
|
|
|
|
|
template instances (e.g. <literal>serial-getty@ttyS0</literal>)
|
|
|
|
|
from being started.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
requiredBy = mkOption {
|
|
|
|
|
default = [];
|
|
|
|
|
type = types.listOf types.string;
|
|
|
|
|
description = "Units that require (i.e. depend on and need to go down with) this unit.";
|
|
|
|
|
};
|
|
|
|
|
wantedBy = mkOption {
|
|
|
|
|
default = [];
|
|
|
|
|
type = types.listOf types.string;
|
|
|
|
|
description = "Units that want (i.e. depend on) this unit.";
|
|
|
|
|
};
|
|
|
|
|
unit = mkOption {
|
|
|
|
|
internal = true;
|
|
|
|
|
description = "The generated unit.";
|
|
|
|
|
};
|
2014-01-15 12:58:16 +01:00
|
|
|
|
linkTarget = mkOption {
|
|
|
|
|
default = null;
|
|
|
|
|
description = "The file to symlink this target to.";
|
|
|
|
|
type = types.nullOr types.path;
|
|
|
|
|
};
|
2014-01-18 17:10:39 +01:00
|
|
|
|
extraConfig = mkOption {
|
|
|
|
|
default = {};
|
|
|
|
|
example = { "foo@1.conf" = "X-RestartIfChanged=false"; };
|
|
|
|
|
type = types.attrsOf types.lines;
|
|
|
|
|
description = ''
|
|
|
|
|
Extra files to be appended to the configuration for the unit.
|
|
|
|
|
This can be used to override configuration for a unit provided
|
|
|
|
|
by systemd or another package, or to override only a single instance
|
|
|
|
|
of a template unit.
|
|
|
|
|
'';
|
|
|
|
|
};
|
2013-11-18 13:18:58 +01:00
|
|
|
|
};
|
|
|
|
|
config = {
|
|
|
|
|
unit = makeUnit name config;
|
|
|
|
|
};
|
2012-06-18 05:31:21 +02:00
|
|
|
|
};
|
2012-06-18 21:28:31 +02:00
|
|
|
|
};
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.packages = mkOption {
|
2012-08-21 17:28:47 +02:00
|
|
|
|
default = [];
|
|
|
|
|
type = types.listOf types.package;
|
|
|
|
|
description = "Packages providing systemd units.";
|
|
|
|
|
};
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.targets = mkOption {
|
2012-10-02 00:58:11 +02:00
|
|
|
|
default = {};
|
|
|
|
|
type = types.attrsOf types.optionSet;
|
2012-10-09 21:14:15 +02:00
|
|
|
|
options = [ unitOptions unitConfig ];
|
2012-10-02 00:58:11 +02:00
|
|
|
|
description = "Definition of systemd target units.";
|
|
|
|
|
};
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.services = mkOption {
|
2012-06-18 21:28:31 +02:00
|
|
|
|
default = {};
|
|
|
|
|
type = types.attrsOf types.optionSet;
|
2012-10-09 21:14:15 +02:00
|
|
|
|
options = [ serviceOptions unitConfig serviceConfig ];
|
2012-10-02 00:58:11 +02:00
|
|
|
|
description = "Definition of systemd service units.";
|
|
|
|
|
};
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.sockets = mkOption {
|
2012-10-02 00:58:11 +02:00
|
|
|
|
default = {};
|
|
|
|
|
type = types.attrsOf types.optionSet;
|
2012-10-09 21:14:15 +02:00
|
|
|
|
options = [ socketOptions unitConfig ];
|
2012-10-02 00:58:11 +02:00
|
|
|
|
description = "Definition of systemd socket units.";
|
2012-06-15 00:44:56 +02:00
|
|
|
|
};
|
|
|
|
|
|
2013-03-02 01:03:13 +01:00
|
|
|
|
systemd.timers = mkOption {
|
|
|
|
|
default = {};
|
|
|
|
|
type = types.attrsOf types.optionSet;
|
|
|
|
|
options = [ timerOptions unitConfig ];
|
|
|
|
|
description = "Definition of systemd timer units.";
|
|
|
|
|
};
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.mounts = mkOption {
|
2012-12-28 13:29:53 +01:00
|
|
|
|
default = [];
|
|
|
|
|
type = types.listOf types.optionSet;
|
|
|
|
|
options = [ mountOptions unitConfig mountConfig ];
|
|
|
|
|
description = ''
|
|
|
|
|
Definition of systemd mount units.
|
|
|
|
|
This is a list instead of an attrSet, because systemd mandates the names to be derived from
|
|
|
|
|
the 'where' attribute.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2013-09-23 22:56:05 +02:00
|
|
|
|
systemd.automounts = mkOption {
|
|
|
|
|
default = [];
|
|
|
|
|
type = types.listOf types.optionSet;
|
|
|
|
|
options = [ automountOptions unitConfig automountConfig ];
|
|
|
|
|
description = ''
|
|
|
|
|
Definition of systemd automount units.
|
|
|
|
|
This is a list instead of an attrSet, because systemd mandates the names to be derived from
|
|
|
|
|
the 'where' attribute.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.defaultUnit = mkOption {
|
2012-06-18 05:31:21 +02:00
|
|
|
|
default = "multi-user.target";
|
2013-10-30 11:02:04 +01:00
|
|
|
|
type = types.str;
|
2012-06-18 05:31:21 +02:00
|
|
|
|
description = "Default unit started when the system boots.";
|
|
|
|
|
};
|
2012-07-19 23:32:50 +02:00
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.globalEnvironment = mkOption {
|
2012-10-30 17:27:14 +01:00
|
|
|
|
type = types.attrs;
|
|
|
|
|
default = {};
|
|
|
|
|
example = { TZ = "CET"; };
|
|
|
|
|
description = ''
|
|
|
|
|
Environment variables passed to <emphasis>all</emphasis> systemd units.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2013-11-15 19:49:01 +01:00
|
|
|
|
systemd.extraConfig = mkOption {
|
|
|
|
|
default = "";
|
|
|
|
|
type = types.lines;
|
|
|
|
|
example = "DefaultLimitCORE=infinity";
|
|
|
|
|
description = ''
|
|
|
|
|
Extra config options for systemd. See man systemd-system.conf for
|
|
|
|
|
available options.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2012-07-19 23:32:50 +02:00
|
|
|
|
services.journald.console = mkOption {
|
|
|
|
|
default = "";
|
2013-10-30 11:02:04 +01:00
|
|
|
|
type = types.str;
|
2012-11-02 14:10:06 +01:00
|
|
|
|
description = "If non-empty, write log messages to the specified TTY device.";
|
2012-07-19 23:32:50 +02:00
|
|
|
|
};
|
2012-08-06 17:45:59 +02:00
|
|
|
|
|
2012-12-16 20:28:45 +01:00
|
|
|
|
services.journald.rateLimitInterval = mkOption {
|
|
|
|
|
default = "10s";
|
2013-10-30 11:02:04 +01:00
|
|
|
|
type = types.str;
|
2012-12-16 20:28:45 +01:00
|
|
|
|
description = ''
|
|
|
|
|
Configures the rate limiting interval that is applied to all
|
|
|
|
|
messages generated on the system. This rate limiting is applied
|
|
|
|
|
per-service, so that two services which log do not interfere with
|
|
|
|
|
each other's limit. The value may be specified in the following
|
|
|
|
|
units: s, min, h, ms, us. To turn off any kind of rate limiting,
|
|
|
|
|
set either value to 0.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
services.journald.rateLimitBurst = mkOption {
|
|
|
|
|
default = 100;
|
|
|
|
|
type = types.uniq types.int;
|
|
|
|
|
description = ''
|
|
|
|
|
Configures the rate limiting burst limit (number of messages per
|
|
|
|
|
interval) that is applied to all messages generated on the system.
|
|
|
|
|
This rate limiting is applied per-service, so that two services
|
|
|
|
|
which log do not interfere with each other's limit.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2013-11-29 11:57:42 +01:00
|
|
|
|
services.journald.extraConfig = mkOption {
|
|
|
|
|
default = "";
|
|
|
|
|
type = types.lines;
|
|
|
|
|
example = "Storage=volatile";
|
|
|
|
|
description = ''
|
|
|
|
|
Extra config options for systemd-journald. See man journald.conf
|
|
|
|
|
for available options.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2013-02-24 11:54:01 +01:00
|
|
|
|
services.logind.extraConfig = mkOption {
|
|
|
|
|
default = "";
|
2013-11-15 21:56:45 +01:00
|
|
|
|
type = types.lines;
|
2013-02-24 11:54:01 +01:00
|
|
|
|
example = "HandleLidSwitch=ignore";
|
|
|
|
|
description = ''
|
|
|
|
|
Extra config options for systemd-logind. See man logind.conf for
|
|
|
|
|
available options.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2013-01-21 21:01:48 +01:00
|
|
|
|
systemd.enableEmergencyMode = mkOption {
|
|
|
|
|
default = true;
|
|
|
|
|
type = types.bool;
|
|
|
|
|
description = ''
|
|
|
|
|
Whether to enable emergency mode, which is an
|
|
|
|
|
<command>sulogin</command> shell started on the console if
|
|
|
|
|
mounting a filesystem fails. Since some machines (like EC2
|
|
|
|
|
instances) have no console of any kind, emergency mode doesn't
|
|
|
|
|
make sense, and it's better to continue with the boot insofar
|
|
|
|
|
as possible.
|
|
|
|
|
'';
|
|
|
|
|
};
|
|
|
|
|
|
2012-06-15 00:44:56 +02:00
|
|
|
|
};
|
|
|
|
|
|
2012-08-06 17:45:59 +02:00
|
|
|
|
|
2012-06-02 02:15:07 +02:00
|
|
|
|
###### implementation
|
|
|
|
|
|
|
|
|
|
config = {
|
|
|
|
|
|
2012-06-16 06:19:43 +02:00
|
|
|
|
system.build.units = units;
|
|
|
|
|
|
2012-06-02 02:15:07 +02:00
|
|
|
|
environment.systemPackages = [ systemd ];
|
2012-08-06 17:45:59 +02:00
|
|
|
|
|
2013-05-09 16:21:42 +02:00
|
|
|
|
environment.etc."systemd/system".source = units;
|
|
|
|
|
|
|
|
|
|
environment.etc."systemd/system.conf".text =
|
|
|
|
|
''
|
|
|
|
|
[Manager]
|
2013-11-15 19:49:01 +01:00
|
|
|
|
${config.systemd.extraConfig}
|
2013-05-09 16:21:42 +02:00
|
|
|
|
'';
|
|
|
|
|
|
|
|
|
|
environment.etc."systemd/journald.conf".text =
|
|
|
|
|
''
|
|
|
|
|
[Journal]
|
|
|
|
|
RateLimitInterval=${config.services.journald.rateLimitInterval}
|
|
|
|
|
RateLimitBurst=${toString config.services.journald.rateLimitBurst}
|
|
|
|
|
${optionalString (config.services.journald.console != "") ''
|
|
|
|
|
ForwardToConsole=yes
|
|
|
|
|
TTYPath=${config.services.journald.console}
|
|
|
|
|
''}
|
2013-11-29 11:57:42 +01:00
|
|
|
|
${config.services.journald.extraConfig}
|
2013-05-09 16:21:42 +02:00
|
|
|
|
'';
|
|
|
|
|
|
|
|
|
|
environment.etc."systemd/logind.conf".text =
|
|
|
|
|
''
|
|
|
|
|
[Login]
|
|
|
|
|
${config.services.logind.extraConfig}
|
|
|
|
|
'';
|
2012-06-15 00:44:56 +02:00
|
|
|
|
|
2013-05-09 16:25:24 +02:00
|
|
|
|
environment.etc."systemd/sleep.conf".text =
|
|
|
|
|
''
|
|
|
|
|
[Sleep]
|
|
|
|
|
'';
|
|
|
|
|
|
2013-07-19 21:18:44 +02:00
|
|
|
|
system.activationScripts.systemd = stringAfter [ "groups" ]
|
2012-11-29 18:51:44 +01:00
|
|
|
|
''
|
2013-07-20 17:07:26 +02:00
|
|
|
|
mkdir -m 0755 -p /var/lib/udev
|
|
|
|
|
mkdir -p /var/log/journal
|
|
|
|
|
chmod 0755 /var/log/journal
|
2012-11-29 18:51:44 +01:00
|
|
|
|
|
2013-07-19 21:18:44 +02:00
|
|
|
|
# Make all journals readable to users in the wheel and adm
|
|
|
|
|
# groups, in addition to those in the systemd-journal group.
|
|
|
|
|
# Users can always read their own journals.
|
|
|
|
|
${pkgs.acl}/bin/setfacl -nm g:wheel:rx,d:g:wheel:rx,g:adm:rx,d:g:adm:rx /var/log/journal
|
2012-11-29 18:51:44 +01:00
|
|
|
|
'';
|
|
|
|
|
|
2012-11-01 23:32:12 +01:00
|
|
|
|
# Target for ‘charon send-keys’ to hook into.
|
2014-02-11 13:00:10 +01:00
|
|
|
|
users.extraGroups.keys.gid = config.ids.gids.keys;
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.targets.keys =
|
2012-11-01 23:32:12 +01:00
|
|
|
|
{ description = "Security Keys";
|
2014-02-26 19:35:04 +01:00
|
|
|
|
unitConfig.X-StopOnReconfiguration = true;
|
2012-11-01 23:32:12 +01:00
|
|
|
|
};
|
|
|
|
|
|
2013-01-16 12:33:18 +01:00
|
|
|
|
systemd.units =
|
2013-01-08 18:24:06 +01:00
|
|
|
|
mapAttrs' (n: v: nameValuePair "${n}.target" (targetToUnit n v)) cfg.targets
|
2012-10-02 00:58:11 +02:00
|
|
|
|
// mapAttrs' (n: v: nameValuePair "${n}.service" (serviceToUnit n v)) cfg.services
|
2012-12-28 13:29:53 +01:00
|
|
|
|
// mapAttrs' (n: v: nameValuePair "${n}.socket" (socketToUnit n v)) cfg.sockets
|
2013-03-02 01:03:13 +01:00
|
|
|
|
// mapAttrs' (n: v: nameValuePair "${n}.timer" (timerToUnit n v)) cfg.timers
|
2012-12-28 13:29:53 +01:00
|
|
|
|
// listToAttrs (map
|
2013-01-01 14:42:43 +01:00
|
|
|
|
(v: let n = escapeSystemdPath v.where;
|
2013-09-23 22:56:05 +02:00
|
|
|
|
in nameValuePair "${n}.mount" (mountToUnit n v)) cfg.mounts)
|
|
|
|
|
// listToAttrs (map
|
|
|
|
|
(v: let n = escapeSystemdPath v.where;
|
|
|
|
|
in nameValuePair "${n}.automount" (automountToUnit n v)) cfg.automounts);
|
2012-06-15 00:44:56 +02:00
|
|
|
|
|
2012-11-03 05:41:46 +01:00
|
|
|
|
system.requiredKernelConfig = map config.lib.kernelConfig.isEnabled [
|
|
|
|
|
"CGROUPS" "AUTOFS4_FS" "DEVTMPFS"
|
|
|
|
|
];
|
2012-11-29 18:51:44 +01:00
|
|
|
|
|
2012-11-30 16:12:04 +01:00
|
|
|
|
environment.shellAliases =
|
|
|
|
|
{ start = "systemctl start";
|
|
|
|
|
stop = "systemctl stop";
|
|
|
|
|
restart = "systemctl restart";
|
|
|
|
|
status = "systemctl status";
|
|
|
|
|
};
|
|
|
|
|
|
2013-07-19 21:18:44 +02:00
|
|
|
|
users.extraGroups.systemd-journal.gid = config.ids.gids.systemd-journal;
|
|
|
|
|
|
2013-10-09 14:28:35 +02:00
|
|
|
|
# Generate timer units for all services that have a ‘startAt’ value.
|
|
|
|
|
systemd.timers =
|
|
|
|
|
mapAttrs (name: service:
|
|
|
|
|
{ wantedBy = [ "timers.target" ];
|
|
|
|
|
timerConfig.OnCalendar = service.startAt;
|
|
|
|
|
})
|
|
|
|
|
(filterAttrs (name: service: service.startAt != "") cfg.services);
|
|
|
|
|
|
2013-09-22 21:04:54 +02:00
|
|
|
|
# FIXME: These are borrowed from upstream systemd.
|
|
|
|
|
systemd.services."systemd-update-utmp" =
|
|
|
|
|
{ description = "Update UTMP about System Reboot/Shutdown";
|
2013-09-22 16:44:41 +02:00
|
|
|
|
wantedBy = [ "sysinit.target" ];
|
2013-09-22 21:04:54 +02:00
|
|
|
|
after = [ "systemd-remount-fs.service" ];
|
|
|
|
|
before = [ "sysinit.target" "shutdown.target" ];
|
|
|
|
|
conflicts = [ "shutdown.target" ];
|
|
|
|
|
unitConfig = {
|
|
|
|
|
DefaultDependencies = false;
|
|
|
|
|
RequiresMountsFor = "/var/log";
|
|
|
|
|
};
|
|
|
|
|
serviceConfig = {
|
|
|
|
|
Type = "oneshot";
|
|
|
|
|
RemainAfterExit = true;
|
|
|
|
|
ExecStart = "${systemd}/lib/systemd/systemd-update-utmp reboot";
|
|
|
|
|
ExecStop = "${systemd}/lib/systemd/systemd-update-utmp shutdown";
|
|
|
|
|
};
|
|
|
|
|
restartIfChanged = false;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
systemd.services."systemd-random-seed" =
|
|
|
|
|
{ description = "Load/Save Random Seed";
|
|
|
|
|
wantedBy = [ "sysinit.target" "multi-user.target" ];
|
|
|
|
|
after = [ "systemd-remount-fs.service" ];
|
|
|
|
|
before = [ "sysinit.target" "shutdown.target" ];
|
|
|
|
|
conflicts = [ "shutdown.target" ];
|
|
|
|
|
unitConfig = {
|
|
|
|
|
DefaultDependencies = false;
|
|
|
|
|
RequiresMountsFor = "/var/lib";
|
|
|
|
|
};
|
|
|
|
|
serviceConfig = {
|
|
|
|
|
Type = "oneshot";
|
|
|
|
|
RemainAfterExit = true;
|
|
|
|
|
ExecStart = "${systemd}/lib/systemd/systemd-random-seed load";
|
|
|
|
|
ExecStop = "${systemd}/lib/systemd/systemd-random-seed save";
|
|
|
|
|
};
|
2013-09-22 16:44:41 +02:00
|
|
|
|
};
|
|
|
|
|
|
2012-06-02 02:15:07 +02:00
|
|
|
|
};
|
|
|
|
|
}
|