{ stdenv , lib , makeWrapper , socat , iptables , iproute2 , bridge-utils , conntrack-tools , buildGoPackage , runc , kmod , libseccomp , pkg-config , ethtool , util-linux , fetchFromGitHub , fetchurl , fetchzip , fetchgit , zstd }: with lib; # k3s is a kinda weird derivation. One of the main points of k3s is the # simplicity of it being one binary that can perform several tasks. # However, when you have a good package manager (like nix), that doesn't # actually make much of a difference; you don't really care if it's one binary # or 10 since with a good package manager, installing and running it is # identical. # Since upstream k3s packages itself as one large binary with several # "personalities" (in the form of subcommands like 'k3s agent' and 'k3s # kubectl'), it ends up being easiest to mostly mimic upstream packaging, with # some exceptions. # K3s also carries patches to some packages (such as containerd and cni # plugins), so we intentionally use the k3s versions of those binaries for k3s, # even if the upstream version of those binaries exist in nixpkgs already. In # the end, that means we have a thick k3s binary that behaves like the upstream # one for the most part. # However, k3s also bundles several pieces of unpatched software, from the # strongswan vpn software, to iptables, to socat, conntrack, busybox, etc. # Those pieces of software we entirely ignore upstream's handling of, and just # make sure they're in the path if desired. let k3sVersion = "1.21.4+k3s1"; # k3s git tag k3sCommit = "3e250fdbab72d88f7e6aae57446023a0567ffc97"; # k3s git commit at the above version k3sRepoSha256 = "1w7drvk0bmlmqrxh1y6dxjy7dk6bdrl72pkd25lc1ir6wbzb05h9"; traefikChartVersion = "9.18.2"; # taken from ./manifests/traefik.yaml at spec.version traefikChartSha256 = "sha256-9d7p0ngyMN27u4OPgz7yI14Zj9y36t9o/HMX5wyDpUI="; k3sRootVersion = "0.9.1"; # taken from ./scripts/download at ROOT_VERSION k3sRootSha256 = "sha256-qI84KYJKY/T6pqWZW9lOTq5NzZiu//v1zrMzUCiRTGQ="; k3sCNIVersion = "0.8.6-k3s1"; # taken from ./scripts/version.sh at VERSION_CNIPLUGINS k3sCNISha256 = "sha256-uAy17eRRAXPCcnh481KxFMvFQecnnBs24jn5YnVNfY4="; baseMeta = { description = "A lightweight Kubernetes distribution"; license = licenses.asl20; homepage = "https://k3s.io"; maintainers = with maintainers; [ euank superherointj ]; platforms = platforms.linux; }; # bundled into the k3s binary traefikChart = fetchurl { url = "https://helm.traefik.io/traefik/traefik-${traefikChartVersion}.tgz"; sha256 = traefikChartSha256; }; # so, k3s is a complicated thing to package # This derivation attempts to avoid including any random binaries from the # internet. k3s-root is _mostly_ binaries built to be bundled in k3s (which # we don't care about doing, we can add those as build or runtime # dependencies using a real package manager). # In addition to those binaries, it's also configuration though (right now # mostly strongswan configuration), and k3s does use those files. # As such, we download it in order to grab 'etc' and bundle it into the final # k3s binary. k3sRoot = fetchzip { # Note: marked as apache 2.0 license url = "https://github.com/k3s-io/k3s-root/releases/download/v${k3sRootVersion}/k3s-root-amd64.tar"; sha256 = k3sRootSha256; stripRoot = false; }; k3sPlugins = buildGoPackage rec { name = "k3s-cni-plugins"; version = k3sCNIVersion; goPackagePath = "github.com/containernetworking/plugins"; subPackages = [ "." ]; src = fetchFromGitHub { owner = "rancher"; repo = "plugins"; rev = "v${version}"; sha256 = k3sCNISha256; }; meta = baseMeta // { description = "CNI plugins, as patched by rancher for k3s"; }; }; # Grab this separately from a build because it's used by both stages of the # k3s build. k3sRepo = fetchgit { url = "https://github.com/k3s-io/k3s"; rev = "v${k3sVersion}"; sha256 = k3sRepoSha256; }; # Stage 1 of the k3s build: # Let's talk about how k3s is structured. # One of the ideas of k3s is that there's the single "k3s" binary which can # do everything you need, from running a k3s server, to being a worker node, # to running kubectl. # The way that actually works is that k3s is a single go binary that contains # a bunch of bindata that it unpacks at runtime into directories (either the # user's home directory or /var/lib/rancher if run as root). # This bindata includes both binaries and configuration. # In order to let nixpkgs do all its autostripping/patching/etc, we split this into two derivations. # First, we build all the binaries that get packed into the thick k3s binary # (and output them from one derivation so they'll all be suitably patched up). # Then, we bundle those binaries into our thick k3s binary and use that as # the final single output. # This approach was chosen because it ensures the bundled binaries all are # correctly built to run with nix (we can lean on the existing buildGoPackage # stuff), and we can again lean on that tooling for the final k3s binary too. # Other alternatives would be to manually run the # strip/patchelf/remove-references step ourselves in the installPhase of the # derivation when we've built all the binaries, but haven't bundled them in # with generated bindata yet. k3sBuildStage1 = buildGoPackage rec { name = "k3s-build-1"; version = k3sVersion; goPackagePath = "github.com/rancher/k3s"; src = k3sRepo; # Patch build scripts so that we can use them. # This makes things more dynamically linked (because nix can deal with # dynamically linked dependencies just fine), removes the upload at the # end, and skips building runc + cni, since we have our own derivations for # those. patches = [ ./patches/0002-Add-nixpkgs-patches.patch ]; nativeBuildInputs = [ pkg-config ]; buildInputs = [ libseccomp ]; # Versioning info for build script DRONE_TAG = "v${version}"; DRONE_COMMIT = k3sCommit; buildPhase = '' pushd go/src/${goPackagePath} patchShebangs ./scripts/build ./scripts/version.sh mkdir -p bin ./scripts/build popd ''; installPhase = '' pushd go/src/${goPackagePath} mkdir -p "$out/bin" install -m 0755 -t "$out/bin" ./bin/* popd ''; meta = baseMeta // { description = "The various binaries that get packaged into the final k3s binary"; }; }; k3sBin = buildGoPackage rec { name = "k3s-bin"; version = k3sVersion; goPackagePath = "github.com/rancher/k3s"; src = k3sRepo; # See the above comment in k3sBuildStage1 patches = [ ./patches/0002-Add-nixpkgs-patches.patch ]; nativeBuildInputs = [ pkg-config zstd ]; # These dependencies are embedded as compressed files in k3s at runtime. # Propagate them to avoid broken runtime references to libraries. propagatedBuildInputs = [ k3sPlugins k3sBuildStage1 runc ]; # k3s appends a suffix to the final distribution binary for some arches archSuffix = if stdenv.hostPlatform.system == "x86_64-linux" then "" else if stdenv.hostPlatform.system == "aarch64-linux" then "-arm64" else throw "k3s isn't being built for ${stdenv.hostPlatform.system} yet."; DRONE_TAG = "v${version}"; DRONE_COMMIT = k3sCommit; # In order to build the thick k3s binary (which is what # ./scripts/package-cli does), we need to get all the binaries that script # expects in place. buildPhase = '' pushd go/src/${goPackagePath} patchShebangs ./scripts/build ./scripts/version.sh ./scripts/package-cli mkdir -p bin install -m 0755 -t ./bin ${k3sBuildStage1}/bin/* install -m 0755 -T "${k3sPlugins}/bin/plugins" ./bin/cni # Note: use the already-nixpkgs-bundled k3s rather than the one bundled # in k3s because the k3s one is completely unmodified from upstream # (unlike containerd, cni, etc) install -m 0755 -T "${runc}/bin/runc" ./bin/runc cp -R "${k3sRoot}/etc" ./etc mkdir -p "build/static/charts" cp "${traefikChart}" "build/static/charts/traefik-${traefikChartVersion}.tgz" ./scripts/package-cli popd ''; installPhase = '' pushd go/src/${goPackagePath} mkdir -p "$out/bin" install -m 0755 -T ./dist/artifacts/k3s${archSuffix} "$out/bin/k3s" popd ''; meta = baseMeta // { description = "The k3s go binary which is used by the final wrapped output below"; }; }; in stdenv.mkDerivation rec { pname = "k3s"; version = k3sVersion; # `src` here is a workaround for the updateScript bot. It couldn't be empty. src = builtins.filterSource (path: type: false) ./.; # Important utilities used by the kubelet, see # https://github.com/kubernetes/kubernetes/issues/26093#issuecomment-237202494 # Note the list in that issue is stale and some aren't relevant for k3s. k3sRuntimeDeps = [ kmod socat iptables iproute2 bridge-utils ethtool util-linux # kubelet wants 'nsenter' from util-linux: https://github.com/kubernetes/kubernetes/issues/26093#issuecomment-705994388 conntrack-tools ]; buildInputs = [ k3sBin ] ++ k3sRuntimeDeps; nativeBuildInputs = [ makeWrapper ]; unpackPhase = "true"; # And, one final derivation (you thought the last one was it, right?) # We got the binary we wanted above, but it doesn't have all the runtime # dependencies k8s wants, including mount utilities for kubelet, networking # tools for cni/kubelet stuff, etc # Use a wrapper script to reference all the binaries that k3s tries to # execute, but that we didn't bundle with it. installPhase = '' runHook preInstall mkdir -p "$out/bin" makeWrapper ${k3sBin}/bin/k3s "$out/bin/k3s" \ --prefix PATH : ${lib.makeBinPath k3sRuntimeDeps} \ --prefix PATH : "$out/bin" runHook postInstall ''; doInstallCheck = true; installCheckPhase = '' $out/bin/k3s --version | grep v${k3sVersion} > /dev/null ''; passthru.updateScript = ./update.sh; meta = baseMeta; }