Project 02: From Code to Deployment Using Jenkins Declarative CICD Pipeline

Project 02: From Code to Deployment Using Jenkins Declarative CICD Pipeline

Continuous Integration and Continuous Delivery (CICD) are crucial aspects of software development that help us build, test, and deploy software quickly and efficiently.

Project Overview

Our project is a microservice-based web application that consists of five different services: Redis, Postgres DB, Vote App, Worker App, and Result App. The purpose of the application is to allow users to vote for their preferred option and display the results of the poll in real time.

To automate the deployment process of this application, we have set up a CI/CD pipeline using Jenkins. The pipeline consists of several stages, including code checkout, building Docker images, deploying services to different servers using Docker Compose, updating the Docker registry, sending notification emails to the concerned person, and performing cleanup tasks on the hosting server.

By setting up this pipeline, we aim to streamline the deployment process of our microservice application and ensure that each new change to the codebase is automatically deployed to the production/ testing/ development environment. This approach will save us time and effort while ensuring the reliability and scalability of our application.

Pre-requisites

  1. A GitHub account to store the source code.

  2. Three servers/machines :

    a) Jenkins Master Node

    b) Jenkins Agent Node-1 [For development/testing environment]

    c) Jenkins Agent Node-2 [For production environment]

  3. Jenkins, Docker and Docker Compose are installed.

  4. Docker registry to store the Docker-versioned images.

  5. Knowledge of Groovy syntax to create the Jenkins pipelines.

Let's get started--->

Step 1: Get the source code of this Microservice App

Fork this repo:

https://github.com/Chaitannyaa/Jenkins-Project.git

Step 2: Make three servers ready for deployment purposes

Create ec2 instances, one as the master node and the other as Agent nodes for deployment purposes.

Learn more >>

Instances can be t2-micro, 1 CPU, 1 GiB Memory, and ports 8080, 80, 22, 8501 are open to allow incoming traffic

Step 3: Setup these servers to use Jenkins

  1. Jenkins Master Node must have Jenkins installed.

    Learn more >>

  2. Jenkins Agent nodes must have Java, docker, docker-compose, and mail utility installed and configured.

    Learn more >> for Java and docker installation

    sudo apt install docker-compose -y for docker-compose installation

    Learn more >> for mail utility installation and configuration

  3. Access your Jenkins on http://Localhost[master-node]:8080

    Learn more >>

  4. Add agent nodes to the Jenkins Master server/node

    Learn more >>

Step 4: Create a Jenkins Pipeline project

Step 5: Set the GitHub webhook to trigger the project pipeline

  1. Go to the GitHub repository that you forked in Step 1 -->

  2. If you find a tick mark then you are done!

Step 6: Create a Jenkinsfile for our pipeline to run

We have three branches for this project

pipeline {
    agent {
        label 'dev'
    }
    environment {
        DOCKER_IMAGE_TAG = "${BUILD_NUMBER}"
    }
    stages {
        stage ("code"){
            steps {
                git url: 'https://github.com/Chaitannyaa/Jenkins-   Project.git', branch: 'development'
            }
        }
        stage ("Build docker images"){
            steps {
                sh 'docker build -t chaitannyaa/multi_container_app:worker-D-${DOCKER_IMAGE_TAG} ./worker'
                sh 'docker build -t chaitannyaa/multi_container_app:result-D-${DOCKER_IMAGE_TAG} ./result'
                sh 'docker build -t chaitannyaa/multi_container_app:vote-D-${DOCKER_IMAGE_TAG} ./vote'
            }
        }
        stage("Deploy services"){
            steps {
                 sh "docker-compose down && docker-compose up -d"
            }
        }        
        stage ("Update docker images"){
            steps {
                echo 'Logging into docker and pushing an image on dockerhub'
                withCredentials([usernamePassword(credentialsId: 'DockerHub', passwordVariable: 'password', usernameVariable: 'user')]) {
                    sh "docker login -u ${env.user} -p ${env.password}"
                    sh "docker push chaitannyaa/multi_container_app:worker-D-${DOCKER_IMAGE_TAG}"
                    sh "docker push chaitannyaa/multi_container_app:result-D-${DOCKER_IMAGE_TAG}"
                    sh "docker push chaitannyaa/multi_container_app:vote-D-${DOCKER_IMAGE_TAG}"
                }
            }
        }
        stage('Notify Management team') {
            steps {
                // Send a notification email to the team
                sh 'echo "The Updated microservices has been deployed on the development environment." | mail -s "Microservices are deployed successfully!" -a "From: DevOps-Chaitannyaa Gaikwad" chaitannyaagaikwad@gmail.com'
            }
        }
        stage("Clean-up"){
            steps {
                sh '''
                set +e
                docker rmi -f $(docker images -a -q)
                if [ $? -eq 0 ]; then
                    echo "All Docker images have been removed successfully."
                else
                    echo "Error: Could not remove all Docker images."
                fi
                set -e
                '''
            }
        }        
    }
}

This Jenkinsfile automates the build, deployment, and cleanup process for a multi-container application.

The pipeline has several stages that are executed sequentially:

  1. "code" stage: This stage checks out the code from the GitHub repository using Git.

  2. "Build docker images" stage: This stage builds Docker images for the different components of the multi-container application, such as worker, result, and vote.

  3. "Deploy services" stage: This stage deploys the Docker containers for the application using docker-compose.

  4. "Update docker images" stage: This stage updates the Docker images on DockerHub by pushing the newly built images.

  5. "Notify Management team" stage: This stage sends a notification email to the management team that the microservices have been deployed successfully.

  6. "Clean-up" stage: This stage removes all the Docker images that were built during the pipeline to clean up the system.

The pipeline is executed on an agent labeled "dev" and sets an environment variable DOCKER_IMAGE_TAG using the build number. The pipeline uses several shell scripts to perform the Docker-related tasks and sends notification emails using the 'mail' command. The pipeline also uses credentials to log in to DockerHub and push images.

Similarly, we can create a "Jenkinsfile" for other branches too.

Step 7: Run the Pipeline manually to check for any errors

The build Succeeded and we can access the Application on development-server

Step 8: Check whether the "Update docker image" and "Notify Management team" stages worked correctly or not.

Step 9: Make code changes to the production branch and commit to autostart build #2

The build Succeeded and we can access the Application on production-server

Step 10: Check whether the "Update docker image" and "Notify Management team" stages worked correctly or not.

Done!

Summary:

In this blog, we explored the significance of implementing Continuous Integration/Continuous Delivery (CI/CD) pipelines in software development. Demonstrated how to create a Declarative pipeline in Jenkins that can automatically build Docker images for updated app code and deploy them to the targeted deployment environment, notify management teams about new releases/updates, and perform clean-up for the host server.

Using CICD pipelines teams can streamline their software development processes, improve software quality, and achieve faster release cycles. By automating the repetitive and error-prone tasks of building, testing, and deployment, teams can focus on writing high-quality code and delivering value to their users.

I hope you learned something today with me!

Stay tuned for my next blog on the "DevOps Project - 03". I will keep sharing my learnings and knowledge here with you.

Let's learn together! I appreciate any comments or suggestions you may have to improve my blog content.

Thank you,

Chaitannyaa Gaikwad

Chaitannyaa Gaikwad | LinkedIn

Did you find this article valuable?

Support Chaitannyaa Gaikwad by becoming a sponsor. Any amount is appreciated!