2021-12-19 01:06:50 +00:00
|
|
|
{ config, lib, pkgs, utils, ...}:
|
2020-04-24 23:36:52 +00:00
|
|
|
|
2021-12-19 01:06:50 +00:00
|
|
|
with utils.systemdUtils.unitOptions;
|
|
|
|
with utils.systemdUtils.lib;
|
2020-04-24 23:36:52 +00:00
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
cfg = config.systemd.nspawn;
|
|
|
|
|
|
|
|
checkExec = checkUnitConfig "Exec" [
|
|
|
|
(assertOnlyFields [
|
|
|
|
"Boot" "ProcessTwo" "Parameters" "Environment" "User" "WorkingDirectory"
|
|
|
|
"PivotRoot" "Capability" "DropCapability" "NoNewPrivileges" "KillSignal"
|
2023-07-15 17:15:38 +00:00
|
|
|
"Personality" "MachineID" "PrivateUsers" "NotifyReady" "SystemCallFilter"
|
2020-04-24 23:36:52 +00:00
|
|
|
"LimitCPU" "LimitFSIZE" "LimitDATA" "LimitSTACK" "LimitCORE" "LimitRSS"
|
|
|
|
"LimitNOFILE" "LimitAS" "LimitNPROC" "LimitMEMLOCK" "LimitLOCKS"
|
|
|
|
"LimitSIGPENDING" "LimitMSGQUEUE" "LimitNICE" "LimitRTPRIO" "LimitRTTIME"
|
|
|
|
"OOMScoreAdjust" "CPUAffinity" "Hostname" "ResolvConf" "Timezone"
|
2022-05-18 14:49:53 +00:00
|
|
|
"LinkJournal" "Ephemeral" "AmbientCapability"
|
2020-04-24 23:36:52 +00:00
|
|
|
])
|
|
|
|
(assertValueOneOf "Boot" boolValues)
|
|
|
|
(assertValueOneOf "ProcessTwo" boolValues)
|
|
|
|
(assertValueOneOf "NotifyReady" boolValues)
|
|
|
|
];
|
|
|
|
|
|
|
|
checkFiles = checkUnitConfig "Files" [
|
|
|
|
(assertOnlyFields [
|
|
|
|
"ReadOnly" "Volatile" "Bind" "BindReadOnly" "TemporaryFileSystem"
|
2022-05-18 14:49:53 +00:00
|
|
|
"Overlay" "OverlayReadOnly" "PrivateUsersChown" "BindUser"
|
2022-09-14 18:05:37 +00:00
|
|
|
"Inaccessible" "PrivateUsersOwnership"
|
2020-04-24 23:36:52 +00:00
|
|
|
])
|
|
|
|
(assertValueOneOf "ReadOnly" boolValues)
|
|
|
|
(assertValueOneOf "Volatile" (boolValues ++ [ "state" ]))
|
|
|
|
(assertValueOneOf "PrivateUsersChown" boolValues)
|
2022-09-14 18:05:37 +00:00
|
|
|
(assertValueOneOf "PrivateUsersOwnership" [ "off" "chown" "map" "auto" ])
|
2020-04-24 23:36:52 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
checkNetwork = checkUnitConfig "Network" [
|
|
|
|
(assertOnlyFields [
|
|
|
|
"Private" "VirtualEthernet" "VirtualEthernetExtra" "Interface" "MACVLAN"
|
|
|
|
"IPVLAN" "Bridge" "Zone" "Port"
|
|
|
|
])
|
|
|
|
(assertValueOneOf "Private" boolValues)
|
|
|
|
(assertValueOneOf "VirtualEthernet" boolValues)
|
|
|
|
];
|
|
|
|
|
|
|
|
instanceOptions = {
|
2022-10-30 15:09:59 +00:00
|
|
|
options =
|
|
|
|
(getAttrs [ "enable" ] sharedOptions)
|
|
|
|
// {
|
2020-04-24 23:36:52 +00:00
|
|
|
execConfig = mkOption {
|
|
|
|
default = {};
|
|
|
|
example = { Parameters = "/bin/sh"; };
|
|
|
|
type = types.addCheck (types.attrsOf unitOption) checkExec;
|
2022-08-21 13:32:41 +00:00
|
|
|
description = lib.mdDoc ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Each attribute in this set specifies an option in the
|
2022-08-21 13:32:41 +00:00
|
|
|
`[Exec]` section of this unit. See
|
|
|
|
{manpage}`systemd.nspawn(5)` for details.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
filesConfig = mkOption {
|
|
|
|
default = {};
|
|
|
|
example = { Bind = [ "/home/alice" ]; };
|
|
|
|
type = types.addCheck (types.attrsOf unitOption) checkFiles;
|
2022-08-21 13:32:41 +00:00
|
|
|
description = lib.mdDoc ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Each attribute in this set specifies an option in the
|
2022-08-21 13:32:41 +00:00
|
|
|
`[Files]` section of this unit. See
|
|
|
|
{manpage}`systemd.nspawn(5)` for details.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
networkConfig = mkOption {
|
|
|
|
default = {};
|
|
|
|
example = { Private = false; };
|
|
|
|
type = types.addCheck (types.attrsOf unitOption) checkNetwork;
|
2022-08-21 13:32:41 +00:00
|
|
|
description = lib.mdDoc ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Each attribute in this set specifies an option in the
|
2022-08-21 13:32:41 +00:00
|
|
|
`[Network]` section of this unit. See
|
|
|
|
{manpage}`systemd.nspawn(5)` for details.
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
instanceToUnit = name: def:
|
|
|
|
let base = {
|
|
|
|
text = ''
|
|
|
|
[Exec]
|
|
|
|
${attrsToSection def.execConfig}
|
|
|
|
|
|
|
|
[Files]
|
|
|
|
${attrsToSection def.filesConfig}
|
|
|
|
|
|
|
|
[Network]
|
|
|
|
${attrsToSection def.networkConfig}
|
|
|
|
'';
|
|
|
|
} // def;
|
|
|
|
in base // { unit = makeUnit name base; };
|
|
|
|
|
|
|
|
in {
|
|
|
|
|
|
|
|
options = {
|
|
|
|
|
|
|
|
systemd.nspawn = mkOption {
|
|
|
|
default = {};
|
|
|
|
type = with types; attrsOf (submodule instanceOptions);
|
2022-08-12 12:06:08 +00:00
|
|
|
description = lib.mdDoc "Definition of systemd-nspawn configurations.";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
config =
|
|
|
|
let
|
|
|
|
units = mapAttrs' (n: v: let nspawnFile = "${n}.nspawn"; in nameValuePair nspawnFile (instanceToUnit nspawnFile v)) cfg;
|
2020-08-20 17:08:02 +00:00
|
|
|
in
|
2020-04-24 23:36:52 +00:00
|
|
|
mkMerge [
|
2020-08-20 17:08:02 +00:00
|
|
|
(mkIf (cfg != {}) {
|
2022-04-03 18:54:34 +00:00
|
|
|
environment.etc."systemd/nspawn".source = mkIf (cfg != {}) (generateUnits {
|
|
|
|
allowCollisions = false;
|
|
|
|
type = "nspawn";
|
|
|
|
inherit units;
|
|
|
|
upstreamUnits = [];
|
|
|
|
upstreamWants = [];
|
|
|
|
});
|
2020-04-24 23:36:52 +00:00
|
|
|
})
|
|
|
|
{
|
|
|
|
systemd.targets.multi-user.wants = [ "machines.target" ];
|
|
|
|
}
|
|
|
|
];
|
|
|
|
}
|