There are over 90 Certified Kubernetes offerings as of this blog publication. One such project currently under Linux Foundation is k3s and below quote is directly taken from it’s landing page:
K3s is a highly available, certified Kubernetes distribution designed for production workloads in unattended, resource-constrained, remote locations or inside IoT appliances.
Clubbing two words resource-constrained
and IoT
from the quote we can infer that this distribution is also best suited for setting up our local Kubernetes cluster. Going a step further and inspired by KinD
project, k3d
is created which runs k3s
Kubernetes distribution in docker and as a result k3d
has a single dependency which is docker.
Pre-requisites §
Conceptual knowledge of docker and kubernetes to utilize k3d
created cluster, you can refer below resources*:
- Docker: Video , Installation
- Kubernetes: Video or Book
*Non-affiliated, I referred all these resources atleast once and so recommending them
Although docker is updated to work with cgroup v2, I was only able to setup k3d
cluster on falling back to cgroup v1 using below method:
- To cgroup v1:
dnf install grubby && grubby --update-kernel=ALL --args="systemd.unified_cgroup_hierarchy=0" && reboot
- To switch back to cgroup v2:
grubby --update-kernel=ALL --args="systemd.unified_cgroup_hierarchy" && reboot
Fedora recommended option is to use podman and buildah however the steps in this post tested using docker.
I used Fedora 32 Server Edition with 8Gi of RAM, /var
parition mounted on a 100GB (having free space on /var
partition never hurts 😄) disk, tested installation and subsequent operations as root
(just having sudo
access also suffices) user. YMMV may vary depending on how well-versed you are with your machine.
Installing Binary §
k3d version 4 is preferred as it has k3d-managed
registry and it comes handy to create a repository along with cluster creation with no extra steps.
-> curl -OL https://github.com/rancher/k3d/releases/download/v4.2.0/k3d-linux-amd64
-> chmod +x k3d-linux-amd64
-> mv k3d-linux-amd64 /usr/local/bin/k3d
After downloading binary you can verify the verison and that’s all it needs to create a cluster
-> k3d version
k3d version v4.2.0
k3s version v1.20.2-k3s1 (default)
Cluster Operations §
Let’s go through lifecycle of a k3d cluster and later we can move on to customizing the cluster to our needs. Please refer docs for command tree
|
|
We can infer below from creating the cluster:
- We got a loadbalancer(nginx lb) with the cluster (line 15) and can be reached at port 38871 on localhost (lines 19, 29)
- We can provide
--api-port PORT
while creating a cluster to make sure lb always use that internally - k3d uses
containerd
runtime for running containers (lines 35, 42) - We can’t share local docker images directly with k3d nodes, either we need to save local images to tar and import into k3d cluster or create a local registry
- For accessing services from pods deployed in k3d cluster, we need to deploy ingress rules, controller and thus should have a rough idea of the services that we’ll be using before creating the cluster itself or for testing/debugging we can use
kubectl port-forward
functionality
I highly recommend going through the docs for customizing the cluster during creation, as I’m fine with defaults and mostly concerned with storage I didn’t explore networking and other components enough in k3d to blog about.
As we don’t want to pull images always from remote repository we’ll be concentrating on point 4 from above for using local docker images.
NOTE: If k3d nodes are tainted, tolerations inhibit pod scheduling, before scheduling any pods and after verifying nodes are online, remove taints by running following command:
# Verify presence of taints on the nodes (jq is a command line JSON Processor)
-> kubectl get nodes -o json | jq '.items[].spec.taints'
[
{
"effect": "NoSchedule",
"key": "node.cloudprovider.kubernetes.io/uninitialized",
"value": "true"
}
]
[
{
"effect": "NoSchedule",
"key": "node.cloudprovider.kubernetes.io/uninitialized",
"value": "true"
}
]
[
{
"effect": "NoSchedule",
"key": "node.cloudprovider.kubernetes.io/uninitialized",
"value": "true"
}
]
[
{
"effect": "NoSchedule",
"key": "node.cloudprovider.kubernetes.io/uninitialized",
"value": "true"
}
]
If taints are found in above o/p, remove them by running (’-’ at the end of command unset the value)
# for name in $(kubectl get nodes -o jsonpath={'..name'}); do kubectl taint nodes $name node.cloudprovider.kubernetes.io/uninitialized-; done;
Optimizing workflow §
We’ll look at two scenarios of using local docker images in k3d cluster. One will be docker save and import into cluster, second is using a local registry. Both of the methods have use cases associated.
Save and Import §
Let’s deploy a busybox container, find the image source from k3d container runtime and pull that image locally from docker.
-> bat deployment-busybox.yaml --plain
---
apiVersion: apps/v1
kind: Deployment
metadata:
name: test
spec:
replicas: 1
selector:
matchLabels:
app: test
template:
metadata:
labels:
app: test
spec:
containers:
- name: test-pod
image: busybox
imagePullPolicy: IfNotPresent
command:
- '/bin/tail'
- '-f'
- '/dev/null'
livenessProbe:
exec:
command:
- 'sh'
- '-ec'
- 'df'
initialDelaySeconds: 3
periodSeconds: 3
Apply the manifest and verify pod creation, please not when a k3d cluster is created kubectl context is set to use newly created cluster, thus kubectl
is able to access the Kubernetes API and it’s resources without any manual interventions.
|
|
We can see from above the pod is running on k3d-test-agent-1
, now we’ll query the image existing in that node and pull the image locally from remote repo/hub.
-> docker exec k3d-test-agent-1 sh -c 'ctr image list -q | grep "busybox:latest"'
docker.io/library/busybox:latest
# Incase if you do not know the image name, you can query that pod spec
-> kubectl get pod test-d77db976d-qxsvr -o jsonpath={'..image'}
docker.io/library/busybox:latest busybox
# Pulling image based on ctr images from k3d node
-> for image in $(docker exec k3d-test-agent-1 sh -c 'ctr image list -q | grep "busybox:latest"'); do docker pull $image; done;
latest: Pulling from library/busybox
8b3d7e226fab: Pull complete
Digest: sha256:ce2360d5189a033012fbad1635e037be86f23b65cfd676b436d0931af390a2ac
Status: Downloaded newer image for busybox:latest
docker.io/library/busybox:latest
# (or)
# Pulling image based on currently deployed pods
-> for image in $(kubectl get pod test-d77db976d-qxsvr -o jsonpath="{..image}"); do docker pull $image; done;
Using default tag: latest
latest: Pulling from library/busybox
Digest: sha256:ce2360d5189a033012fbad1635e037be86f23b65cfd676b436d0931af390a2ac
Status: Image is up to date for busybox:latest
docker.io/library/busybox:latest
latest: Pulling from library/busybox
Digest: sha256:ce2360d5189a033012fbad1635e037be86f23b65cfd676b436d0931af390a2ac
Status: Image is up to date for busybox:latest
docker.io/library/busybox:latest
# Verify image exists in local docker
-> docker images | grep busybox
busybox latest a9d583973f65 13 hours ago 1.23MB
busybox stable a9d583973f65 13 hours ago 1.23MB
Now that we have images pulled from repo/hub and exists locally, we can save them in a tar with correct tags and import them into k3d after the cluster is created
-> docker save $(docker images --format '{{.Repository}}:{{.Tag}}' | grep busybox) -o localimages.tar
# Delete earlier created cluster (or) you can create a new cluster and import above created tarball
-> k3d cluster delete test
[...]
-> k3d cluster create test
[...]
# Perform below before deploying busybox
-> k3d image import -k localimages.tar -c test
[...]
After above operation, image existing in the k3d cluster is used for running container. -k
option is for not deleting local tarball once uploaded to cluster and -c
specifies the cluster name.
Use case:
- We deploy a operator manifest and it may in turn pull many images from remote repository, we’ll verify changes/perform operations using k3d cluster and delete the cluster.
- However, all the images pulled by operator will be deleted and next time if we create the cluster we need to pull images again which is not optimal.
Caveat:
- Busybox image is showed just as a demo and above can be followed for any other deployments, however the time increases based on image size
- Atleast the images has to pulled two times before using them in a new cluster and I wasn’t able to find any other way to overcome that
- For images with lesser sizes and latest tags above method may not necessarily optimize network transactions and at the same time we can easily observe gains if we are dealing with larger images with versioned tags.
Local Registry §
As per the docs k3d
has inbuilt capability for creating a registry associated with cluster itself however I’m just using docker for running a local registry and connecting k3d network with registry container.
Let’s take a detour from k3d and use docker to build a image. I don’t want to re-hash the details/best practices about building images as they are explained in greater detail in docker documentation, I recommend going through storagedriver and multistage builds at a minimum.
Let’s build a minimal image which i generally use to verify checksum and create IO in a storage system and I’ll touch upon some docker concepts along the way.
|
|
About above dockerfile:
- We need
gcc
to compileC
src (#6) and evenCrefi
(#12) depends onpython3-dev
package forPython.h
header file, however we just need resulting binary and installed python packages, so we are better off with using multistage builds (#14) - Approach at #11 is quite helpful in isolating system packages and application packages using virtual environment
- As a result of operations at #7 and #11, the required binary and python packages are installed at
/opt/
and we just need this as part of final image - Contents of
/opt
ofcompile
image is copied to/opt
of build image (#16) and PATH is set to/opt
(#18) in final image.
Build the docker image and observe some other details after image creation
|
|
About above image creation:
- As a result of build command (#1) a image is generated from Dockerfile, tagged and stored locally (#23)
- The bulk of the image consists of Python executable (#31), size of binary and packages copied from
compile
stage is relatively less (#29) - If we hadn’t used multistaged build the image size could easily cross 250MB, I’m not after trimming image sizes indefinitely and so the result is good enough.
Well, that’s it for the interlude, coming back to k3d
, we’ll follow below steps in brief:
- Start a local docker registry container, tag above created image
- Push that to local registry, inform
k3d
at the start of cluster creation to take a note of the registry - Use images from the local registry in resource deployments.
# Start registry container
-> docker container run -d --name registry.localhost --restart always -p 5000:5000 registry:2
# Attach container to k3d cluster network
-> docker network connect k3d-test registry.localhost
# Tag local image with local registry
-> docker tag test-fs:latest registry.localhost:5000/test-fs:latest
# Push tagged image to local registry
-> docker push registry.localhost:5000/test-fs:latest
After performing above operations, you can use image: registry.localhost:5000/test-fs:latest
in deployment yaml file and use the image from local registry.
Use case:
- This greatly improves feedback loop while developing/unit testing applications using test pods.
- You can go a step ahead and replace all image definitions in yaml manifests to point to your local registry and never use save and import of images. (It’s quite extreme though 😅)
Caveat:
- I had to edit
dockerd
service file and reload systemctl to allow insecure-registries as adding to usualdaemon.json
file didn’t work for me.
-> bat /usr/lib/systemd/system/docker.service | grep ExecStart
ExecStart=/usr/bin/dockerd -H fd:// --containerd=/run/containerd/containerd.sock --insecure-registry registry.localhost:5000
- Make sure you inform
k3d
about local registry while creating the cluster by mappingregistries.yaml
ink3s
default directory as below:
# Create below file
-> bat ~/.k3d/registries.yaml --plain
mirrors:
"registry.localhost:5000":
endpoint:
- "http://registry.localhost:5000"
# Supply registries.yaml location while creating cluster
-> k3d cluster create test -a 3 -v $HOME/.k3d/registries.yaml:/etc/rancher/k3s/registries.yaml
Well, that brings us to the end of the blog post. I covered only the setup and opinionated workflow while testing/debugging kubernetes workloads in k3d, intentionally left out the usage of local registry in resource deployments as I intend to cover that in a later post.
Stay tuned to learn about Container Storage Interface (CSI ) and how to work with CSI by deploying the resources in a k3d cluster.