Skip to content

GitOps with Argo CD: Simplifying Kubernetes Deployments Across Multiple Environments

Published: at 08:31 PMSuggest Changes

GitOps with Argo CD: Simplifying Kubernetes Deployments Across Multiple Environments

What is GitOps and Why Should You Care?

GitOps is an operational framework that takes DevOps best practices used for application development such as version control, collaboration, compliance, and CI/CD, and applies them to infrastructure automation.

In simpler terms, GitOps makes Git your single source of truth for infrastructure. Changes to your infrastructure happen through Git operations (commits, pull requests, merges), and automated tools ensure your actual infrastructure matches what’s defined in your Git repository.

For Kubernetes environments, GitOps has emerged as the gold standard for deployment strategies. Why? Because it provides:

The Challenge: Managing Multiple Environments

Many organizations struggle when implementing GitOps across multiple environments (dev, test, staging, production). A common approach is to use a branch strategy—maintaining a separate Git branch for each environment. This might seem intuitive initially, but it’s actually considered an anti-pattern for several reasons:

The Solution: Directories + Kustomize + Argo CD

Instead of using branches, a better approach leverages the power of directories combined with tools like Kustomize or Helm. This structure provides a clean, manageable way to handle multiple environments while maintaining a single source of truth.

Directory Structure That Works

Here’s an example structure using Kustomize:

apps/
    example-app/
        base/
            kustomization.yaml
            deployment.yaml
            service.yaml
            configmap.yaml
        overlays/
            dev/
                kustomization.yaml
                configmap.patch.yaml
            test/
                kustomization.yaml
                deployment.patch.yaml
            qa/
                kustomization.yaml
                deployment.patch.yaml
            prod/
                kustomization.yaml
                deployment.patch.yaml
                hpa.yaml
    example-app-2/
        ...
    application-set.yaml

In this structure:

Kustomize: The Secret Sauce

Kustomize makes environment-specific configuration simple:

# overlays/dev/kustomization.yaml
apiVersion: kustomize.config.k8s.io/v1beta1
kind: Kustomization
resources:
  - ../../base
namespace: example-dev
patchesStrategicMerge:
  - configmap.patch.yaml
images:
  - name: example-app
    newTag: dev-latest
# overlays/prod/kustomization.yaml
apiVersion: kustomize.config.k8s.io/v1beta1
kind: Kustomization
resources:
  - ../../base
  - hpa.yaml
namespace: example-prod
patchesStrategicMerge:
  - deployment.patch.yaml
images:
  - name: example-app
    newTag: v1.2.3

With this approach, you can:

All while maintaining a clean separation and clear visibility of what’s different between environments.

Application Sets: Automating the Automation

Argo CD ApplicationSets supercharge your GitOps workflow by eliminating manual application creation. They’re like templates that automatically generate Argo CD Applications based on patterns.

Here’s an example ApplicationSet that would create applications for all environments of all apps:

apiVersion: argoproj.io/v1alpha1
kind: ApplicationSet
metadata:
  name: all-apps
spec:
  generators:
    - matrix:
        generators:
          - git:
              repoURL: https://github.com/your-org/gitops-repo.git
              revision: HEAD
              directories:
                - path: apps/*/
          - list:
              elements:
                - env: dev
                  cluster: https://kubernetes.default.svc
                - env: test
                  cluster: https://kubernetes.default.svc
                - env: qa
                  cluster: https://kubernetes.default.svc
                - env: prod
                  cluster: https://kubernetes.default.svc
  template:
    metadata:
      name: "{{path.basename}}-{{env}}"
    spec:
      project: default
      source:
        repoURL: https://github.com/your-org/gitops-repo.git
        targetRevision: HEAD
        path: "{{path}}/overlays/{{env}}"
      destination:
        server: "{{cluster}}"
        namespace: "{{path.basename}}-{{env}}"
      syncPolicy:
        automated:
          prune: true
          selfHeal: true

This single resource would create applications like:

No more manual application creation in the Argo CD UI!

The Full Picture: From Code to Cluster

Let’s tie it all together with a CI/CD workflow that makes deployments smooth:

  1. Developer pushes code to application repository
  2. CI pipeline runs:
    • Builds and tests the application
    • Creates a container image
    • Pushes the image to a registry
    • Updates the image tag in the GitOps repo (using tools like kustomize edit set image)
  3. Argo CD detects the change in the GitOps repo
  4. Argo CD synchronizes the application configuration with the cluster
  5. New version is deployed to the environment

GitOps Workflow

With this workflow, you get:

Summary

By implementing these patterns, you’ll create a smoother development experience, reduce operational overhead, and build a more reliable deployment process for your Kubernetes applications.

Happy GitOps-ing!


Next Post
GitOps with Argo CD: Simplifying Kubernetes Deployments Across Multiple Environments