Operator installation with Argo CD
- - 6 min read
GitOps for application deployment and cluster configuration is a must-have I am trying to convince every customer to follow from the very beginning when starting the Kubernetes journey. For me, as more on the infrastructure side of things, I am more focused on the configuration of an environment. Meaning, configuring a cluster, installing an operator etc.
In this article, I would like to share how I deal with cluster configuration when certain Kubernetes objects are dependent on each other and how to use Kubernetes but also Argo CD features to resolve these dependencies.
This article assumes that you have the openshift-gitops Operator, which provides Argo CD, already installed, and configured. If you are new to GitOps check out this article: Argo CD |
TL;DR
If you want to jump directly to the technical fun part, go here: Let’s start.
The Idea
Everything should be seen as a code. Everything should be possible to be deployed in a repeatable way. With a GitOps approach, everything is stored naturally in Git and from there, a GitOps agent validates and synchronizes changes to one or more clusters.
When it comes to OpenShift, Red Hat supports Argo CD using the Operator openshift-gitops. This gives you everything you need to deploy an Argo CD instance. The only thing you need to take care of is a Git repository, no matter if it is GitHub, Gitlab, Bitbucket etc.
The Problem
Sometimes Kubernetes objects depend on each other. This is especially true when you would like to install and configure Operators, where the configuration, based on a Customer Resource Definition (CRD), can only happen after the Operator has been installed and is ready.
Why is that? Well, when you want to deploy an Operator, you will store a “Subscription object” in Git. Argo CD will take this object and applies it to the cluster. However, for an Operator, the creation of the Subscription object is just the first step. A lot of other steps are required until the Operator gets ready. Unfortunately, Argo CD cannot verify if the installation is successful. All it sees is that the Subscription object has been created and then it immediately tries to deploy the CRD. The CRD which is not yet available on the system because the Operator is still installing it.
Even if you use Argo CD features like Sync waves it would not wait until the Operator is successfully installed because for Argo CD the “success” is the creation of the Subscription object.
Subsequently, the Argo CD synchronisation process will fail. You could now try to automatically “Retry” the sync or use multiple Argo CD applications that you execute one after each other, but I was not fully happy with that and tried a different approach.
My Solution
Let’s say I would like to deploy and configure the Compliance Operator. The steps would be:
Install the Operator.
Wait until the Operator is ready.
Configure Operator specific CRDs.
This “Wait until the Operator is ready” is the tricky party for Argo CD. What I have done is the following:
Install the Operator, this is the first step and is done during Sync Wave 0.
Create a Kubernetes Job that verifies the status of the Operator. This Job additionally requires a ServiceAccount and a role with a binding. They are configured during Sync Wave is 1. Moreover, I use a Hook (another Argo CD feature) with the deletion policy “HookSucceeded”. This makes sure that the Job, ServiceAccount, Role and RoleBinding are removed after the status has been verified. The verification is successful as soon as the Operator status says “Succeeded”. In fact, all the Job does is to execute some oc commands. For example,
oc get clusterserviceversion openshift-gitops-operator.v1.8.0 -n openshift-gitops -o jsonpath={.status.phase} Succeeded
Finally, during the next Sync Wave (2+) the CRD can be deployed. In this case, I deploy the object ScanSettingBinding.
In Argo CD everything is correctly synchronized, and the Operator and its configuration is in place.
If you are new to the compliance operator, I recommend the following article: Compliance Operator |
I use this approach for every Operator that I would like to install and configure at the same time. For example, I do the same for Advanced Cluster Security or Advanced Cluster Management where I use the Job to verify if everything is ready before I let Argo CD continue.
More information about Sync Waves and Hooks can be found in the official Argo CD documentation: Sync Phases and Waves |
Let’s see this in Action
Prerequisites
OpenShift cluster 4.x
openshift-gitops is installed and ready to be used.
Access to GitHub (or to your own Repository)
I will be using my Helm Chart repository at https://charts.stderr.at/ and from there the charts:
compliance-operator-full-stack
helper-operator (sub chart): Responsible to install the Operators.
helper-status-checker (sub chart): Responsible to check the status of the Operator.
Why do I use Helm charts? There is no specific reason for that. I started with Helm for the cluster configuration and now it has evolved with a separate Chart repository and sub-charts and so on.
Argo CD Application
In Argo CD I have the following Application:
apiVersion: argoproj.io/v1alpha1
kind: Application
metadata:
name: in-cluster-install-compliance-scans
namespace: openshift-gitops
spec:
destination:
namespace: default
server: 'https://kubernetes.default.svc' (1)
info:
- name: Description
value: Deploy and configure the Compliance Scan Operator
project: in-cluster
source:
path: charts/compliance-operator-full-stack (2)
repoURL: 'https://github.com/tjungbauer/helm-charts'
targetRevision: main
1 | Installing on the local cluster where Argo CD is installed. |
2 | Git configuration, including path and revision. |
Actually, this Application is created out of an ApplicationSet, but I did not want to make it too complex :) |
The Application would like to synchronize the objects:
Subscription
OperatorGroup
Namespace (openshift-compliance)
ScanSettingBinding
Where are the objects we need for the Job? Since they are only available during the Sync-Hook they will not show up here. In fact, they will only show up during the time they are alive and will disappear again after the status of the operator has been verified. |
Helm Chart Configuration
The Helm Chart gets its configuration from a values file. You can verify the whole file on GitHub.
The important pieces here are that some variables are handed over to the appropriate Sub Charts.
Operator Configuration
This part is handed over to the Chart “helper-operator”.
helper-operator:
operators:
compliance-operator:
enabled: true
syncwave: '0'
namespace:
name: openshift-compliance
create: true
subscription:
channel: release-0.1
approval: Automatic
operatorName: compliance-operator
source: redhat-operators
sourceNamespace: openshift-marketplace
operatorgroup:
create: true
notownnamespace: true
It is executed during Sync Wave 0 and defines if a Namespace (openshift-compliance) shall be created (true) and the specification of the Operator which you need to know upfront:
channel: Defines which channel shall be used. Some operators offer different channels.
approval: Either Automatic or Manual … defines if the Operator shall be updated automatically or requires an approval.
operatorName: the actual name of the Operator (compliance-operator)
source: Where does this Operator come from (redhat-operator)
sourceNamespace: In this case openshift-marketplace
You can fetch these values by looking at the Packagemanifest:
oc get packagemanifest compliance-operator -o yaml
Status Checker Configuration
This part is handed over to the Sub-Chart "helper-status-checker"". The main values here are the operatorName and the namespace where the Operator is installed.
What is not visible here is the Sync Wave, which is per default set to 1 inside the Helm Chart. If you need to overwrite it, it can be configured in this section as well.
helper-status-checker:
enabled: true (1)
# use the value of the currentCSV (packagemanifest) but WITHOUT the version !!
operatorName: compliance-operator (2)
# where operator is installed
namespace:
name: openshift-compliance (3)
serviceAccount:
create: true
name: "sa-compliance" (4)
1 | Is the status checker enabled or is it not. |
2 | The name of the operator as it is reported by the value currentCSV inside the packageManifest |
3 | The namespace where the Operator has been installed. |
4 | The name of the ServiceAccount that is created temporarily. |
The operatorName is sometimes different than the Operator name required for helper-operator chart. Here it seems the value of the currentCSV must be used but without the version number. (The Job will look up the version itself) |
Operator CRD configuration
The final section of the values file manages the configuration for the Operator itself. This section does not use a Sub Chart. Instead, the variables are used in the Main-Chart. In this example, the ScanSettingBinding will be configured during Sync Wave 3, which is all we need to basic functionality.
compliance:
scansettingbinding:
enabled: true
syncwave: '3' (1)
profiles: (2)
- name: ocp4-cis-node
- name: ocp4-cis
scansetting: default
1 | Define the Sync Wave. This value must be higher than the Sync Wave of the helper-status-checker |
2 | ScanSettingBinding configuration. Two profiles are used in this example. |
Synchronizing Argo CD
Basic Application in Argo CD before it is synced:
Figure 2. Argo CD: ApplicationSync Wave 0: Synchronization has started. Namespace and Subscription are deployed.
Figure 3. Argo CD: Synchronization is started (Sync Wave 0)Sync Wave 1: Status Checker Job has started and tries to verify the Operator.
Figure 4. Argo CD: Status Checker Job started (Sync Wave 1)The Log output of the Operator. You can see that the status switches from Pending to Installing to Succeeded.
Figure 5. Argo CD: Log of the Status Checker PodAfter Sync Wave 3, the whole Application has been synchronized and the Checker Job has been removed.
Figure 6. Argo CD: Compliance Operator is fully deployed
Copyright © 2020 - 2025 Toni Schmidbauer & Thomas Jungbauer