DEV Community

ranjit sahoo
ranjit sahoo

Posted on

# Kubernetes for Absolute Beginners: Build Your First Local Cluster with kind

A hands-on, copy-paste guide to going from a bare Linux machine to running, scaling, and exposing real workloads on Kubernetes — no cloud account required.


When I started learning Kubernetes, every tutorial either assumed I already had a cloud
cluster or buried the basics under a mountain of YAML. So I did what most of us do: I
opened a terminal, broke things, fixed them, and wrote down what actually worked.

This post is that notebook, cleaned up. By the end you'll have a real Kubernetes cluster
running on your own laptop with kind, and you'll understand the core building blocks —
Pods, Deployments, Services, Ingress, storage, autoscaling — not as abstract diagrams, but
as commands you can run right now.

Let's get into it.


Step 1: Spin Up a Cluster on Your Own Machine

You don't need AWS or GCP to learn Kubernetes. kind ("Kubernetes IN Docker") runs an
entire cluster inside Docker containers on your machine. It's fast, free, and disposable.

You'll need three tools, and it's worth calling out that they're separate: Docker (kind
runs the cluster inside it), kind itself, and kubectl — the CLI you actually drive
the cluster with. kind does not install kubectl for you, which trips up a lot of
beginners. Grab all three from their official docs; kubectl is a one-liner on Linux:

curl -LO "https://dl.k8s.io/release/$(curl -L -s https://dl.k8s.io/release/stable.txt)/bin/linux/amd64/kubectl"
sudo install -o root -g root -m 0755 kubectl /usr/local/bin/kubectl
Enter fullscreen mode Exit fullscreen mode

On Linux, Docker needs sudo by default. Add yourself to the docker group and refresh
your session so you don't have to keep typing it:

sudo usermod -aG docker $USER && newgrp docker
Enter fullscreen mode Exit fullscreen mode

Now — and this is the part most quick-start guides skip — don't just run
kind create cluster
. A bare cluster has no way to accept web traffic on ports 80/443,
and later when you add an Ingress its controller will sit there Pending forever because no
node is marked as "ingress-ready". You fix both up front with a tiny config file. Save this
as kind-config.yml:

kind: Cluster
apiVersion: kind.x-k8s.io/v1alpha4
nodes:
- role: control-plane
  kubeadmConfigPatches:
  - |
    kind: InitConfiguration
    nodeRegistration:
      kubeletExtraArgs:
        node-labels: "ingress-ready=true"
  extraPortMappings:
  - containerPort: 80
    hostPort: 80
    protocol: TCP
  - containerPort: 443
    hostPort: 443
    protocol: TCP
Enter fullscreen mode Exit fullscreen mode

The extraPortMappings wire your laptop's localhost:80/443 straight into the cluster, and
the ingress-ready=true label gives the ingress controller a node to land on. Create the
cluster with that config:

kind create cluster --name practice --config kind-config.yml
kubectl get nodes
Enter fullscreen mode Exit fullscreen mode

A node in Ready state? You're officially running Kubernetes. Two add-ons round out the
setup — they aren't bundled with kind, so install them now and forget about them:

# Ingress controller (kind-flavored build) — powers the /apache and /nginx routing later
kubectl apply -f https://kind.sigs.k8s.io/examples/ingress/deploy-ingress-nginx.yaml

# metrics-server — required before any autoscaling or `kubectl top` will work
kubectl apply -f https://github.com/kubernetes-sigs/metrics-server/releases/latest/download/components.yaml
Enter fullscreen mode Exit fullscreen mode

(One quirk: on kind, metrics-server usually needs a small --kubelet-insecure-tls tweak to
go healthy — we'll cover that when we reach autoscaling.)


Step 2: Namespaces — Your Cluster's Folders

Before deploying anything, it helps to create a namespace: a logical partition that
keeps your resources tidy and isolated.

kubectl create namespace nginx
Enter fullscreen mode Exit fullscreen mode

You can also do it declaratively in a YAML file — and honestly, the declarative approach
is what you should get comfortable with, because it's how real teams manage clusters:

apiVersion: v1
kind: Namespace
metadata:
  name: nginx
Enter fullscreen mode Exit fullscreen mode
kubectl apply -f namespace.yml
Enter fullscreen mode Exit fullscreen mode

From here on, almost every command takes a -n <namespace> flag to say where it should
act.


Step 3: Pods — The Smallest Unit That Runs

A Pod is the smallest thing Kubernetes runs: one (or a few tightly-coupled) containers
sharing a network and storage. You can launch one directly:

kubectl run nginx --image=nginx -n nginx
Enter fullscreen mode Exit fullscreen mode

But here's the thing nobody tells beginners early enough: you almost never create bare
Pods in real life.
If that pod dies, nothing brings it back. Instead, you let a higher-
level controller — a Deployment — manage Pods for you so they self-heal and scale.


Step 4: Deployments, and Their Cousins

This is where Kubernetes gets powerful. A Deployment doesn't just run your Pods — it
keeps the desired number alive, replaces crashed ones, and rolls out new versions safely.

apiVersion: apps/v1
kind: Deployment
metadata:
  name: nginx-deployment
  namespace: nginx
spec:
  replicas: 3
  selector:
    matchLabels:
      app: nginx
  template:
    metadata:
      labels:
        app: nginx
    spec:
      containers:
      - name: nginx
        image: nginx:latest
        ports:
        - containerPort: 80
Enter fullscreen mode Exit fullscreen mode

Apply it, then scale it with a single command:

kubectl apply -f deployment.yml
kubectl scale deployment/nginx-deployment -n nginx --replicas=5
Enter fullscreen mode Exit fullscreen mode

Once you understand Deployments, three siblings fall into place:

  • ReplicaSet — keeps N identical pods running. Its YAML is almost identical to a Deployment (just a different kind), but it can't do rollouts or rollbacks. In practice a Deployment manages ReplicaSets for you, so you rarely write one yourself.
  • DaemonSet — guarantees one pod on every node in the cluster. Perfect for node-level agents like log shippers or monitoring.
  • StatefulSet — for stateful apps like databases. When a pod crashes, it comes back with the same name and the same storage, unlike a Deployment where replacements get random names. This stable identity is exactly what databases need.

Step 5: Updating Without Downtime (and Undoing Mistakes)

Two concepts often get confused here, so let's be precise — because they're different.

Rolling update is the default Deployment strategy. When you change the image, new pods
start before old ones are removed, so users never see downtime:

kubectl set image deployment/nginx-deployment -n nginx nginx=nginx:1.27.3
kubectl rollout status deployment/nginx-deployment -n nginx
Enter fullscreen mode Exit fullscreen mode

Rollback is the safety net for when an update goes wrong. Kubernetes keeps a history,
so you can revert in seconds:

kubectl rollout history deployment/nginx-deployment -n nginx
kubectl rollout undo deployment/nginx-deployment -n nginx
Enter fullscreen mode Exit fullscreen mode

So: rolling update = how new versions ship smoothly. Rollback = how you undo a bad
one. Don't mix them up — I did at first.


Step 6: Making Pods Reachable — Services & Ingress

Pods are ephemeral and get new IPs constantly, so you never talk to them directly. A
Service gives a stable address in front of a group of pods (selected by label):

apiVersion: v1
kind: Service
metadata:
  name: nginx-service
  namespace: nginx
spec:
  selector:
    app: nginx
  ports:
  - port: 80
    targetPort: 80
  type: ClusterIP
Enter fullscreen mode Exit fullscreen mode

To poke at it from your laptop during development, port-forward it:

kubectl port-forward service/nginx-service -n nginx 80:80 --address=0.0.0.0
Enter fullscreen mode Exit fullscreen mode

For real HTTP routing — exposing multiple apps under one entry point, routing by path like
/apache and /nginx — you graduate to an Ingress:

apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
  name: web-ingress
  namespace: apache
  annotations:
    nginx.ingress.kubernetes.io/rewrite-target: /$2
    nginx.ingress.kubernetes.io/use-regex: "true"
spec:
  ingressClassName: nginx
  rules:
  - host: practice.local
    http:
      paths:
      - path: /apache(/|$)(.*)
        pathType: ImplementationSpecific
        backend:
          service:
            name: apache-service
            port:
              number: 80
Enter fullscreen mode Exit fullscreen mode

Three things tripped me up with Ingress, so learn from my mistakes:

  1. An Ingress is namespaced — it can only route to Services in its own namespace.
  2. It does nothing until you install an ingress controller (like ingress-nginx).
  3. Path prefixes usually need a rewrite annotation, or your backend (which serves at /) will return 404 for /apache.

Step 7: Storage That Survives a Restart

Containers are throwaway — kill the pod and the data is gone. For anything that must
persist (think databases), Kubernetes splits storage into two pieces:

  • A PersistentVolume (PV) — the actual storage in the cluster. It's cluster-scoped, so it has no namespace.
  • A PersistentVolumeClaim (PVC) — a pod's request for storage. It's namespaced, and it binds to a PV that matches its size, access mode, and storage class.

Your pod then mounts the claim like any other volume:

volumeMounts:
- name: data
  mountPath: /var/lib/mysql
volumes:
- name: data
  persistentVolumeClaim:
    claimName: mysql-pvc
Enter fullscreen mode Exit fullscreen mode

Now your database keeps its data even when the pod is rescheduled.


Step 8: Configuration & Secrets

Hardcoding config into images is an anti-pattern. Kubernetes gives you two tools:

  • ConfigMap — non-sensitive key/value config, injected as env vars or mounted files.
  • Secret — sensitive values (passwords, API keys), referenced via secretKeyRef.
env:
- name: MYSQL_ROOT_PASSWORD
  valueFrom:
    secretKeyRef:
      name: mysql-secret
      key: mysql-root-password
Enter fullscreen mode Exit fullscreen mode

One important warning: a Secret is only base64-encoded, not encrypted. Never commit
real credentials to Git.


Step 9: Autoscaling — Let the Cluster React to Load

Here's the feature that makes Kubernetes feel like magic. A Horizontal Pod Autoscaler
(HPA)
watches a metric (usually CPU) and adds or removes pods automatically between a
min and max you define.

But HPA can't read CPU out of the box on a fresh kind cluster — you first need
metrics-server:

kubectl apply -f https://github.com/kubernetes-sigs/metrics-server/releases/latest/download/components.yaml
Enter fullscreen mode Exit fullscreen mode

On kind specifically, metrics-server often gets stuck because it can't verify the
kubelet's TLS cert. The local-only fix:

kubectl patch deployment metrics-server -n kube-system --type=json \
  -p='[{"op":"add","path":"/spec/template/spec/containers/0/args/-","value":"--kubelet-insecure-tls"}]'
Enter fullscreen mode Exit fullscreen mode

Confirm metrics are flowing:

kubectl top nodes
kubectl top pods -n nginx
Enter fullscreen mode Exit fullscreen mode

(Note: kubectl top nodes is cluster-wide, so no -n flag — that one caught me.)

Now define the HPA. It needs your Deployment to declare CPU requests, then it targets
a utilization percentage and scales between min and max replicas — and you can even tune
how quickly it reacts:

apiVersion: autoscaling/v2
kind: HorizontalPodAutoscaler
metadata:
  name: apache-hpa
  namespace: apache
spec:
  scaleTargetRef:
    apiVersion: apps/v1
    kind: Deployment
    name: apache-deployment
  minReplicas: 1
  maxReplicas: 5
  metrics:
  - type: Resource
    resource:
      name: cpu
      target:
        type: Utilization
        averageUtilization: 50
Enter fullscreen mode Exit fullscreen mode

The Other Direction: Vertical Scaling with VPA

HPA answers "how many pods?" But there's a second question hiding in plain sight: "how
big should each pod be?" That's the job of the Vertical Pod Autoscaler (VPA). Instead
of adding replicas, it right-sizes a pod's CPU and memory based on what it actually uses —
so you stop guessing at requests: values and stop paying for headroom you never touch.

VPA isn't bundled with the cluster like HPA, so you install it from the official autoscaler
repository:

git clone https://github.com/kubernetes/autoscaler.git
cd autoscaler/vertical-pod-autoscaler
./hack/vpa-up.sh
Enter fullscreen mode Exit fullscreen mode

Then point it at a Deployment and choose how bold it should be with updateMode:

apiVersion: autoscaling.k8s.io/v1
kind: VerticalPodAutoscaler
metadata:
  name: apache-vpa
  namespace: apache
spec:
  targetRef:
    apiVersion: "apps/v1"
    kind: Deployment
    name: apache-deployment
  updatePolicy:
    updateMode: "Auto"
Enter fullscreen mode Exit fullscreen mode

"Off" just hands you recommendations to eyeball (kubectl describe vpa apache-vpa -n
apache
), "Initial" applies them only to brand-new pods, and "Auto" actively evicts and
recreates pods to resize them on the fly.

One hard-earned warning: never aim HPA and VPA at the same metric on the same workload.
If both manage CPU, they fight — VPA resizing pods while HPA changes their count, each one
reacting to the other's moves. Pick one for CPU and you'll save yourself a very confusing
afternoon.

While we're on the topic of resources: every container should declare requests
(guaranteed) and limits (ceiling) for CPU and memory. And if you share a cluster
across teams, a ResourceQuota caps total consumption per namespace.


Step 10: The Debugging Commands You'll Use Every Day

When something won't start, these four commands answer 90% of "why?":

# Get a shell inside a running pod
kubectl exec -it nginx-pod -n nginx -- bash      # use -- sh if bash isn't installed

# See events and the real reason a pod is unhealthy
kubectl describe pod/nginx-pod -n nginx

# Read the logs (-f to follow, --previous for a crashed container)
kubectl logs nginx-pod -n nginx -f

# See everything in a namespace at once
kubectl get all -n nginx
Enter fullscreen mode Exit fullscreen mode

A small but real detail: the separator before the command is -- (two hyphens), and the
shell is bash or sh. Copying en-dashes from notes will silently break the command.


Wrapping Up

If you followed along, you went from a bare machine to a running cluster where you can
deploy apps, expose them over HTTP, give them persistent storage, secure their config, and
let them scale themselves under load. That's genuinely most of what you do with Kubernetes
day to day.

My advice: don't just read this — kind create cluster and run every command. Break a
Deployment, watch it self-heal. Crank the HPA and hammer it with load. The concepts only
click once you've watched the cluster react in real time.

If this helped, I'd love to hear what you build next. Happy shipping! 🚀


Found this useful? Follow for more hands-on DevOps and Kubernetes write-ups.

Top comments (0)