nixpkgs/nixos/modules/virtualisation/lxd.nix

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

264 lines
8.2 KiB
Nix
Raw Normal View History

2015-09-14 06:27:31 +00:00
# Systemd services for lxd.
{ config, lib, pkgs, ... }:
let
cfg = config.virtualisation.lxd;
2023-09-04 15:58:53 +00:00
preseedFormat = pkgs.formats.yaml {};
in {
meta = {
maintainers = lib.teams.lxc.members;
};
imports = [
2023-09-04 15:08:13 +00:00
(lib.mkRemovedOptionModule [ "virtualisation" "lxd" "zfsPackage" ] "Override zfs in an overlay instead to override it globally")
];
2015-09-14 06:27:31 +00:00
options = {
2018-03-16 09:58:54 +00:00
virtualisation.lxd = {
2023-09-04 15:08:13 +00:00
enable = lib.mkOption {
type = lib.types.bool;
2015-09-14 06:27:31 +00:00
default = false;
2018-03-16 09:58:54 +00:00
description = lib.mdDoc ''
This option enables lxd, a daemon that manages
containers. Users in the "lxd" group can interact with
the daemon (e.g. to start or stop containers) using the
{command}`lxc` command line tool, among others.
Most of the time, you'll also want to start lxcfs, so
that containers can "see" the limits:
```
virtualisation.lxc.lxcfs.enable = true;
```
2018-03-16 09:58:54 +00:00
'';
2015-09-14 06:27:31 +00:00
};
2024-04-02 03:59:10 +00:00
package = lib.mkPackageOption pkgs "lxd-lts" { };
lxcPackage = lib.mkPackageOption pkgs "lxc" {
extraDescription = ''
Required for AppArmor profiles.
'';
};
2023-09-04 15:08:13 +00:00
zfsSupport = lib.mkOption {
type = lib.types.bool;
default = config.boot.zfs.enabled;
2023-09-04 15:08:13 +00:00
defaultText = lib.literalExpression "config.boot.zfs.enabled";
2018-03-16 09:58:54 +00:00
description = lib.mdDoc ''
Enables lxd to use zfs as a storage for containers.
2018-03-16 09:58:54 +00:00
This option is enabled by default if a zfs pool is configured
with nixos.
'';
};
2020-06-05 10:57:18 +00:00
2023-09-04 15:08:13 +00:00
recommendedSysctlSettings = lib.mkOption {
type = lib.types.bool;
default = false;
description = lib.mdDoc ''
Enables various settings to avoid common pitfalls when
running containers requiring many file operations.
Fixes errors like "Too many open files" or
"neighbour: ndisc_cache: neighbor table overflow!".
See https://lxd.readthedocs.io/en/latest/production-setup/
for details.
'';
};
2023-09-04 15:58:53 +00:00
preseed = lib.mkOption {
type = lib.types.nullOr (lib.types.submodule {
freeformType = preseedFormat.type;
});
default = null;
description = lib.mdDoc ''
Configuration for LXD preseed, see
<https://documentation.ubuntu.com/lxd/en/latest/howto/initialize/#initialize-preseed>
for supported values.
Changes to this will be re-applied to LXD which will overwrite existing entities or create missing ones,
but entities will *not* be removed by preseed.
'';
example = lib.literalExpression ''
{
networks = [
{
name = "lxdbr0";
type = "bridge";
config = {
"ipv4.address" = "10.0.100.1/24";
"ipv4.nat" = "true";
};
}
];
profiles = [
{
name = "default";
devices = {
eth0 = {
name = "eth0";
network = "lxdbr0";
type = "nic";
};
root = {
path = "/";
pool = "default";
size = "35GiB";
type = "disk";
};
};
}
];
storage_pools = [
{
name = "default";
driver = "dir";
config = {
source = "/var/lib/lxd/storage-pools/default";
};
}
];
}
'';
};
2023-09-04 15:08:13 +00:00
startTimeout = lib.mkOption {
type = lib.types.int;
default = 600;
apply = toString;
description = lib.mdDoc ''
Time to wait (in seconds) for LXD to become ready to process requests.
If LXD does not reply within the configured time, lxd.service will be
considered failed and systemd will attempt to restart it.
'';
};
ui = {
enable = lib.mkEnableOption (lib.mdDoc "(experimental) LXD UI");
package = lib.mkPackageOption pkgs [ "lxd-ui" ] { };
};
2018-03-16 09:58:54 +00:00
};
2015-09-14 06:27:31 +00:00
};
###### implementation
2023-09-04 15:08:13 +00:00
config = lib.mkIf cfg.enable {
environment.systemPackages = [ cfg.package ];
2015-09-14 06:27:31 +00:00
# Note: the following options are also declared in virtualisation.lxc, but
# the latter can't be simply enabled to reuse the formers, because it
# does a bunch of unrelated things.
systemd.tmpfiles.rules = [ "d /var/lib/lxc/rootfs 0755 root root -" ];
security.apparmor = {
packages = [ cfg.lxcPackage ];
policies = {
"bin.lxc-start".profile = ''
include ${cfg.lxcPackage}/etc/apparmor.d/usr.bin.lxc-start
'';
"lxc-containers".profile = ''
include ${cfg.lxcPackage}/etc/apparmor.d/lxc-containers
'';
};
};
2018-02-10 08:18:53 +00:00
# TODO: remove once LXD gets proper support for cgroupsv2
# (currently most of the e.g. CPU accounting stuff doesn't work)
systemd.enableUnifiedCgroupHierarchy = false;
systemd.sockets.lxd = {
description = "LXD UNIX socket";
wantedBy = [ "sockets.target" ];
socketConfig = {
ListenStream = "/var/lib/lxd/unix.socket";
SocketMode = "0660";
SocketGroup = "lxd";
Service = "lxd.service";
};
};
2018-03-16 09:58:54 +00:00
systemd.services.lxd = {
description = "LXD Container Management Daemon";
2015-09-14 06:27:31 +00:00
2018-03-16 09:58:54 +00:00
wantedBy = [ "multi-user.target" ];
after = [
"network-online.target"
2023-09-04 15:08:13 +00:00
(lib.mkIf config.virtualisation.lxc.lxcfs.enable "lxcfs.service")
];
requires = [
"network-online.target"
"lxd.socket"
2023-09-04 15:08:13 +00:00
(lib.mkIf config.virtualisation.lxc.lxcfs.enable "lxcfs.service")
];
documentation = [ "man:lxd(1)" ];
2015-09-14 06:27:31 +00:00
path = [ pkgs.util-linux ]
2023-09-04 15:08:13 +00:00
++ lib.optional cfg.zfsSupport config.boot.zfs.package;
2018-02-10 08:18:53 +00:00
2023-09-04 15:08:13 +00:00
environment = lib.mkIf (cfg.ui.enable) {
"LXD_UI" = cfg.ui.package;
};
2018-03-16 09:58:54 +00:00
serviceConfig = {
ExecStart = "@${cfg.package}/bin/lxd lxd --group lxd";
ExecStartPost = "${cfg.package}/bin/lxd waitready --timeout=${cfg.startTimeout}";
ExecStop = "${cfg.package}/bin/lxd shutdown";
2018-03-16 09:58:54 +00:00
KillMode = "process"; # when stopping, leave the containers alone
LimitMEMLOCK = "infinity";
LimitNOFILE = "1048576";
LimitNPROC = "infinity";
TasksMax = "infinity";
# By default, `lxd` loads configuration files from hard-coded
# `/usr/share/lxc/config` - since this is a no-go for us, we have to
# explicitly tell it where the actual configuration files are
2023-09-04 15:08:13 +00:00
Environment = lib.mkIf (config.virtualisation.lxc.lxcfs.enable)
"LXD_LXC_TEMPLATE_CONFIG=${pkgs.lxcfs}/share/lxc/config";
2015-09-14 06:27:31 +00:00
};
unitConfig.ConditionPathExists = "!/var/lib/incus/.migrated-from-lxd";
2018-03-16 09:58:54 +00:00
};
2023-09-04 15:58:53 +00:00
systemd.services.lxd-preseed = lib.mkIf (cfg.preseed != null) {
description = "LXD initialization with preseed file";
wantedBy = ["multi-user.target"];
requires = ["lxd.service"];
after = ["lxd.service"];
script = ''
${pkgs.coreutils}/bin/cat ${preseedFormat.generate "lxd-preseed.yaml" cfg.preseed} | ${cfg.package}/bin/lxd init --preseed
'';
serviceConfig = {
Type = "oneshot";
};
};
users.groups.lxd = {};
2015-09-14 06:27:31 +00:00
users.users.root = {
2015-09-14 06:27:31 +00:00
subUidRanges = [ { startUid = 1000000; count = 65536; } ];
subGidRanges = [ { startGid = 1000000; count = 65536; } ];
};
2023-09-04 15:08:13 +00:00
boot.kernel.sysctl = lib.mkIf cfg.recommendedSysctlSettings {
"fs.inotify.max_queued_events" = 1048576;
"fs.inotify.max_user_instances" = 1048576;
"fs.inotify.max_user_watches" = 1048576;
"vm.max_map_count" = 262144; # TODO: Default vm.max_map_count has been increased system-wide
"kernel.dmesg_restrict" = 1;
"net.ipv4.neigh.default.gc_thresh3" = 8192;
"net.ipv6.neigh.default.gc_thresh3" = 8192;
"kernel.keys.maxkeys" = 2000;
};
nixos/lxd: explicitly load kernel modules This is analogous to #70447 and #76487. These are all needed to attach a container to the default bridge network, without which the final line of the following script fails with the error for each respective kernel module listed below. ```sh lxc storage create foo dir lxc launch -s foo ubuntu:trusty bar lxc network attach lxdbr0 bar ``` veth ---- > Error: Failed to start device 'lxdbr0': Failed to create the veth interfaces vethefbc3cd6 and vetha4abbcbc: Failed to run: ip link add dev vethefbc3cd6 type veth peer name vetha4abbcbc: RTNETLINK answers: Operation not supported iptable_mangle -------------- > lvl=eror msg="Failed to bring up network" err="Failed to list ipv4 rules for LXD network lxdbr0 (table mangle)" name=lxdbr0 xt_comment ---------- > lvl=error msg="Failed to bring up network" err="Failed to run: iptables -w -t filter -I INPUT -i lxdbr0 -p udp --dport 67 -j ACCEPT -m comment --comment generated for LXD network lxdbr0: iptables v1.8.4 (legacy): Couldn't load match `comment':No such file or directory\n\nTry `iptables -h' or 'iptables --help' for more information." name=lxdbr0 xt_MASQUERADE ------------- > vl=eror msg="Failed to bring up network" err="Failed to run: iptables -w -t nat -I POSTROUTING -s 10.0.107.0/24 ! -d 10.0.107.0/24 -j MASQUERADE -m comment --comment generated for LXD network lxdbr0: iptables v1.8.4 (legacy): Couldn't load target `MASQUERADE':No such file or directory\n\nTry `iptables -h' or 'iptables --help' for more information." name=lxdbr0
2020-09-10 16:13:07 +00:00
boot.kernelModules = [ "veth" "xt_comment" "xt_CHECKSUM" "xt_MASQUERADE" "vhost_vsock" ]
2023-09-04 15:08:13 +00:00
++ lib.optionals (!config.networking.nftables.enable) [ "iptable_mangle" ];
2015-09-14 06:27:31 +00:00
};
}