Skip to content

Deploying a Todo app Helm Charts stack consisting of Backend (ExpressJS), Frontend (Angular), NodeJS (Javascript runtime), and Mongo (DB) into Kubernetes using GitOps.

Notifications You must be signed in to change notification settings

SheblMohamed/Todo-Full-Stack-Deployment-GitOps

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Todo Full Stack Deployment

Introducing a full deployment cycle for events app deployed on Kubernetes using the ArgoCD tool and Helm charts. it uses AngularJS as a Frontend framework connected to Express.js as a backend web framework and it interacts with MongoDB to retrieve or store data. using Hashicorp vault to store the application secrets & validating certificates for the application ingress, External secret operator for creating the secrets along with the vault, and Github actions as a pipeline for building & pushing the backend & frontend image to GitHub package registry and for updating each image tag on the app charts for main, stage, dev branches according to each branch case. then Argo triggers the new commits and updates each app manifests.

project

Application components

  • MongoDB - A NoSQL database.
  • ExpressJS - A backend web application framework for NodeJS.
  • Angular - A JavaScript-based front-end web development framework for building dynamic, single-page web applications.
  • NodeJS - A JavaScript runtime environment that enables running JavaScript code outside the browser, among other things.

how the different components of the app might work together:

  • A user visits the event posting app’s website on their browser. AngularJS, the frontend framework, retrieves the necessary HTML, CSS, and JavaScript files from the server and renders the initial view of the website.
  • When the user wants to view a list of events or create a new event, AngularJS sends an HTTP request to the backend server.
  • Express.js, the backend web framework, receives the request and processes it. This step includes interacting with the MongoDB database to retrieve or store data and providing an API for the frontend to access the data.
  • The back-end server sends a response to the frontend, which AngularJS receives and uses to update the view.
  • When a user creates a new event, AngularJS sends a POST request to the backend server, which Express.js receives and processes. Express.js stores the new event in the MongoDB database.
  • The backend server sends a confirmation response to the front-end, which AngularJS receives and uses to update the view and display the new event.
  • Node.js, the JavaScript runtime, handles the server-side logic for the application and allows for real-time updates. This includes running the Express.js server, handling real-time updates using WebSockets, and handling any other server-side tasks. Node-based Express server.

Helm setup

  • Helm is always the best choice for me to deploy any component I need inside the cluster with customized values, during the project, I used some helm packages I would like to describe them:
    • Cert-manager
      • for being able to deploy issuer & certificate manifests that are required for ingress to be secured with HTTPS protocol.
    • Hashicorp-Vault
      • for deploying the secrets required by each app in my case I stored the private registry credentials that are required for each container to access & pull the images.
      • issuer reference for creating a valid certificate for ingress with the custom domain todo.com.
    • External-secrets
      • for handling the secrets inside each namespace, I used it to deploy a secret store connected with the vault to retrieve any secrets required by any application on any namespace.
    • Argo
      • for deploying the argo stack with custom values.

charts

GitHub actions cycle

  • The repo has 3 branches: Main - Stage - Dev Each branch will have 3 folders:
    • Backend - the Backend code of the application.
    • Frontend - the Frontend code of the application.
    • App-charts - the application manifests are built with custom helm charts.
  • I built a custom workflow for each folder to trigger any commits that will be pushed by developers.
  • Let's say that we have 2 teams working on the app stack, the Backend team and the frontend team.
  • Each team tries to release a new feature on the dev branch the backend team tries to release a feature on the Backend code and the same for the frontend team.
  • If the backend team made a commit on backend/ folder, the Backend workflow will trigger that commit and will automatically run the pipeline to build the backend image and then upload it to the Github packages registry.
  • Same previous steps for Frontend/ folder.
  • Each workflow is configured to update image tags on the app helm charts according to each branch.
  • For example, if the commit happened on the frontend code folder on the stage branch. the frontend workflow will trigger that commit and will build the frontend image then update the new frontend tag image on the app helm charts value on the stage branch with a custom commit msgs describing the updated image if backend / frontend with the commit version.
  • The same previous step will work for the other branches / folders.
  • After setting the app charts update, the GitHub actions will generate a new release package zipped with a new update on the GitHub release section configured with the branch name, and image tag.

Argo cycle

  • I deployed 3 apps on Argo, each app deployed from the app charts on each branch.
  • Argo is configured to watch the app charts on each branch.
  • Argo will trigger and fetch the new updates for any branch in my case after the GitHub actions will update the tag on folder A branch A, Argo will detect the new tag for the app which been deployed from this branch and will ask for update the app manifests with the new tag.
  • each pod using the registry credentials will be asked to authenticate & pull the new image tag and do a rolling update.

Application helm charts

  • I generated custom helm charts for the app stack which can be helpful to deploy the app on different namespaces with custom configurations and capacity. a good idea to deploy the app without shared resources conflict with Argo. This is also helpful for dynamically changing the application values live without any problems.
  • Helm charts of the app have these features:
    • ability to set vault credentials value from Argo.
    • ability to adjust the persistent volume & claim of the database.
    • ability to set a custom subdomain with the domain name todo.com verified by the Vault issuer.
    • all the global resources which not namespaced will have a custom name according to the hostname, for example, ingress and all its components names like issuer, and cert-manager. database mounted volume names, cluster secret store. all these resources will have a unique name with only the hostname value.
    • ability to scale up/down the replica number for each app.
    • ability to set up the database service name for the Mongo-express web app so it can connect to the right database and visualize it.
    • ability to set up the image repository & tag name for front & backend deployment. this will be customized dynamically by GitHub actions.
  • The great point here is that Argo supports the values setup during the app deployment. this is great for some options like vault tokens or even any other credential.

Full cycle test

  • First, the vault helm charts must be installed and set up with application credentials & the domain which will be used for ingress.
  • added credentials for the GitHub package registry.

vault

  • Setting up the PKI secrets engine with a custom role to generate dynamic certificates for a custom domain, in my case I added todo.com as a custom domain.
  • this role is configured with a custom service account authorized with PKI to request the certificates with the issuer.

vault-1

  • Installing External-secrets helm charts, I will set up the authentication between ESO and vault during Argo.

  • After that, I can deploy 3 apps from each branch, each app will be deployed on a different namespace, configured with custom values according to each branch name and also with a vault token. for example, this is the main app.

argo-1 argo argo-2

  • After deploying each app, I can check the ingress if it got the certificates and if the issuer was able to connect & authenticate with the vault or not.
  • From issuer pod

issuer

  • From Certificate

cert

  • After deploying the apps, if we checked the Vault/PKI secret path, we can see the requested certificates for each subdomain under the registered domain

cert-1 cert-2 cert-3

  • Let's check if External-secrets was able to connect with the vault

ex ex-1

  • Docker registry Secret created by External-secrets

ex-2

  • Let's test the values of helm charts if we can set some values and see the effect, for example, I will change the desired pods count from 1 to 2 and change the hostname of dev to dev new.
  • Before

old old-1 old-2

  • After

new new-1 new-2

  • we can check the pod's logs to see if the apps working fine.

pod pod-1 pod-2

  • From DB logs we can see that backend started to connect with the DB too

pod-3

  • Let's browse the main app to check if HTTPS protocol works. using events.todo.com we can see from page resources that it accepts the URL with HTTPS protocol and uses the 443 port. returns a status code of 200.

app

  • From the terminal, using curl

cert-4

  • Let's test the GitHub actions with some cases. Assume that I am a developer For example, I will make a little edit on file and commit it on stage, backend folder.

test test-1

  • From the action section, we can see that the backend workflow triggered the new commit 5707132 pushed it to the stage branch and started the pipeline

back

  • And its done

back-1

  • Let's check the pipeline steps:

    • Job 1 - will query all the changed files inside backend folder back-2
    • Job 2 - according to the previous step, it will print the changed files. in my case, I only made some changes on backend/tsconfig.json file. back-3
    • Job 3 - because the backend workflow is the current running pipeline, it will build the backend image. back-4
    • Job 4 - image scanning using trivy scanner. I set it with all security levels Critical, medium, and low. content-type is a table because I don't want a report file, I also disable the option of stopping the pipeline if the security lvl is medium ~ higher just for demonstration but sure in production it must be enabled. back-5 back-6
    • Job 5 - login to my private registry. back-7
    • Job 6 - pushing the backend image to the GitHub package registry using a custom tag. For the tag I used GitHub sha tag and made it shorter with 7 digits only for better commit contents and set the repo name to lowercase for being accepted. back-8
    • Job 7 - the important step, once the new image has been pushed to the registry, this job will catch the image tag and will check the current branch according to if condition. Because the commit has been added to the stage branch and the pipeline run from it, it will detect the stage branch and update the backend image tag value on the app charts values file. it also will release a new helm chart package on the tags section configured with the branch name and the tag version. back-9 back-10 back-11
  • If we checked the stage app on Argo, we will see that it detected the new tag updated and asked me to sync for updating the app, I closed the auto sync because I would like to review the changes first before they get applied.

argo-3

  • We can see the new tag changes from app diff

argo-4

  • Now we can sync for letting the new image be pulled.

argo-5 argo-6 argo-7

  • On GitHub

git git-1

  • And same all these steps for Frontend Workflow.

Application use cases & Mongo Express

  • The application stores any count of events to MongoDB, with some info about each event.

todo-1 todo

  • I deployed Mongo Express with the stack configured to connect with the DB service. it can help us to check the saved events inside Mongo and control on it.

express express-1

  • Another way, without Express, we can ssh into MongoDB internal service using Mongo shell.

express-2

  • We are already on test DB, we need to check event collections inside test DB.

express-3 express-4 express-5

  • We can use the DNS as API too

api

Ref