Ansible-Driven Container Deployment: Building, Pushing, and Deploying with Docker, Kubernetes, and Helm
Embark on an Epic DevOps Voyage with Ansible-Driven Container Deployment
Are you ready to set sail on a DevOps adventure like no other? In this journey, you’ll harness the power of Ansible to orchestrate the deployment of your containerized applications with Docker, Kubernetes (K8s), and Helm. So, fasten your seatbelts and let’s get started!
Prerequisites:
Before we hoist the anchor, make sure you’ve got these tools ready to roll:
Ansible: Our trusty captain, Ansible, will be steering the ship. Make sure you have it installed and ready for action.
Docker: Ahoy there, containerization! We’ll be using Docker to package our applications. If you don’t have it, you can easily set sail by installing it from the official website.
Git: You’ll need Git to navigate the high seas of version control. Install it on your system if you haven’t already.
Installation and Usage: We’ll explore how to create an Ansible script to automate the installation of Minikube, Docker, and Helm. With this Ansible playbook, we’ll streamline the process of building Docker images for your React and Node.js applications, pushing them to Docker Hub. Plus, we’ll create Helm charts in the helm-v1 directory, making it easy to deploy and manage your applications with Helm
So this is Directory Structure
Create playbook.yml
- name: Install and start Minikube
hosts: localhost
become: true
gather_facts: false
tasks:
- name: Install dependencies
apt:
name:
- curl
- virtualbox
- conntrack
- libvirt-clients
- libvirt-daemon-system
- name: Download and install Minikube
block:
- name: Download Minikube
get_url:
url: "https://storage.googleapis.com/minikube/releases/latest/minikube-linux-amd64"
dest: "/usr/local/bin/minikube"
mode: 0755
validate_certs: no
- name: Add Minikube to the PATH
lineinfile:
dest: "$HOME/.bashrc"
line: 'export PATH=/usr/local/bin:$PATH'
insertafter: EOF
become: false
- name: Start Minikube
shell: minikube start
args:
executable: /bin/bash
creates: "$HOME/.kube/config"
handlers:
- name: Restart shell
shell: exec $SHELL
- name: Install and configure Kubernetes with Minikube
hosts: localhost
connection: local
become: true
tasks:
- name: Install Docker
snap:
name: docker
- name: Install Docker.io package
apt:
name: docker.io
state: present
- name: Start Docker service
service:
name: docker
state: started
- name: Download kubectl
shell: |
curl -LO "https://dl.k8s.io/release/$(curl -L -s https://dl.k8s.io/release/stable.txt)/bin/linux/amd64/kubectl"
- name: Verify kubectl integrity
shell: |
curl -LO "https://dl.k8s.io/$(curl -L -s https://dl.k8s.io/release/stable.txt)/bin/linux/amd64/kubectl.sha256"
echo "$(cat kubectl.sha256) kubectl" | sha256sum --check
- name: Install kubectl
shell: |
sudo install -o root -g root -m 0755 kubectl /usr/local/bin/kubectl
- name: Login to Docker Hub & Build Docker Images
hosts: localhost
gather_facts: false
vars:
inventory_file: "inventory.yml"
tasks:
- name: Load inventory file
include_vars:
file: "{{ inventory_file }}"
- name: Change Directory to Simple-crud-react-nodejs
shell: cd /home/saurabhadhau/Desktop/P-2/simple-crud-react-nodejs/
- name: Log in to Docker Hub
docker_login:
username: nacromancer858 #you should use your docker hub username
password: your-docker_hub_password
registry_url: https://index.docker.io/v2/
- name: Print Docker Hub login status
debug:
msg: "Docker Hub login successful!"
- name: Change Directory to Simple-crud-react-nodejs/frontend
shell: cd /home/saurabhadhau/Desktop/P-2/simple-crud-react-nodejs/frontend/
- name: Build Frontend Docker Image
shell: docker build -t frontend-image-777 .
args:
chdir: /home/saurabhadhau/Desktop/P-2/simple-crud-react-nodejs/frontend/
- name: Tag Frontend Docker Image
shell: docker tag frontend-image-777 nacromancer858/frontend-image-777:"{{ image_tags.frontend_tag }}"
args:
chdir: /home/saurabhadhau/Desktop/P-2/simple-crud-react-nodejs/frontend/
- name: Push Frontend Docker Image
shell: docker push nacromancer858/frontend-image-777:"{{ image_tags.frontend_tag }}"
args:
chdir: /home/saurabhadhau/Desktop/P-2/simple-crud-react-nodejs/frontend/
- name: Change Directory to Simple-crud-react-nodejs/backend
shell: cd /home/saurabhadhau/Desktop/P-2/simple-crud-react-nodejs/backend/
- name: Build Backend Docker Image
shell: |
docker build -t nacromancer858/backend-image -f Dockerfile .
docker tag nacromancer858/backend-image nacromancer858/backend-image:"{{ image_tags.backend_tag }}"
docker push nacromancer858/backend-image:"{{ image_tags.backend_tag }}"
args:
chdir: /home/saurabhadhau/Desktop/P-2/simple-crud-react-nodejs/backend/
- name: Install and configure Kubernetes with Minikube
hosts: localhost
connection: local
tasks:
- name: Start Minikube
command: minikube start
- name: Check Minikube status
command: minikube status
- name: Install helm
shell: |
curl -fsSL -o get_helm.sh https://raw.githubusercontent.com/helm/helm/main/scripts/get-helm-3
chmod 700 get_helm.sh
./get_helm.sh
- name: Pulling the Images and Install Helm
hosts: localhost
connection: local
tasks:
- name: Pull frontend Docker image
docker_image:
name: nacromancer858/frontend-image-777
tag: "{{ image_tags.frontend_tag }}"
source: pull
- name: Pull backend Docker image
docker_image:
name: nacromancer858/backend-image
tag: "{{ image_tags.backend_tag }}"
source: pull
- name: Change Directory to Helm-charts
shell: cd /home/saurabhadhau/Desktop/P-2/helm-v1/
- name: Update frontend values file
hosts: localhost
gather_facts: false
tasks:
- name: Read inventory file
include_vars:
file: inventory.yml
name: inventory_data
- name: Update frontend tag value
lineinfile:
path: helm-v1/values.frontend.yml
regexp: '^frontend:\n[[:space:]]*tag:'
line: 'frontend:\n tag: "{{ inventory_data.image_tags.frontend_tag }}"'
- name: Update Backend tag value
lineinfile:
path: helm-v1/values.backend.yml
regexp: '^backend:\n[[:space:]]*tag:'
line: 'backend:\n tag: "{{ inventory_data.image_tags.backend_tag }}"'
- name: Build Helm-Chart for Frontend
shell: helm install frontend ./frontend-chart -f values.frontend.yml
args:
chdir: /home/saurabhadhau/Desktop/P-2/helm-v1/
- name: Build Helm-Chart for Backend
shell: helm install backend ./backend-chart -f values.backend.yml
args:
chdir: /home/saurabhadhau/Desktop/P-2/helm-v1/
Create helm-v1 directory and as per the directory structure create the other files as well!
frontend-chart/templates/frontend-deployment.yaml:
apiVersion: apps/v1
kind: Deployment
metadata:
name: {{ include "frontend-chart.fullname" . }}
spec:
replicas: {{ .Values.frontend.replicaCount }}
selector:
matchLabels:
app: frontend
template:
metadata:
labels:
app: frontend
spec:
containers:
- name: frontend
image: {{ .Values.frontend.image.repository }}:{{ .Values.frontend.image.tag }}
ports:
- containerPort: {{ .Values.frontend.service.port }}
resources:
limits:
cpu: {{ .Values.frontend.resources.limits.cpu }}
memory: {{ .Values.frontend.resources.limits.memory }}
requests:
cpu: {{ .Values.frontend.resources.requests.cpu }}
memory: {{ .Values.frontend.resources.requests.memory }}
frontend-chart/templates/frontend-service.yaml:
{{- if eq .Values.frontend.service.type "NodePort" }}
apiVersion: v1
kind: Service
metadata:
name: {{ include "frontend-chart.fullname" . }}
spec:
selector:
app: frontend
type: {{ .Values.frontend.service.type }}
ports:
- port: {{ .Values.frontend.service.port }}
targetPort: {{ .Values.frontend.service.targetPort }}
nodePort: {{ .Values.frontend.service.nodePort }}
{{- else }}
apiVersion: v1
kind: Service
metadata:
name: {{ include "frontend-chart.fullname" . }}
spec:
selector:
app: frontend
type: ClusterIP
ports:
- port: {{ .Values.frontend.service.port }}
targetPort: {{ .Values.frontend.service.targetPort }}
{{- end }}
backend-chart/templates/backend-deployment.yaml:
apiVersion: apps/v1
kind: Deployment
metadata:
name: {{ include "backend.fullname" . }}
spec:
replicas: {{ .Values.backend.replicas }}
selector:
matchLabels:
app: backend
template:
metadata:
labels:
app: backend
spec:
containers:
- name: backend
image: {{ .Values.backend.image.repository }}:{{ .Values.backend.image.tag }}
ports:
- containerPort: {{ .Values.backend.port }}
volumeMounts:
- name: backend-storage
mountPath: /data
resources:
limits:
cpu: {{ .Values.backend.resources.limits.cpu }}
memory: {{ .Values.backend.resources.limits.memory }}
requests:
cpu: {{ .Values.backend.resources.requests.cpu }}
memory: {{ .Values.backend.resources.requests.memory }}
volumes:
- name: backend-storage
persistentVolumeClaim:
claimName: backend-pvc
backend-chart/templates/backend-service.yaml:
apiVersion: v1
kind: Service
metadata:
name: {{ include "backend.fullname" . }}
spec:
selector:
app: backend
{{- if eq .Values.backend.serviceType "NodePort" }}
type: NodePort
{{- else }}
type: ClusterIP
{{- end }}
ports:
- port: {{ .Values.backend.port }}
targetPort: {{ .Values.backend.port }}
{{- if eq .Values.backend.serviceType "NodePort" }}
nodePort: {{ .Values.backend.nodePort }}
{{- end }}
backend-chart/templates/backend-pvc.yaml:
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: backend-pvc
spec:
accessModes:
- ReadWriteOnce
resources:
requests:
storage: "1Gi"
You can dive into the complete code in all its glory right here at our repository:
https://github.com/Saurabh-DevOpsVoyager77/ansible-driven-deployment-Docker_K8s_Helm
Now, let’s add some magic to your inventory.yaml file by setting those crucial tags:
image_tags:
frontend_tag: 1.0
backend_tag: 1.0
With the tags in place, you’re all set for the grand finale! Just execute this command:
sudo ansible-playbook playbook.yml -i inventory.yml
Celebrate Your Success 🎉
Congratulations! You’ve just embarked on a journey towards efficient container deployment with Ansible, Docker, Kubernetes, and Helm. This powerful combination allows you to take control of your applications and streamline your deployment process.
Now that you’ve completed the setup and executed your Ansible playbook, your React Node.js application is ready to shine ✨. Keep experimenting, exploring, and pushing your DevOps skills to new heights 🚀. There’s no limit to what you can achieve with this versatile toolkit.