Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

post: gitops for devs part-1 initial commit #27

Merged
merged 1 commit into from
Dec 21, 2023
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
272 changes: 272 additions & 0 deletions src/content/blog/gitops-for-devs-installation.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,272 @@
---
title: "GitOps for Devs - Part 01: Installation"
description: "In this article, we'll be setting up ArgoCD a GitOps controller and deploying a sample application"
pubDate: "Dec 21 2023"
heroImage: 'https://bizkt.imgix.net/posts/gitopsfordevs/GitOps-for-devs.jpeg'
badge: "New"
---

GitOps is a hot topic in DevOps circles these days, and this article series aims to break down its core concepts starting from the basics. We'll kick things off by setting up ArgoCD a GitOps controller and deploying a sample application in this initial article. The following parts will dive deeper into configuration specifics.

## GitOps 101

GitOps automates modern cloud infrastructure setup by treating configuration files like code—similar to application source code. It ensures consistent and replicable infrastructure deployments, aligning with DevOps practices for quick code deployment, efficient cloud resource management, and adaptation to rapid development cycles.

In the context of Kubernetes and microservices, GitOps serves as a model for managing infrastructure and application deployment.

- **Declarative Infrastructure Management:** GitOps takes a declarative approach, defining the desired state of the Kubernetes cluster and its resources in configuration files (often YAML) stored in a Git repository.
- **Automated Synchronization:** Changes to these configuration files trigger an automated synchronization process in GitOps, ensuring the Kubernetes cluster's actual state matches the defined state in the repository.
- **Version Control Using Git:** Leveraging Git for version control allows teams to track changes, revert as needed, and collaborate on configurations.

Now that we've covered the basics of GitOps, let's jump into the example. We'll deploy a simple application on Kubernetes using `minikube` to illustrate these concepts.

## Install minikube

If you haven't installed Minikube already you can follow the official documentation [here](https://minikube.sigs.k8s.io/docs/start/)

Or you can use below shell script to upgrade/install `minikube`, if you are a Linux user.

```bash
#! /bin/sh

# Minikube update script file
# Ref: https://stackoverflow.com/questions/57821066/how-to-update-minikube-latest-version

minikube delete && \
sudo rm -rf /usr/local/bin/minikube && \
sudo curl -Lo minikube https://storage.googleapis.com/minikube/releases/latest/minikube-linux-amd64 && \
sudo chmod +x minikube && \
sudo cp minikube /usr/local/bin/ && \
sudo rm minikube && \
minikube start &&\

# Enabling addons: ingress, dashboard
minikube addons enable ingress && \
minikube addons enable dashboard && \
minikube addons enable metrics-server && \
# Showing enabled addons
echo '\n\n\033[4;33m Enabled Addons \033[0m' && \
minikube addons list | grep STATUS && minikube addons list | grep enabled && \

# Showing the current status of Minikube
echo '\n\n\033[4;33m Current status of Minikube \033[0m' && minikube status
```

## Install Argo CD

Now if your `minikube` cluster is up and running you are ready to install the [ArgoCD](https://argo-cd.readthedocs.io/en/stable/). GitOps controller in your cluster.

ArgoCD installation is straightforward.

1. Create the namespace for the ArgoCD controller:

```bash
kubectl create namespace argocd
```

2. Install the Controller and CRDs

```bash
kubectl apply -n argocd -f https://raw.githubusercontent.com/argoproj/argo-cd/stable/manifests/install.yaml
```

You should expect to find similar resources within the argocd namespace.

```bash
NAME READY STATUS RESTARTS AGE
pod/argocd-application-controller-0 1/1 Running 0 137m
pod/argocd-applicationset-controller-6b67b96c9f-kcbxl 1/1 Running 0 137m
pod/argocd-dex-server-c9d4d46b5-phltz 1/1 Running 2 (136m ago) 137m
pod/argocd-notifications-controller-6975bff68d-c5cdl 1/1 Running 0 137m
pod/argocd-redis-7d8d46cc7f-gjk25 1/1 Running 0 137m
pod/argocd-repo-server-59f5479b7-xznv6 1/1 Running 0 137m
pod/argocd-server-7d7fdcb49-st9vj 1/1 Running 0 137m

NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
service/argocd-applicationset-controller ClusterIP 10.102.75.218 <none> 7000/TCP,8080/TCP 137m
service/argocd-dex-server ClusterIP 10.98.223.112 <none> 5556/TCP,5557/TCP,5558/TCP 137m
service/argocd-metrics ClusterIP 10.107.209.210 <none> 8082/TCP 137m
service/argocd-notifications-controller-metrics ClusterIP 10.108.175.166 <none> 9001/TCP 137m
service/argocd-redis ClusterIP 10.107.100.213 <none> 6379/TCP 137m
service/argocd-repo-server ClusterIP 10.96.189.5 <none> 8081/TCP,8084/TCP 137m
service/argocd-server LoadBalancer 10.100.189.178 <pending> 80:31040/TCP,443:30770/TCP 137m
service/argocd-server-metrics ClusterIP 10.109.158.60 <none> 8083/TCP 137m

NAME READY UP-TO-DATE AVAILABLE AGE
deployment.apps/argocd-applicationset-controller 1/1 1 1 137m
deployment.apps/argocd-dex-server 1/1 1 1 137m
deployment.apps/argocd-notifications-controller 1/1 1 1 137m
deployment.apps/argocd-redis 1/1 1 1 137m
deployment.apps/argocd-repo-server 1/1 1 1 137m
deployment.apps/argocd-server 1/1 1 1 137m

NAME DESIRED CURRENT READY AGE
replicaset.apps/argocd-applicationset-controller-6b67b96c9f 1 1 1 137m
replicaset.apps/argocd-dex-server-c9d4d46b5 1 1 1 137m
replicaset.apps/argocd-notifications-controller-6975bff68d 1 1 1 137m
replicaset.apps/argocd-redis-7d8d46cc7f 1 1 1 137m
replicaset.apps/argocd-repo-server-59f5479b7 1 1 1 137m
replicaset.apps/argocd-server-7d7fdcb49 1 1 1 137m

NAME READY AGE
statefulset.apps/argocd-application-controller 1/1 137m

```

If you focus on the pods deployed in the namespace, there are three main pods, which are noteworthy.

- `pod/argocd-application-controller-0`

The controller continuously monitors running applications, identifies and resolves inconsistencies in their states (OutOfSync), and executes user-defined hooks for synchronization lifecycle events.

- `pod/argocd-repo-server-59f5479b7-xznv6`

The Repo server serves as an internal service supporting the application infrastructure by maintaining a local cache of Git repositories and generating Kubernetes manifests using repository-specific details.

- `pod/argocd-server-7d7fdcb49-st9vj`

Argo CD server aka the API server acts as a central interface for various components, managing application operations, credentials, authentication, RBAC, and Git webhook events while serving as a gRPC/REST server for Web UI, CLI, and CI/CD systems.

As a part of the installation, following CRDs should be created in your cluster.

```bash
applications.argoproj.io
applicationsets.argoproj.io
appprojects.argoproj.io
```

## Terminlogies

Assuming you're familiar with core concepts in Git, Docker, Kubernetes, Continuous Delivery, and GitOps, here are some specific Argo CD terminologies:

- **Application:** A defined group of Kubernetes resources outlined in a manifest, treated as a Custom Resource Definition (CRD).
- **Target State:** The intended state of an application, represented by files in a Git repository.
- **Live State:** The current operational state of that application, including deployed pods and other components.
- **Sync Status:** Indicates whether the live state matches the intended target state described in Git—essentially, is the deployed application aligned with what's defined in the repository?
- **Sync:** The process of transitioning an application to its target state, often done by applying changes to a Kubernetes cluster.
- **Sync Operation Status:** Indicates the success or failure of a sync process.
- **Refresh:** The action of comparing the latest code in Git with the live state to identify any differences.
- **Health:** Reflects the operational health of the application—whether it's functioning correctly and able to handle requests.

See [here](https://argo-cd.readthedocs.io/en/stable/core_concepts/) for further information

### ArgoCD UI

ArgoCD ships with a very powerful User Interface. To access the UI,

1. Set the port forwarding
2. Grab the admin password

```bash
# Port Forwarding
kubectl port-forward svc/argocd-server -n argocd 8080:443

# Intial admin password
kubectl -n argocd get secret argocd-initial-admin-secret -o jsonpath="{.data.password}" | base64 -d
```

Now you should be able to access ArgoCD UI using `https://127.0.0.1:8080` using the user name **"admin"**.

ArgoCD has its own CLI as well, follow [this](https://argo-cd.readthedocs.io/en/stable/getting_started/#2-download-argo-cd-cli) article to install the `argocli`.

## Install the Sample App - Album App

For this tutorial, I am using two repos:

- **Config**: <https://github.com/krishanthisera/album-app-config>
- **Application**: <https://github.com/krishanthisera/album-app-config>

As the names suggest, the config repo contains the infrastructure configuration, in this case, helm charts, while the application repo contains the code for **frontend** and **backend**.

_From now on **application** is referred to **ArgoCD Application** objects inside the Kubernetes cluster._

To install the Album App,

1. Clone the config repo (I would recommend you to fork the repo and then clone the fork)
2. Install the helm chart

Here we are using ArgoCD **App-of-Apps pattern**. The root ArgoCD application object is defined in the `root-app` helm chart. The root application is responsible for creating ArgoCD application objects for Frontend and Backend.

```bash
# Clone the repo
git clone https://github.com/krishanthisera/album-app-config

# Create a namespace for the application
kubectl create ns album-app

# Install the root app
helm install root-app ./root-app -n album-app

# Check the installed application
kubectl get applications.argoproj.io -n argocd
NAME SYNC STATUS HEALTH STATUS
root-app OutOfSync Healthy
```

Once you install the helm chart, you should be able to see `root-app` under the applications in ArgoCD UI.

![Argo UI](https://bizkt.imgix.net/posts/gitopsfordevs/ARGO_UI.png)

![Root App](https://bizkt.imgix.net/posts/gitopsfordevs/ARGO_ROOT-APP.png)

If you look closely, the `root-app` is `out-sync`; let's sync the `root-app`.

For now let's keep the default settings as it is.

![Sync Root App](https://bizkt.imgix.net/posts/gitopsfordevs/ARGO_SYNC_ROOT-APP.png)

Once the `root-app` is synced, both the Frontend and Backend ArgoCD application objects are created but they are yet to be synced.

![Unsync Album Apps](https://bizkt.imgix.net/posts/gitopsfordevs/ARGO_ALBUM-APPS.png)

Let's do the same for the Frontend and Backend application; let's get them synced.

Once both the Frontend and Backed apps are synced your ArgoCD applications should be look like this.

![Synced Frontend](https://bizkt.imgix.net/posts/gitopsfordevs/ARGO-ALBUM-APP_FE.png)

![Synced Backend](https://bizkt.imgix.net/posts/gitopsfordevs/ARGO_ALBUM-APP_BE.png)

All the applications should be synced now 💫

```bash
kubectl get applications.argoproj.io -n argocd
NAME SYNC STATUS HEALTH STATUS
album-app-backend Synced Healthy
album-app-frontend Synced Healthy
root-app Synced Healthy
```

If you do a `kubectl get all` on the `album-app` namespace,

```bash
NAME READY STATUS RESTARTS AGE
pod/album-app-backend-backend-dcbbc657-stqmc 1/1 Running 0 11h
pod/album-app-frontend-frontend-789b5bbc67-9vd5x 1/1 Running 0 11h

NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
service/album-app-backend ClusterIP 10.104.144.238 <none> 8080/TCP 11h
service/album-app-frontend-frontend ClusterIP 10.108.6.129 <none> 80/TCP 11h

NAME READY UP-TO-DATE AVAILABLE AGE
deployment.apps/album-app-backend-backend 1/1 1 1 11h
deployment.apps/album-app-frontend-frontend 1/1 1 1 11h

NAME DESIRED CURRENT READY AGE
replicaset.apps/album-app-backend-backend-dcbbc657 1 1 1 11h
replicaset.apps/album-app-frontend-frontend-789b5bbc67 1 1 1 11h
```

Further, you can port-forward the `album-app` Frontend service.

```bash
kubectl port-forward -n album-app services/album-app-frontend-frontend 8090:80
Forwarding from 127.0.0.1:8090 -> 3000
Forwarding from [::1]:8090 -> 3000
```

![Album App UI](https://bizkt.imgix.net/posts/gitopsfordevs/ARGO_FE_UI.png)

Now that we have installed the Album App, let's dive deep into the ArgoCD-specific configuration.

Stay tuned. To be continued...