Author: Ojas Jawale
Table of Contents
- Project Overview
- Installation of Minikube Cluster
- Step 1 : Namespace Creation
- Step 2 : Create Deployment
- Step 3 : Create Service
- Step 4 : Persistent Volumes and Claims
- Step 5 : Ingress Controller
- Step 6 : Network Policies and CNI
- Step 7 : Job and CronJob
- Step 8 : Horizontal Pod Auto-scaling
- Step 9 : Role Base Access Control (RBAC)
- Step 10 : Testing application on browser
- Project Overview
In this Kubernetes project, the goal is to deploy and manage an application using the ojasjawale/notes-app:latest
image within a dedicated namespace, notes-app
on minikube cluster. This deployment will incorporate a wide range of Kubernetes concepts, each contributing to a robust and secure application environment.
This project is an extension of previous project deployed -> Django Notes App
Where, deployed an Django Notes application using Docker and pushed image on DockerHub. Now as an add-on of kubernetes, using same image deployed on DockerHub.
For detailed steps of installation of minikube cluster visit -> Minikube Installation
In Next steps we will directly going to setup k8s cluster for project.
- Create a namespace to organize all Kubernetes resources for this project.
- Create
namespace.yml
file, - This YAML file creates a namespace named
notes-app
to isolate resources and manage them more easily.
apiVersion: v1
kind: Namespace
metadata:
name: notes-app
- Apply file
kubectl apply -f namespace.yml
data:image/s3,"s3://crabby-images/ae349/ae3497dee517f93d59b191bf132cf96ad32eb9ad" alt=""
data:image/s3,"s3://crabby-images/377e6/377e61f3d543aa648eedfa727bc300ed909734e9" alt=""
- Step 2 : Create Deployment
- Deploy the
notes-app
using a Deployment resource to manage replicas and updates. - Create
deployment.yml
file, - This Deployment ensures three replicas of the
notes-app
container are running and manages updates seamlessly.
apiVersion: apps/v1
kind: Deployment
metadata:
name: notes-app-deployment
namespace: notes-app
spec:
replicas: 3
selector:
matchLabels:
app: notes-app
template:
metadata:
labels:
app: notes-app
spec:
containers:
- name: notes-app
image: ojasjawale/notes-app:latest
ports:
- containerPort: 8000
- Apply file
kubectl apply -f deployment.yml
data:image/s3,"s3://crabby-images/b6301/b63017b30a697535f1102bc43670ae429b017257" alt=""
data:image/s3,"s3://crabby-images/a0b47/a0b47200bebb57e879882c9960df28a03b3f4f17" alt=""
data:image/s3,"s3://crabby-images/28df0/28df0b0e641890e67c56be4c4a730374aa19a66b" alt=""
- Step 3 : Create Service
- Expose the
notes-app
deployment using a Service. - Create
service.yml
file,
apiVersion: v1
kind: Service
metadata:
name: notes-app-service
namespace: notes-app
spec:
selector:
app: notes-app
ports:
- protocol: TCP
port: 80
targetPort: 8000
- Apply file
kubectl apply -f service.yml
data:image/s3,"s3://crabby-images/205c2/205c21b1a068badb9deea45394d5879fbc8ac7ac" alt=""
- For testing, port forwarding of traffic.
kubectl port-forward service/notes-app-service 8000:80 --address=0.0.0.0 -n notes-app
- Copy Public IP of instance with port 8000 and access application on browser.
data:image/s3,"s3://crabby-images/38f5e/38f5e341e1635f432afe47c0bb88f79cbef824b8" alt=""
- Set up Persistent Volumes (PV) and Persistent Volume Claims (PVC) for stateful applications.
- Create
pv.yml
file,
apiVersion: v1
kind: PersistentVolume
metadata:
name: notes-app-pv
namespace: notes-app
spec:
capacity:
storage: 5Gi
accessModes:
- ReadWriteOnce
hostPath:
path: /mnt/data
- Apply file
kubectl apply -f pv.yml
data:image/s3,"s3://crabby-images/88da5/88da56ff88576e605db5dea2c31d7ebb57ee1659" alt=""
- Till now persistentVolume is not claimed.
- PV defines the storage resource, while PVC requests storage for a pod.
- Create
pvc.yml
file
Claiming PV storage for POD’s.
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: notes-app-pvc
namespace: notes-app
spec:
accessModes:
- ReadWriteOnce
resources:
requests:
storage: 5Gi
- Apply file
kubectl apply -f pvc.yml
- Attach storage to deployment
apiVersion: apps/v1
kind: Deployment
metadata:
name: notes-app-deployment
namespace: notes-app
spec:
replicas: 3
selector:
matchLabels:
app: notes-app
template:
metadata:
labels:
app: notes-app
spec:
containers:
- name: notes-app
image: ojasjawale/notes-app:latest
ports:
- containerPort: 8000
volumeMounts:
- name: notes-app-storage
mountPath: /data # Mount path inside the container
volumes:
- name: notes-app-storage
persistentVolumeClaim:
claimName: notes-app-pvc # Reference to the PVC
To setup Ingress in minikube, need to enable addon called ingress
minikube addon enable ingress
data:image/s3,"s3://crabby-images/b07bc/b07bcc079b4410495fdef7b8d5495f37137b1847" alt=""
- Set up an Ingress for path-based routing to the application.
- Create
ingress.yml
file, - The Ingress routes traffic based on the URL path to the
notes-app-service
, which listens on port 80 and forwards it to port 8000 in the pods.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: notes-app-ingress
namespace: notes-app
spec:
rules:
- host: notes-app
http:
paths:
- path: /
pathType: Prefix
backend:
service:
name: notes-app-service
port:
number: 80
- path: /app
pathType: Prefix
backend:
service:
name: notes-app-service
port:
number: 80
data:image/s3,"s3://crabby-images/fed8d/fed8d22e393f03586c8da179d939b79423c7df53" alt=""
- Apply file
kubectl apply -f ingress.yml
- Add an domain entry in
/etc/hosts
file
sudo nano /etc/hosts
data:image/s3,"s3://crabby-images/5779a/5779a821a48f427a52c407374b6e0a95e99b3aa7" alt=""
- Now, try to send query on URL
curl notes-app
data:image/s3,"s3://crabby-images/2c858/2c858680d9ade56e6a9fd2a586e42d096a36c4d3" alt=""
- This is an we called it as path based routing using Ingress controller.
Step 6 : Network Policies and CNI
- Define Network Policies to control traffic and configure CNI for network management.
- Create
networkPolicy.yml
file, - Network Policies control the communication between pods based on labels. CNI (Container Network Interface) is typically configured at the cluster level.
apiVersion: networking.k8s.io/v1
kind: NetworkPolicy
metadata:
name: notes-app-network-policy
namespace: notes-app
spec:
podSelector:
matchLabels:
app: notes-app
policyTypes:
- Ingress
- Egress
ingress:
- from:
- podSelector:
matchLabels:
app: notes-app
egress:
- to:
- podSelector:
matchLabels:
app: notes-app
- Apply file
kubectl apply -f networkPolicy.yml
data:image/s3,"s3://crabby-images/617ff/617ffb053b80d25ef2426a28383b5efad99c5da9" alt=""
- Define a Job for one-time tasks and a CronJob for scheduled tasks.
- Create
job.yml
file
apiVersion: batch/v1
kind: Job
metadata:
name: notes-app-job
namespace: notes-app
spec:
template:
spec:
containers:
- name: notes-app-job
image: ojasjawale/notes-app:latest
ports:
- containerPort: 8000
restartPolicy: OnFailure
data:image/s3,"s3://crabby-images/73c5b/73c5bd838ded6a816c39e4c8a59f34161bdd3413" alt=""
data:image/s3,"s3://crabby-images/c5b7f/c5b7f17043e38edae7ce633e92878538eb1860f7" alt=""
- Create
cron.yml
file
apiVersion: batch/v1
kind: CronJob
metadata:
name: notes-app-cronjob
namespace: notes-app
spec:
schedule: "0 1 * * *" # Daily at 1 AM
jobTemplate:
spec:
template:
spec:
containers:
- name: notes-app-cronjob
image: ojasjawale/notes-app:latest
ports:
- containerPort: 8000
restartPolicy: OnFailure
data:image/s3,"s3://crabby-images/d809d/d809d376a86dc9e7fd6cdf87fa537a937b81a41a" alt=""
For HPA setup in minikube, Need to enable addon called metrics-server
minikube addons enable metrics-server
data:image/s3,"s3://crabby-images/ba4f6/ba4f679297086de316fb0a08aab3389ffe65d131" alt=""
- Modify deployment for resource limiting
apiVersion: apps/v1
kind: Deployment
metadata:
name: notes-app-deployment
namespace: notes-app
spec:
replicas: 1
selector:
matchLabels:
app: notes-app
template:
metadata:
labels:
app: notes-app
spec:
containers:
- name: notes-app
image: ojasjawale/notes-app:latest
ports:
- containerPort: 8000
resources:
requests:
cpu: "100m" # Minimum CPU required
limits:
cpu: "200m" # Maximum CPU allowed
- Implement HPA to scale pods based on CPU utilization.
- Create
hpa.yml
file
apiVersion: autoscaling/v2
kind: HorizontalPodAutoscaler
metadata:
name: notes-app-hpa
namespace: notes-app
spec:
scaleTargetRef:
apiVersion: apps/v1
kind: Deployment
name: notes-app-deployment
minReplicas: 1
maxReplicas: 5
metrics:
- type: Resource
resource:
name: cpu
target:
type: Utilization
averageUtilization: 15
data:image/s3,"s3://crabby-images/6f576/6f5762616666cb22c50f7f769e7eb66967a34a2e" alt=""
- Apply file
kubectl apply -f hpa.yml
data:image/s3,"s3://crabby-images/779bb/779bb95f9985e6691f4729d01c159c3bbe952648" alt=""
- Currently, there is only one POD is running,
data:image/s3,"s3://crabby-images/43f63/43f633274c155f2db2d80346f208c7b2a216cd65" alt=""
- Create two screens and in one screen try port-forwarding
kubectl port-forward service/notes-app-service 8000:80 --address=0.0.0.0 -n notes-app
data:image/s3,"s3://crabby-images/53e92/53e92238e51aa4227c9f7e9edc893702d06b1169" alt=""
- Second screen watch for real-time CPU utilization and auto-scaling of POD’s
watch kubectl get hpa -n notes-app
- POD’s are scaling up automatically,
data:image/s3,"s3://crabby-images/5e8bd/5e8bd34d649333efe97107ffe5b224a483af715d" alt=""
data:image/s3,"s3://crabby-images/eca43/eca438daebb8f5190b05cb13a61ff3681657e6d3" alt=""
data:image/s3,"s3://crabby-images/dd5fa/dd5fa99eaacbc642e5db78538ca61527c8e2b13d" alt=""
data:image/s3,"s3://crabby-images/7f71e/7f71ee6ae33c97fe0860df610f2a4bd467b48cb4" alt=""
- Create service account
kubectl create serviceaccount k8s-user
data:image/s3,"s3://crabby-images/ec109/ec109168ab82263efdffae6d35dad5c44b78ea8e" alt=""
- Implement RBAC for fine-grained access control.
- Create
role.yml
file
apiVersion: rbac.authorization.k8s.io/v1
kind: Role
metadata:
name: notes-app-role
namespace: notes-app
rules:
- apiGroups: [""]
resources: ["pods"]
verbs: ["get", "list", "watch"]
data:image/s3,"s3://crabby-images/8acc3/8acc35a8f74b3e3267434a49007e57bfb4b3b60e" alt=""
- Create
roleBinding.yml
file
apiVersion: rbac.authorization.k8s.io/v1
kind: RoleBinding
metadata:
name: notes-app-rolebinding
namespace: notes-app
subjects:
- kind: User
name: k8s-user # Replace with your Kubernetes user
apiGroup: rbac.authorization.k8s.io
roleRef:
kind: Role
name: notes-app-role
apiGroup: rbac.authorization.k8s.io
data:image/s3,"s3://crabby-images/830e0/830e015b95515fcfd1a4fac5b413547cc6a79041" alt=""
- Role defines permissions, and RoleBinding assigns those permissions to a user or set of users.
- Testing of RBAC configurations,
- Run commands using –as=user flag (Where
user
is the name of the user you wish to impersonate.) - As
k8s-user
has get permissions,
data:image/s3,"s3://crabby-images/a0ceb/a0ceb67cc1e5fd925b85021476bfa8ae10bc4b0e" alt=""
- As k8s-user don’t have create resource permission, hence action is forbidden.
data:image/s3,"s3://crabby-images/02262/022625bf0211f9adfd783968badf9d4cf5d9ecd8" alt=""
- Forward port on containerPort from local port and try to access application
kubectl port-forward service/notes-app-service 8000:80 --address=0.0.0.0 -n notes-app
data:image/s3,"s3://crabby-images/7c1ba/7c1ba1b59efe1c144576332f9a94e8d74de5ae62" alt=""
data:image/s3,"s3://crabby-images/205af/205af289b97ffabd850dd2d98cfdd00b9c91998c" alt=""
data:image/s3,"s3://crabby-images/91e8a/91e8a1ebada31eed638bd1b3566e9b2c013583b0" alt=""
Connect With Me
Thank you for reading. I hope you were able to understand and learn something new from my blog.
Happy Learning!