/vote-app-gitops

A demo of cloud-native Inner Loop and Outer Loop controlling a 2-tier app (Python + Go) with Red Hat OpenShift using Tekton Pipelines, Argo CD GitOps, Eclipse Che aka Codeready Workspaces and Quay.io registry

OpenShift GitOps Demo (Tekton + Argo CD)

The aim of this demo is to show how to interconnect Tekton Continuous Integration (CI) capabilities to Argo CD enhanced features for Continuous Delivery (CD) with Red Hat OpenShift.

Architecture

The flow of this demo is:

The demo is based on the Vote App 2-tier app (Python Flask frontend + Go backend):

Assets and repos

Git

We use three git repos.

App source repos

Vote App source repos (Frontend + Backend):

GitOps repo

This repo, used to store Kubernetes manifests and Argo application:

There are 2 branches:

  • develop: used to deploy and sync manifests in a Dev environment
  • main: used to deploy and sync manifests Prod, uses Argo CD Sync Waves to order manifests sync.

Create a Personal Access Token for GitHub

The pipelines will update the Kubernetes manifests for the applications, so they need to push to the both Vote App UI and Vote App API repos.

Create a Personal Access Token for your user, it will be used later.

Container Registry

In this demo we use Quay.io as public container registry with private access to store container images.

Example:

  • quay.io/bluesman/vote-api:latest
  • quay.io/bluesman/vote-ui:latest

Setup

To run this demo you need OpenShift >=4.7 with cluster-admin privileges.

Quay.io

Account

Create an account on Quay.io if you do not already have one.

Repositories

Create two repositories with public access (pull), you will use credentials in the next step to push container images.

From right-side menu, click Create New Repository

Create two repositories:

  • vote-ui
  • vote-api

Flag it as Public Repository and click Create Public Repository

Create repo

Repositories are created and listed:

Create repo

Create secret

  • Login to quay.io in the web user interface and click on your username in the top right corner.
  • Select account settings.
  • Click the blue hyperlink Generate Encrypted Password.
  • Re-enter your password when prompted.
  • Copy the password

Create repo

Setup OpenShift

Login to OpenShift Web Console to install prerequisites.

Install Operators

OpenShift Pipelines

OpenShift Pipelines is provided as an add-on on top of OpenShift that can be installed via an operator available in the OpenShift OperatorHub. Follow these instructions in order to install OpenShift Pipelines on OpenShift via the OperatorHub.

From the left-side menu under Administrator perspective, go to Operators-> OperatorHub. In the search box, search for pipelines, then click to OpenShift Pipelines Operator:

OperatorHub

From the description view, click Install to review all installation settings.

Install Pipelines

Ensure Update Channel is set to stable , and click Install to start installing the Operator.

Install Operator

After few seconds, the installation should be completed with success and you can verify it looking at Status column, check if the status is Succeeded.

Pipelines Installed

OpenShift GitOps

Log into OpenShift Web Console as a cluster admin and navigate to the Administrator perspective and then OperatorsOperatorHub.

In the OperatorHub, search for OpenShift GitOps and follow the operator install flow to install it.

OpenShift GitOps operator

OpenShift GitOps operator

OpenShift GitOps operator

Add permission to Argo CD service account

IMPORTANT Give permission to the Argo CD service account to control the cluster:

oc adm policy add-cluster-role-to-user cluster-admin -z openshift-gitops-argocd-application-controller -n openshift-gitops

Once OpenShift GitOps is installed, an instance of Argo CD is automatically installed on the cluster in the openshift-gitops namespace and link to this instance is added to the application launcher in OpenShift Web Console.

Application Launcher

Log into Argo CD dashboard

Argo CD upon installation generates an initial admin password which is stored in a Kubernetes secret. In order to retrieve this password, run the following command to decrypt the admin password:

oc extract secret/openshift-gitops-cluster -n openshift-gitops --to=-

Click on Argo CD from the OpenShift Web Console application launcher and then log into Argo CD with admin username and the password retrieved from the previous step.

Argo CD

Argo CD

Codeready Workspaces

CodeReady Workspace is an in-browser IDE that will be used to edit and test the code from OpenShift with a pre-build Workspace from a Devfile.

Log into OpenShift Web Console as a cluster admin and navigate to the Administrator perspective and then OperatorsOperatorHub.

In the OperatorHub, search for Codeready Workspaces and follow the operator install flow to install it.

In the Red Hat CodeReady Workspaces pop-up window, click the Install button.

CRW

On the Install Operator screen, leave default option:

  • Installation mode: A specific project on the cluster
  • Installed Namespace: Pick an existing project → openshift-workspaces
Creating an instance of CodeReady Workspaces

To create an instance of the Red Hat CodeReady Workspaces Operator, in the left panel, navigate to the OperatorsInstalled Operators section.

In the Installed Operators screen, click the Red Hat CodeReady Workspaces name.

In the Operator Details screen, in the Details tab, inside of the Provided APIs section, click the Create Instance link.

The Create CheCluster page contains the configuration of the overall CodeReady Workspaces instance to create. It is the CheCluster Custom Resource. Keep the default values.

To create the codeready-workspaces cluster, click the Create button in the lower left corner of the window.

On the Operator Details screen, in the Red Hat CodeReady Workspaces Cluster tab, click on the codeready-workspaces link.

To navigate to the codeready-workspaces instance, click the link under Red Hat CodeReady Workspaces URL.

NOTE The installation might take more than 5 minutes. The URL appears after the Red Hat CodeReady Workspaces installation finishes.

Setup vote-ci project

  1. Create a new Project for the CI part with Tekton.
oc new-project vote-ci
  1. Create a Secret with your Quay.io credentials with the encrypted password you copied before:
oc create secret docker-registry quay-secret --docker-server=quay.io --docker-username=<QUAY_USERNAME> --docker-password=<ENCRYPTED_PASSWORD>
  1. Create a Secret with your GitHub Personal Access Token
apiVersion: v1
kind: Secret
metadata:
  name: git-user-pass
  annotations:
    tekton.dev/git-0: https://github.com
type: kubernetes.io/basic-auth
stringData:
  username: <github user>
  password: <github personal access token>

Save it to a file with your credentials and create the secret:

oc create -f git-user-pass.yaml
  1. Link Secrets to pipeline Service Account.

NOTE: Pipelines Operator installs by default a pipeline Service Account in all projects. This service account is used to run non-privileged containers and builds across the cluster.

oc secret link pipeline quay-secret
oc secret link pipeline git-user-pass
  1. Fork repos

In order to enable webhooks, fork source code repos to use them in pipelines:

  1. Clone vote-api repo
git clone https://github.com/blues-man/pipelines-vote-api
cd pipelines-vote-api
  1. Create Tekton pipeline manifests

Change the GitOps repo to your fork:

sed -i 's/bluesman/yourquayuser/g' k8s/pipeline.yaml
sed -i 's/blues-man/yourgithubuser/g' k8s/pipeline.yaml
oc create -f k8s/vote-api-pvc.yaml
oc create -f k8s/git-update-deployment-task.yaml
oc create -f k8s/pipeline.yaml
oc create -f k8s/triggertemplate.yaml
oc create -f k8s/triggerbinding.yaml
oc create -f k8s/eventlistener.yaml
oc create -f k8s/el-route.yaml
  1. Clone vote-ui repo
cd ..
git clone https://github.com/blues-man/pipelines-vote-ui
cd pipelines-vote-ui
  1. Create pipeline manifests

Change the GitOps repo to your fork:

sed -i 's/bluesman/yourquayuser/g' k8s/pipeline.yaml
sed -i 's/blues-man/yourgithubuser/g' k8s/pipeline.yaml
oc create -f k8s/vote-ui-pvc.yaml
oc apply  -f k8s/git-update-deployment-task.yaml
oc create -f k8s/pipeline.yaml
oc create -f k8s/triggertemplate.yaml
oc create -f k8s/triggerbinding.yaml
oc create -f k8s/eventlistener.yaml
oc create -f k8s/el-route.yaml

Demo flow

The goal is to show how to OpenShift can create and connect the CI and CD part with Tekton and ArgoCD, using CRW and Quay.io for the Inner and Outer Loop.

1. Pipelines overview

Pipelines

1. Start vote-api pipeline

Start vote-api pipeline manually.

From the left-side menu under Developer perspective, go to Pipelines.

Click to vote-api pipeline.

From right side menu, click Start.

In GIT_REPO use your fork of the vote-api repo

In IMAGE_NAME put your vote-api container image from Quay.io

Leave all other settings as default.

Select PVC under Workspace section and select vote-api-pvc persistent volume claim.

Click Start

Vote API Pipeline

2. Start vote-ui with a Webhook

Tekton supports Tekton Triggers to enable automation and web hooks to Pipelines. All settings have been already installed from previous command, and both pipelines support web hooks.

Triggers

From Topology view, click to el-vote-ui Deployment, go into Routes section and and copy the el-vote-ui Route URL.

Vote API Pipeline

Once you have the URL copied to your clipboard, navigate to the code repository fork that you have on GitHub.

From your fork page top-right menu, click Settings. Then from result left-side menu, click Webhook, then from right side click Add webhooks.

In the next screen, paste your link into the Payload URL field. You can leave the secret token field blank.

Change the Content Type to application/json.

Finally, click on Add Webhook.

Vote UI Web Hook

NOTE: If you forked the repo, update the TriggerTemplate with your vote-ui repo on quay.io reference adding the IMAGE_NAME parameter, otherwise it will fallback to quay.io/bluesman/vote-ui:

oc edit triggertemplate vote-ui
...
    spec:
      params:
      - name: APP_NAME
        value: $(tt.params.git-repo-name)
      - name: GIT_REPO
        value: $(tt.params.git-repo-url)
      - name: GIT_REVISION
        value: $(tt.params.git-revision)
      - name: IMAGE_NAME
        value: quay.io/<your-username>/vote-ui

Do some change to the source code and verify that the pipeline starts.

You can also use CodeReadyWorkspaces for that (change this URL with the one for your CodeReady Workspaces):

Contribute

Vote UI Pipeline start

NOTE: you can also trigger the Pipeline start by changing and pushing the code at the CRW step later

3. Quay.io security scan

  • Verify generated images have been pushed to Quay.
  • Verify vote-ui image has been scanned with no vulnerabilities found

Vote UI Security scan

2. Argo CD for DEV project

We will pre-deploy the DEV environment in the vote-app-dev project.

  1. Fork this vote-app-gitops repository
  2. Clone your repository fork in the develop branch:
git clone -b develop https://github.com/blues-man/vote-app-gitops.git
cd vote-app-gitops
  1. Update Argo CD application repoURL with your fork, this will be used to deploy the app to the DEV environment
apiVersion: argoproj.io/v1alpha1
kind: Application
metadata:
  name: vote-app-dev
  namespace: openshift-gitops
spec:
  destination:
    namespace: vote-app-dev
    server: https://kubernetes.default.svc 
  project: default 
  source: 
    path: environments/dev
    repoURL: https://github.com/blues-man/vote-app-gitops
    targetRevision: develop
  syncPolicy: 
    automated:
      prune: true
      selfHeal: false
    syncOptions:
    - CreateNamespace=true
  1. Update all references to quay.io with your repos for vote-ui and vote-api images:
sed -i 's/bluesman/yourquayuser/g' k8s/api-deployment.yaml k8s/ui-deployment.yaml
git commit
git push
  1. Create Argo CD Application to deploy DEV env
oc apply -f argo/vote-app-dev.yaml

Vote App Dev details

1. Verify App deployment

Go to Topology view in vote-app-dev Project.

Vote App Dev view

2. Access the app

Access the app from vote-ui Route clicking on the Python icon and then accessing Route URL.

3. Edit app in CodeReady Workspaces

Edit source code from CRW by clicking on the little crw icon next to the vote-ui in the Topology view. This will launch Eclipse Che Factory reading the dev environment from the Devfile in the vote-ui repository.

This will open CRW and you can demo how to edit and run the app from an IDE.

In CRW, from Run Tasks click Install dependencies and Run Python app.

This will open an embedded window with the app running locally.

CRW Vote App

4. Detect drifts

Let Argo CD detect a drift between what declared in Git and what it is available in the cluster.

Change vote-ui replicas to 2 from OpenShift and verify the status is Out of Sync on Argo CD.

TIP: if the dashboard page doesn't update, try to hit the Refresh button from the Argo CD web console

Out of Sync

5. Sync the app

Sync manually the app from the Argo CD console, as we declared in our Application that we don't want to self-heal for DEV project.

From top menu, click SYNC.

From right side window, click SYNCHRONIZE and leave default settings.

This will rollback the vote-ui deployment replicas to 1.

3. Argo CD for PROD project

We create the PROD environment directly from Argo CD Web console this time.

Before doing that, update all the references to quay.io with your images also for the main branch.

git checkout main
sed -i 's/bluesman/yourquayuser/g' k8s/api-deployment.yaml k8s/ui-deployment.yaml
git commit
git push

Access Argo CD Web console, from homepage click on top-left to the +NEW APP button.

Under General:

  • Application: vote-app-prod
  • Project: default
  • Sync Policy: Automatic
    • Prune resources: true
    • Self Heal: true
  • Sync Options: Auto-create namespace

Under Source section:

Under Destination section:

Click CREATE

Create Vote App Prod

Review auto-generated Application manifest:

apiVersion: argoproj.io/v1alpha1
kind: Application
metadata:
  name: vote-app-prod
  namespace: openshift-gitops
spec:
  destination:
    namespace: vote-app-prod
    server: https://kubernetes.default.svc 
  project: default 
  source: 
    path: environments/prod
    repoURL: https://github.com/blues-man/vote-app-gitops
    targetRevision: main
  syncPolicy: 
    automated:
      prune: true
      selfHeal: true
    syncOptions:
    - CreateNamespace=true

1. Review deployment order

PROD environment is using Sync Waves, this means Kubernetes manifests in the main branch are annotated with sync weves to order manifests deployment.

metadata:
  annotations:
    argocd.argoproj.io/sync-wave: "0"
  1. API Deployment
  2. API Service
  3. UI Deployment
  4. UI Service
  5. UI Route

Create Vote App Prod

2. Verify app is deployed in vote-app-prod Project

Verify vote-ui and vote-api are deployed also in vote-app-prod project now.

NOTE: CodeReady Workspaces icons are not present in the Topology view this time, because we haven't annoted the Deployment for that, as this is a Prod app!

3. Auto detect drift

Change vote-ui replicas to 2 from OpenShift Web Console, Argo CD will automatically sync and auto-heal in this case.

Vote App Prod

4. Make a change to PROD from GitHub with a Pull Request

  1. Create a new feature branch in the GitHub repo called feature-ha
  2. Change ui-deployment.yaml with replicas:2
  3. Create PR
  4. Merge into main
  5. Prod app is with 2 replicas

Vote App Prod Scaled

Well done!