2021-12-19 01:06:50 +00:00
|
|
|
{ config, lib, options, pkgs, ... }:
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
with lib;
|
|
|
|
|
|
|
|
let
|
|
|
|
top = config.services.kubernetes;
|
2021-12-19 01:06:50 +00:00
|
|
|
otop = options.services.kubernetes;
|
2020-04-24 23:36:52 +00:00
|
|
|
cfg = top.kubelet;
|
|
|
|
|
|
|
|
cniConfig =
|
|
|
|
if cfg.cni.config != [] && cfg.cni.configDir != null then
|
|
|
|
throw "Verbatim CNI-config and CNI configDir cannot both be set."
|
|
|
|
else if cfg.cni.configDir != null then
|
|
|
|
cfg.cni.configDir
|
|
|
|
else
|
|
|
|
(pkgs.buildEnv {
|
|
|
|
name = "kubernetes-cni-config";
|
|
|
|
paths = imap (i: entry:
|
|
|
|
pkgs.writeTextDir "${toString (10+i)}-${entry.type}.conf" (builtins.toJSON entry)
|
|
|
|
) cfg.cni.config;
|
|
|
|
});
|
|
|
|
|
|
|
|
infraContainer = pkgs.dockerTools.buildImage {
|
|
|
|
name = "pause";
|
|
|
|
tag = "latest";
|
2022-07-18 16:21:45 +00:00
|
|
|
copyToRoot = pkgs.buildEnv {
|
|
|
|
name = "image-root";
|
|
|
|
pathsToLink = [ "/bin" ];
|
|
|
|
paths = [ top.package.pause ];
|
|
|
|
};
|
2021-03-09 03:18:52 +00:00
|
|
|
config.Cmd = ["/bin/pause"];
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
2022-01-13 20:06:32 +00:00
|
|
|
kubeconfig = top.lib.mkKubeConfig "kubelet" cfg.kubeconfig;
|
2020-04-24 23:36:52 +00:00
|
|
|
|
2024-02-29 20:09:43 +00:00
|
|
|
# Flag based settings are deprecated, use the `--config` flag with a
|
|
|
|
# `KubeletConfiguration` struct.
|
|
|
|
# https://kubernetes.io/docs/tasks/administer-cluster/kubelet-config-file/
|
|
|
|
#
|
|
|
|
# NOTE: registerWithTaints requires a []core/v1.Taint, therefore requires
|
|
|
|
# additional work to be put in config format.
|
|
|
|
#
|
|
|
|
kubeletConfig = pkgs.writeText "kubelet-config" (builtins.toJSON ({
|
|
|
|
apiVersion = "kubelet.config.k8s.io/v1beta1";
|
|
|
|
kind = "KubeletConfiguration";
|
|
|
|
address = cfg.address;
|
|
|
|
port = cfg.port;
|
|
|
|
authentication = {
|
|
|
|
x509 = lib.optionalAttrs (cfg.clientCaFile != null) { clientCAFile = cfg.clientCaFile; };
|
|
|
|
webhook = {
|
|
|
|
enabled = true;
|
|
|
|
cacheTTL = "10s";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
authorization = {
|
|
|
|
mode = "Webhook";
|
|
|
|
};
|
|
|
|
cgroupDriver = "systemd";
|
|
|
|
hairpinMode = "hairpin-veth";
|
|
|
|
registerNode = cfg.registerNode;
|
|
|
|
containerRuntimeEndpoint = cfg.containerRuntimeEndpoint;
|
|
|
|
healthzPort = cfg.healthz.port;
|
|
|
|
healthzBindAddress = cfg.healthz.bind;
|
|
|
|
} // lib.optionalAttrs (cfg.tlsCertFile != null) { tlsCertFile = cfg.tlsCertFile; }
|
|
|
|
// lib.optionalAttrs (cfg.tlsKeyFile != null) { tlsPrivateKeyFile = cfg.tlsKeyFile; }
|
|
|
|
// lib.optionalAttrs (cfg.clusterDomain != "") { clusterDomain = cfg.clusterDomain; }
|
|
|
|
// lib.optionalAttrs (cfg.clusterDns != "") { clusterDNS = [ cfg.clusterDns ] ; }
|
|
|
|
// lib.optionalAttrs (cfg.featureGates != []) { featureGates = cfg.featureGates; }
|
|
|
|
));
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
manifestPath = "kubernetes/manifests";
|
|
|
|
|
|
|
|
taintOptions = with lib.types; { name, ... }: {
|
|
|
|
options = {
|
|
|
|
key = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Key of taint.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = name;
|
2022-09-09 14:08:57 +00:00
|
|
|
defaultText = literalMD "Name of this submodule.";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = str;
|
|
|
|
};
|
|
|
|
value = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Value of taint.";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = str;
|
|
|
|
};
|
|
|
|
effect = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Effect of taint.";
|
2020-04-24 23:36:52 +00:00
|
|
|
example = "NoSchedule";
|
|
|
|
type = enum ["NoSchedule" "PreferNoSchedule" "NoExecute"];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
taints = concatMapStringsSep "," (v: "${v.key}=${v.value}:${v.effect}") (mapAttrsToList (n: v: v) cfg.taints);
|
|
|
|
in
|
|
|
|
{
|
|
|
|
imports = [
|
|
|
|
(mkRemovedOptionModule [ "services" "kubernetes" "kubelet" "applyManifests" ] "")
|
|
|
|
(mkRemovedOptionModule [ "services" "kubernetes" "kubelet" "cadvisorPort" ] "")
|
|
|
|
(mkRemovedOptionModule [ "services" "kubernetes" "kubelet" "allowPrivileged" ] "")
|
2022-10-21 18:38:19 +00:00
|
|
|
(mkRemovedOptionModule [ "services" "kubernetes" "kubelet" "networkPlugin" ] "")
|
2023-04-29 16:46:19 +00:00
|
|
|
(mkRemovedOptionModule [ "services" "kubernetes" "kubelet" "containerRuntime" ] "")
|
2020-04-24 23:36:52 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
###### interface
|
|
|
|
options.services.kubernetes.kubelet = with lib.types; {
|
|
|
|
|
|
|
|
address = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Kubernetes kubelet info server listening address.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = "0.0.0.0";
|
|
|
|
type = str;
|
|
|
|
};
|
|
|
|
|
|
|
|
clusterDns = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Use alternative DNS.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = "10.1.0.1";
|
|
|
|
type = str;
|
|
|
|
};
|
|
|
|
|
|
|
|
clusterDomain = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Use alternative domain.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = config.services.kubernetes.addons.dns.clusterDomain;
|
2021-12-19 01:06:50 +00:00
|
|
|
defaultText = literalExpression "config.${options.services.kubernetes.addons.dns.clusterDomain}";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = str;
|
|
|
|
};
|
|
|
|
|
|
|
|
clientCaFile = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Kubernetes apiserver CA file for client authentication.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = top.caFile;
|
2021-12-19 01:06:50 +00:00
|
|
|
defaultText = literalExpression "config.${otop.caFile}";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = nullOr path;
|
|
|
|
};
|
|
|
|
|
|
|
|
cni = {
|
|
|
|
packages = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "List of network plugin packages to install.";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = listOf package;
|
|
|
|
default = [];
|
|
|
|
};
|
|
|
|
|
|
|
|
config = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Kubernetes CNI configuration.";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = listOf attrs;
|
|
|
|
default = [];
|
2021-10-06 13:57:05 +00:00
|
|
|
example = literalExpression ''
|
2020-04-24 23:36:52 +00:00
|
|
|
[{
|
|
|
|
"cniVersion": "0.3.1",
|
|
|
|
"name": "mynet",
|
|
|
|
"type": "bridge",
|
|
|
|
"bridge": "cni0",
|
|
|
|
"isGateway": true,
|
|
|
|
"ipMasq": true,
|
|
|
|
"ipam": {
|
|
|
|
"type": "host-local",
|
|
|
|
"subnet": "10.22.0.0/16",
|
|
|
|
"routes": [
|
|
|
|
{ "dst": "0.0.0.0/0" }
|
|
|
|
]
|
|
|
|
}
|
|
|
|
} {
|
|
|
|
"cniVersion": "0.3.1",
|
|
|
|
"type": "loopback"
|
|
|
|
}]
|
|
|
|
'';
|
|
|
|
};
|
|
|
|
|
|
|
|
configDir = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Path to Kubernetes CNI configuration directory.";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = nullOr path;
|
|
|
|
default = null;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2021-03-09 03:18:52 +00:00
|
|
|
containerRuntimeEndpoint = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Endpoint at which to find the container runtime api interface/socket";
|
2021-03-09 03:18:52 +00:00
|
|
|
type = str;
|
2021-05-20 23:08:51 +00:00
|
|
|
default = "unix:///run/containerd/containerd.sock";
|
2021-03-09 03:18:52 +00:00
|
|
|
};
|
|
|
|
|
2024-04-21 15:54:59 +00:00
|
|
|
enable = mkEnableOption "Kubernetes kubelet";
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
extraOpts = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Kubernetes kubelet extra command line options.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = "";
|
2021-05-03 20:48:10 +00:00
|
|
|
type = separatedString " ";
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
featureGates = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "List set of feature gates";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = top.featureGates;
|
2021-12-19 01:06:50 +00:00
|
|
|
defaultText = literalExpression "config.${otop.featureGates}";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = listOf str;
|
|
|
|
};
|
|
|
|
|
|
|
|
healthz = {
|
|
|
|
bind = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Kubernetes kubelet healthz listening address.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = "127.0.0.1";
|
|
|
|
type = str;
|
|
|
|
};
|
|
|
|
|
|
|
|
port = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Kubernetes kubelet healthz port.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = 10248;
|
2022-12-17 10:02:37 +00:00
|
|
|
type = port;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
hostname = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Kubernetes kubelet hostname override.";
|
2022-11-21 17:40:18 +00:00
|
|
|
defaultText = literalExpression "config.networking.fqdnOrHostName";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = str;
|
|
|
|
};
|
|
|
|
|
2022-01-13 20:06:32 +00:00
|
|
|
kubeconfig = top.lib.mkKubeConfigOptions "Kubelet";
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
manifests = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "List of manifests to bootstrap with kubelet (only pods can be created as manifest entry)";
|
2020-04-24 23:36:52 +00:00
|
|
|
type = attrsOf attrs;
|
|
|
|
default = {};
|
|
|
|
};
|
|
|
|
|
|
|
|
nodeIp = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "IP address of the node. If set, kubelet will use this IP address for the node.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = null;
|
|
|
|
type = nullOr str;
|
|
|
|
};
|
|
|
|
|
|
|
|
registerNode = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Whether to auto register kubelet with API server.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = true;
|
|
|
|
type = bool;
|
|
|
|
};
|
|
|
|
|
|
|
|
port = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Kubernetes kubelet info server listening port.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = 10250;
|
2022-12-17 10:02:37 +00:00
|
|
|
type = port;
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
seedDockerImages = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "List of docker images to preload on system";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = [];
|
|
|
|
type = listOf package;
|
|
|
|
};
|
|
|
|
|
|
|
|
taints = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Node taints (https://kubernetes.io/docs/concepts/configuration/assign-pod-node/).";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = {};
|
|
|
|
type = attrsOf (submodule [ taintOptions ]);
|
|
|
|
};
|
|
|
|
|
|
|
|
tlsCertFile = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "File containing x509 Certificate for HTTPS.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = null;
|
|
|
|
type = nullOr path;
|
|
|
|
};
|
|
|
|
|
|
|
|
tlsKeyFile = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "File containing x509 private key matching tlsCertFile.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = null;
|
|
|
|
type = nullOr path;
|
|
|
|
};
|
|
|
|
|
|
|
|
unschedulable = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = "Whether to set node taint to unschedulable=true as it is the case of node that has only master role.";
|
2020-04-24 23:36:52 +00:00
|
|
|
default = false;
|
|
|
|
type = bool;
|
|
|
|
};
|
|
|
|
|
|
|
|
verbosity = mkOption {
|
2024-04-21 15:54:59 +00:00
|
|
|
description = ''
|
2020-04-24 23:36:52 +00:00
|
|
|
Optional glog verbosity level for logging statements. See
|
2022-08-12 12:06:08 +00:00
|
|
|
<https://github.com/kubernetes/community/blob/master/contributors/devel/logging.md>
|
2020-04-24 23:36:52 +00:00
|
|
|
'';
|
|
|
|
default = null;
|
|
|
|
type = nullOr int;
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
###### implementation
|
|
|
|
config = mkMerge [
|
|
|
|
(mkIf cfg.enable {
|
2021-03-09 03:18:52 +00:00
|
|
|
|
|
|
|
environment.etc."cni/net.d".source = cniConfig;
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
services.kubernetes.kubelet.seedDockerImages = [infraContainer];
|
|
|
|
|
2021-03-09 03:18:52 +00:00
|
|
|
boot.kernel.sysctl = {
|
|
|
|
"net.bridge.bridge-nf-call-iptables" = 1;
|
|
|
|
"net.ipv4.ip_forward" = 1;
|
|
|
|
"net.bridge.bridge-nf-call-ip6tables" = 1;
|
|
|
|
};
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
systemd.services.kubelet = {
|
|
|
|
description = "Kubernetes Kubelet Service";
|
|
|
|
wantedBy = [ "kubernetes.target" ];
|
2021-03-09 03:18:52 +00:00
|
|
|
after = [ "containerd.service" "network.target" "kube-apiserver.service" ];
|
2021-02-05 17:12:51 +00:00
|
|
|
path = with pkgs; [
|
|
|
|
gitMinimal
|
|
|
|
openssh
|
|
|
|
util-linux
|
2021-04-08 16:26:57 +00:00
|
|
|
iproute2
|
2021-02-05 17:12:51 +00:00
|
|
|
ethtool
|
|
|
|
thin-provisioning-tools
|
|
|
|
iptables
|
|
|
|
socat
|
|
|
|
] ++ lib.optional config.boot.zfs.enabled config.boot.zfs.package ++ top.path;
|
2020-04-24 23:36:52 +00:00
|
|
|
preStart = ''
|
|
|
|
${concatMapStrings (img: ''
|
2021-03-09 03:18:52 +00:00
|
|
|
echo "Seeding container image: ${img}"
|
|
|
|
${if (lib.hasSuffix "gz" img) then
|
2021-03-16 09:55:35 +00:00
|
|
|
''${pkgs.gzip}/bin/zcat "${img}" | ${pkgs.containerd}/bin/ctr -n k8s.io image import --all-platforms -''
|
2021-03-09 03:18:52 +00:00
|
|
|
else
|
2021-03-16 09:55:35 +00:00
|
|
|
''${pkgs.coreutils}/bin/cat "${img}" | ${pkgs.containerd}/bin/ctr -n k8s.io image import --all-platforms -''
|
2021-03-09 03:18:52 +00:00
|
|
|
}
|
2020-04-24 23:36:52 +00:00
|
|
|
'') cfg.seedDockerImages}
|
|
|
|
|
|
|
|
rm /opt/cni/bin/* || true
|
|
|
|
${concatMapStrings (package: ''
|
|
|
|
echo "Linking cni package: ${package}"
|
|
|
|
ln -fs ${package}/bin/* /opt/cni/bin
|
|
|
|
'') cfg.cni.packages}
|
|
|
|
'';
|
|
|
|
serviceConfig = {
|
|
|
|
Slice = "kubernetes.slice";
|
|
|
|
CPUAccounting = true;
|
|
|
|
MemoryAccounting = true;
|
|
|
|
Restart = "on-failure";
|
|
|
|
RestartSec = "1000ms";
|
|
|
|
ExecStart = ''${top.package}/bin/kubelet \
|
2024-02-29 20:09:43 +00:00
|
|
|
--config=${kubeletConfig} \
|
2020-04-24 23:36:52 +00:00
|
|
|
--hostname-override=${cfg.hostname} \
|
|
|
|
--kubeconfig=${kubeconfig} \
|
|
|
|
${optionalString (cfg.nodeIp != null)
|
|
|
|
"--node-ip=${cfg.nodeIp}"} \
|
|
|
|
--pod-infra-container-image=pause \
|
|
|
|
${optionalString (cfg.manifests != {})
|
|
|
|
"--pod-manifest-path=/etc/${manifestPath}"} \
|
|
|
|
${optionalString (taints != "")
|
|
|
|
"--register-with-taints=${taints}"} \
|
|
|
|
--root-dir=${top.dataDir} \
|
|
|
|
${optionalString (cfg.verbosity != null) "--v=${toString cfg.verbosity}"} \
|
|
|
|
${cfg.extraOpts}
|
|
|
|
'';
|
|
|
|
WorkingDirectory = top.dataDir;
|
|
|
|
};
|
2021-08-23 08:02:39 +00:00
|
|
|
unitConfig = {
|
|
|
|
StartLimitIntervalSec = 0;
|
|
|
|
};
|
2020-04-24 23:36:52 +00:00
|
|
|
};
|
|
|
|
|
2023-05-24 13:37:59 +00:00
|
|
|
# Always include cni plugins
|
2021-09-18 10:52:07 +00:00
|
|
|
services.kubernetes.kubelet.cni.packages = [pkgs.cni-plugins pkgs.cni-plugin-flannel];
|
2020-04-24 23:36:52 +00:00
|
|
|
|
2021-03-09 03:18:52 +00:00
|
|
|
boot.kernelModules = ["br_netfilter" "overlay"];
|
2020-04-24 23:36:52 +00:00
|
|
|
|
2022-11-21 17:40:18 +00:00
|
|
|
services.kubernetes.kubelet.hostname =
|
2024-07-01 15:47:52 +00:00
|
|
|
mkDefault (lib.toLower config.networking.fqdnOrHostName);
|
2020-04-24 23:36:52 +00:00
|
|
|
|
2022-01-13 20:06:32 +00:00
|
|
|
services.kubernetes.pki.certs = with top.lib; {
|
2020-04-24 23:36:52 +00:00
|
|
|
kubelet = mkCert {
|
|
|
|
name = "kubelet";
|
|
|
|
CN = top.kubelet.hostname;
|
|
|
|
action = "systemctl restart kubelet.service";
|
|
|
|
|
|
|
|
};
|
|
|
|
kubeletClient = mkCert {
|
|
|
|
name = "kubelet-client";
|
|
|
|
CN = "system:node:${top.kubelet.hostname}";
|
|
|
|
fields = {
|
|
|
|
O = "system:nodes";
|
|
|
|
};
|
|
|
|
action = "systemctl restart kubelet.service";
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
services.kubernetes.kubelet.kubeconfig.server = mkDefault top.apiserverAddress;
|
|
|
|
})
|
|
|
|
|
|
|
|
(mkIf (cfg.enable && cfg.manifests != {}) {
|
|
|
|
environment.etc = mapAttrs' (name: manifest:
|
|
|
|
nameValuePair "${manifestPath}/${name}.json" {
|
|
|
|
text = builtins.toJSON manifest;
|
|
|
|
mode = "0755";
|
|
|
|
}
|
|
|
|
) cfg.manifests;
|
|
|
|
})
|
|
|
|
|
|
|
|
(mkIf (cfg.unschedulable && cfg.enable) {
|
|
|
|
services.kubernetes.kubelet.taints.unschedulable = {
|
|
|
|
value = "true";
|
|
|
|
effect = "NoSchedule";
|
|
|
|
};
|
|
|
|
})
|
|
|
|
|
|
|
|
];
|
2022-01-13 20:06:32 +00:00
|
|
|
|
|
|
|
meta.buildDocsInSandbox = false;
|
2020-04-24 23:36:52 +00:00
|
|
|
}
|