𧬠GitOps & Flux
This is an advanced optional section going into two topics; Kustomize and also GitOps, using FluxCD.
πͺ Kustomize
Kustomize is a tool for customizing Kubernetes configurations.
Kustomize traverses Kubernetes manifests to add, remove or update configuration options. It is available both as a standalone binary and as a native feature of kubectl. It can be thought of as similar to Helm where it provides a means to template and parameterize Kubernetes manifests.
Kustomize works by looking for kustomization.yaml
files and operating on their contents.
These slides provide a fairly good introduction.
To demonstrate Kustomize in practice, we can carry out a simple exercise, create a new directory
called base
.
Place the the following two files into it:
Contents of base/deployment.yaml
apiVersion: apps/v1
kind: Deployment
metadata:
name: webserver
spec:
selector:
matchLabels:
app: webserver
template:
metadata:
labels:
app: webserver
spec:
containers:
- name: webserver
image: nginx
resources:
limits:
memory: "128Mi"
cpu: "500m"
ports:
- containerPort: 80
Contents of base/kustomization.yaml
apiVersion: kustomize.config.k8s.io/v1beta1
kind: Kustomization
resources:
- deployment.yaml
Now run kustomize via kubectl, giving it the path to the base directory as follows:
kubectl kustomize ./base
You will see the YAML printed to stdout, as weβve not provided any changes in the kustomization.yaml
all we get is a 1:1 version of the deployment.yaml
file. This isnβt very useful! π¬
To better understand what Kustomize can do, create a second directory at the same level as base
called overlay
.
Contents of overlay/override.yaml
apiVersion: apps/v1
kind: Deployment
metadata:
name: webserver
spec:
template:
spec:
containers:
- name: webserver
resources:
limits:
cpu: 330m
env:
- name: SOME_ENV_VAR
value: Hello!
Contents of overlay/kustomization.yaml
apiVersion: kustomize.config.k8s.io/v1beta1
kind: Kustomization
# Reference to a base kustomization directory
resources:
- ../base
# You can add suffixes and prefixes
nameSuffix: -dev
# Modify the image name or tags
images:
- name: nginx
newTag: 1.21-alpine
# Apply patches to override and set other values
patches:
- ./override.yaml
Some points to highlight:
- The Kustomization adds a suffix to the names of resources.
- Also the Kustomization changes the image tag to reference a specific tag.
- The patch
override.yaml
file looks a little like a regular Kubernetes Deployment but it only contains the part that will be patched/overlayed onto the base resource. On its own itβs not a valid manifest.- The patch file sets fields in the base Deployment such as changing the resource limits and adding an extra environmental variable.
See the reference docs for
all the options available in the kustomization.yaml
file.
The file & directory structure should look as follows:
βββ base
β βββ deployment.yaml
β βββ kustomization.yaml
βββ overlay
βββ kustomization.yaml
βββ override.yaml
π NOTE: The names βbaseβ and βoverlayβ are not special, often βenvironmentsβ is used instead of βoverlayβ, with sub-directories for each environment
Now running:
kubectl kustomize ./overlay
You will now see the overrides and modifications from the overlay applied to the base resources. With the modified nginx image tag, different resource limits and additional env var.
This could be applied to the cluster with the following command kubectl -k ./overlay apply
, however
there is no need to do this.
An interesting feature of kustomize you may want to check out is variable substitution.
GitOps & Flux
GitOps is a methodology where you declaratively describe the entire desired state of your system using git. This includes the apps, config, dashboards, monitoring and everything else. This means you can use git branches and PR processes to enforce control of releases and provide traceability and transparency.
Kubernetes doesnβt support this concept out of the box, it requires special controllers to be deployed and manage this process. These controllers run inside the cluster, monitor git repositories for changes and then make the required updates to the state of the cluster, through a process called reconciliation.
We will use the popular project FluxCD (also just called Flux or Flux v2), however other projects are available such as ArgoCD and support from GitLab.
As GitOps is a βpullβ vs βpushβ approach, it also allows you to lock down your Kubernetes cluster, and prevent developers and admins making direct changes with kubectl.
π NOTE: GitOps is a methodology and an approach, it is not the name of a product.
π½ Install Flux into K3s VM
You can install the Flux CLI with:
curl -s https://fluxcd.io/install.sh | sudo bash
# Flux auto complete to .bashrc
echo "command -v flux >/dev/null && . <(flux completion bash)" >> ~/.bashrc
. ~/.bashrc
Before we configure anything GitOps needs a git repo to work against. Weβll use a fork of this repo, to set this up:
- Got to the repo for this workshop https://github.com/benc-uk/kube-workshop
- Fork the repo to your own personal GitHub account, by clicking the βForkβ button near the top right.
Now to install and set up Flux in your cluster, run the following command, replacing the {YOUR_GITHUB_USER}
part with your GitHub username you used for the fork:
# Install flux in the cluster, create flux pods, ect.
flux install
flux create source git kubeworkshop \
--url="https://github.com/{YOUR_GITHUB_USER}/kube-workshop" \
--branch=main \
--interval=1m
flux create kustomization apps \
--path="gitops/apps" \
--source=kubeworkshop \
--prune=true \
--interval=1m
Check the status of Flux with the following commands:
kubectl get kustomizations -A
flux get kustomization
kubectl get gitrepo -A
kubectl get pod -n flux-system
Good for troubleshooting:
flux logs
kubectl get events -n flux-system
More tips and tricks: Flux Troubleshooting cheatsheet.
You should also see a new namespace called βhello-worldβ, check with kubectl get ns
this has been
created by the gitops/apps/hello-world.yaml
file in the repo and automatically applied by Flux.
In addition, your cluster now has flux components installed, such as pods, which you can view with
kubectl get pods -n flux-system
.
π Deploying Resources
Clone the kube-workshop repo you forked earlier and open the directory in VS Code or other editor.
If you recall from the bootstrap command earlier we gave Flux a path within the repo to use and look
for configurations, which was gitops/apps
directory. The contents of the whole of the gitops
directory is shown here.
gitops
βββ apps
β βββ hello-world.yaml
βββ base
β βββ data-api
β β βββ deployment.yaml
β β βββ kustomization.yaml
β β βββ service.yaml
β βββ frontend
β β βββ deployment.yaml
β β βββ ingress.yaml
β β βββ kustomization.yaml
β β βββ service.yaml
β βββ mongodb
β βββ kustomization.yaml
β βββ mongo-statefulset.yaml
βββ disabled-k3s
βββ mongodb
β βββ kustomization.yaml
β βββ overrides.yaml
βββ smilr
βββ kustomization.yaml
The base directory provides us a library of Kustomization based resources we can use, but as itβs
outside of the gitops/apps
path they will not be picked up by Flux.
β οΈ STOP! Before we proceed, ensure the mongo-creds
Secret from the previous sections is still
in the default namespace. If you have deleted it, hop back to section 7
and quickly create it again. Itβs just a single command. Creating Secrets using the GitOps approach
is problematic, as they need to be committed into a code repo. Flux supports solutions to this, such
as using SOPS and
Sealed Secrets. For an intro such as this workshop,
they require too much extra setup, so we will skip over them.
First letβs deploy MongoDB using Flux:
- Copy the
monogodb/
directory from βdisabled-k3sβ to βappsβ.- Note the
kustomization.yaml
in here is pointing at the base directory../../base/mongodb
and overlaying it.
- Note the
git commit
these changes to the main branch and push up to GitHub.- Wait for ~1 minute for Flux to rescan the git repo.
- Check for any errors with
kubectl get kustomizations -A
. - Check the default namespace for the new MongoDB StatefulSet and Pod using
kubectl get sts,pods -n default
.
Next deploy the Smilr app:
- Copy the
smilr/
directory fromdisabled-k3s
toapps
.- Note the
kustomization.yaml
in here is pointing at several base directories, for the appβs data-api and frontend.
- Note the
- Edit the ACR name in the
gitops/apps/smilr/kustomization.yaml
file. git commit
these changes to the main branch and push up to GitHub.- Wait for ~1 minute for Flux to rescan the git repo.
- Check for any errors with
kubectl get kustomizations -A
. - Check the default namespace for the new resources using
kubectl get deploy,pods,ingress -n default
.
In the smilr
folder weβre using kustomize patching
to modify the deployments to work on our k3s clusters.
If you encounter problems or want to force the reconciliation you can use the flux
CLI, e.g.
flux reconcile source git kubeworkshop
.
If we wanted to deploy this app across multiple environments or multiple times, we could create
sub-directories under apps/
, each containing different Kustomizations and modifying the deployment
to suit that environment.
π§ͺ Experiment: Try deleting one of the deployments and watch it be brought back to life with
flux
reconcile. You can speed up the recreation with flux reconcile kustomization apps
.
Navigation
Return to Main Index π Previous Section βͺ β Next Section β©