If you've just learned the basics of Kubernetes Pods, Deployments, ReplicaSets, and Services the best next step is to actually use them. Reading about self-healing and rolling updates is one thing; watching Kubernetes recreate a deleted Pod in real time is another.
In this guide, you'll deploy a simple Node.js app on a local Kubernetes cluster. We'll cover both Minikube and Kind (Kubernetes in Docker), so you can follow along whichever tool you prefer.
By the end, you'll have:
- A containerised Node.js app running in Kubernetes
- 3 replicas managed by a Deployment and ReplicaSet
- A Service exposing the app to your browser
- Hands-on experience with self-healing and scaling
Prerequisites
Before we start, make sure you have these installed:
- Docker required by both Minikube and Kind
- kubectl the Kubernetes CLI
- Either Minikube or Kind (installation covered below)
Part 1: Setting Up Your Local Cluster
You only need one of these. If you're not sure which to pick:
- Minikube: slightly friendlier for beginners, has a built-in way to open services in the browser
- Kind: lighter, faster, great if you already have Docker set up
Option A: Minikube
Install Minikube
macOS (Homebrew):
brew install minikube
Linux:
curl -LO https://storage.googleapis.com/minikube/releases/latest/minikube-linux-amd64
sudo install minikube-linux-amd64 /usr/local/bin/minikube
Windows (via winget):
winget install Kubernetes.minikube
Start your cluster:
minikube start
Verify it's running:
kubectl get nodes
# NAME STATUS ROLES AGE VERSION
# minikube Ready control-plane 10s v1.x.x
Option B: Kind (Kubernetes in Docker)
Install Kind
macOS (Homebrew):
brew install kind
Linux:
curl -Lo ./kind https://kind.sigs.k8s.io/dl/v0.22.0/kind-linux-amd64
chmod +x ./kind
sudo mv ./kind /usr/local/bin/kind
Windows (via Chocolatey):
choco install kind
Create your cluster:
kind create cluster --name hello-cluster
Verify it's running:
kubectl get nodes
# NAME STATUS ROLES AGE VERSION
# hello-cluster-control-plane Ready control-plane 10s v1.x.x
Part 2: Build the Node.js App
Create a new folder for the project:
mkdir k8s-hello && cd k8s-hello
Create app.js:
nano/vim app.js
const http = require('http');
const os = require('os');
const server = http.createServer((req, res) => {
res.end(`Hello from Pod: ${os.hostname()}\n`);
});
server.listen(3000, () => console.log('Running on port 3000'));
Why
os.hostname()? In Kubernetes, each Pod gets a unique hostname. When the Service load-balances traffic across multiple Pods, you'll see different hostnames on each refresh proving which Pod served you.
Create Dockerfile:
FROM node:18-alpine
WORKDIR /app
COPY app.js .
CMD ["node", "app.js"]
Part 3: Build and Load the Docker Image
This step differs between Minikube and Kind pay attention here.
Minikube
Minikube runs its own Docker daemon inside a VM. Point your local Docker CLI at it so your build lands inside Minikube directly:
eval $(minikube docker-env)
docker build -t hello-app:v1 .
From this point, Minikube can see the image locally without needing Docker Hub.
Kind
Kind doesn't share a Docker daemon. You build the image normally, then explicitly load it into the cluster:
docker build -t hello-app:v1 .
kind load docker-image hello-app:v1 --name hello-cluster
Skipping
kind loadis the most common beginner mistake with Kind. Without it, your Pods will get stuck inImagePullBackOffbecause Kind can't find the image.
Part 4: Write the Kubernetes YAML
Create deployment.yaml in your project folder:
apiVersion: apps/v1
kind: Deployment
metadata:
name: hello-deployment
spec:
replicas: 3
selector:
matchLabels:
app: hello
template:
metadata:
labels:
app: hello
spec:
containers:
- name: hello
image: hello-app:v1
imagePullPolicy: Never # use local image, don't pull from Docker Hub
ports:
- containerPort: 3000
---
apiVersion: v1
kind: Service
metadata:
name: hello-service
spec:
type: NodePort
selector:
app: hello # matches the Pod label above this is how Services find Pods
ports:
- port: 80
targetPort: 3000
nodePort: 30080
What's happening here:
- The Deployment tells Kubernetes to keep 3 replicas of our Pod running at all times
- It automatically creates a ReplicaSet to enforce that replica count
- The Service uses the
app: hellolabel selector to find all matching Pods and route traffic to them -
imagePullPolicy: Nevertells Kubernetes to use the locally available image instead of going to Docker Hub
Part 5: Deploy It
kubectl apply -f deployment.yaml
You should see:
deployment.apps/hello-deployment created
service/hello-service created
Check your Pods are coming up:
kubectl get pods
Wait until all three show Running:
NAME READY STATUS RESTARTS AGE
hello-deployment-57c4d87bf-abc12 1/1 Running 0 15s
hello-deployment-57c4d87bf-def34 1/1 Running 0 15s
hello-deployment-57c4d87bf-ghi56 1/1 Running 0 15s
Check your ReplicaSet and Service too:
kubectl get replicaset
kubectl get service hello-service
Part 6: Open the App in Your Browser
Minikube
minikube service hello-service
Minikube opens the URL in your browser automatically.
Kind
Kind doesn't expose NodePort services directly, so use port-forwarding:
kubectl port-forward service/hello-service 8080:80
Then open http://localhost:8080.
Hit refresh a few times. You'll see the Pod hostname change the Service is load-balancing across your 3 Pods.
Hello from Pod: hello-deployment-57c4d87bf-abc12
Hello from Pod: hello-deployment-57c4d87bf-ghi56
Hello from Pod: hello-deployment-57c4d87bf-def34
Part 7: Experiments (The Real Learning)
Now that everything is running, try these one by one. Each one demonstrates a core Kubernetes behaviour.
1. Self-healing...delete a Pod manually
# grab any pod name
kubectl get pods
# delete it
kubectl delete pod hello-deployment-57c4d87bf-abc12
# watch what happens
kubectl get pods -w
Kubernetes detects the replica count dropped to 2 and immediately creates a new Pod. This is the ReplicaSet controller doing its job.
2. Scaling up
kubectl scale deployment hello-deployment --replicas=5
kubectl get pods
Two new Pods appear almost instantly.
3. Scaling down
kubectl scale deployment hello-deployment --replicas=1
kubectl get pods
Four Pods terminate gracefully, one remains.
4. Rolling update with zero downtime
Edit app.js to change the response message:
res.end(`Hello from Pod v2: ${os.hostname()}\n`);
Build a new image:
# Minikube
eval $(minikube docker-env)
docker build -t hello-app:v2 .
# Kind
docker build -t hello-app:v2 .
kind load docker-image hello-app:v2 --name hello-cluster
Update the Deployment:
kubectl set image deployment/hello-deployment hello=hello-app:v2
Watch the rolling update:
kubectl rollout status deployment/hello-deployment
Kubernetes replaces Pods one at a time, keeping the app available throughout.
5. Inspect a Pod
kubectl describe pod <pod-name>
This shows you the Pod's IP, which Node it's on, its labels, and a full event log — useful for debugging.
6. Roll back
If something goes wrong with an update:
kubectl rollout undo deployment/hello-deployment
Kubernetes switches back to the previous ReplicaSet.
Part 8: Clean Up
kubectl delete -f deployment.yaml
Minikube:
minikube stop
Kind:
kind delete cluster --name hello-cluster
Note for Kind users: Kind clusters don't survive a machine restart. If you reboot and come back to this project, run
kind create cluster --name hello-clusterandkind load docker-image hello-app:v1 --name hello-clusterbefore applying your YAML again.
What You Just Built
Here's what was happening under the hood the whole time:
Your Browser
↓
[Service] ← watched for Pods with label app: hello
↓
[ReplicaSet] ← enforced 3 running replicas at all times
↓ ↓ ↓
[Pod] [Pod] [Pod] ← each ran your Node.js container
Every concept from the Kubernetes basics maps to something you just did:
| Concept | What you observed |
|---|---|
| Pod | The unit running your container, with a unique hostname |
| ReplicaSet | Recreated a Pod immediately after you deleted one |
| Deployment | Managed the rolling update and rollback |
| Service | Load-balanced traffic across all 3 Pods using label selectors |
What's Next?
Now that you have the fundamentals working, here are good next topics to explore:
- Namespaces isolate workloads for different teams or environments
- ConfigMaps & Secrets externalise config and credentials from your container
- Ingress a cleaner alternative to NodePort for routing external traffic
- Persistent Volumes attach storage that survives Pod restarts
- Liveness & Readiness Probes teach Kubernetes when your Pod is actually healthy
If you ran into issues or have questions, drop them in the comments. The most common problems are forgetting kind load docker-image (Kind) or not running eval $(minikube docker-env) before building (Minikube).

Top comments (0)