Skip to content

Commit 2c90be2

Browse files
Enable leader election for the operator by default. (#367)
1 parent 1a8661b commit 2c90be2

File tree

6 files changed

+63
-25
lines changed

6 files changed

+63
-25
lines changed

docs/running-the-operator.md

+22-4
Original file line numberDiff line numberDiff line change
@@ -126,10 +126,28 @@ The final image will only contain the solr-operator binary and necessary License
126126

127127
## Solr Operator Input Args
128128

129-
* **-zookeeper-operator** Whether or not to use the Zookeeper Operator to create dependency Zookeeepers.
130-
Required to use the `spec.zookeeperRef.provided` option.
131-
If _true_, then a Zookeeper Operator must be running for the cluster.
132-
(_true_ | _false_ , defaults to _false_)
129+
* **--zookeeper-operator** Whether or not to use the Zookeeper Operator to create dependency Zookeeepers.
130+
Required to use the `spec.zookeeperRef.provided` option.
131+
If _true_, then a Zookeeper Operator must be running for the cluster.
132+
(_true_ | _false_ , defaults to _false_)
133+
134+
* **--watch-namespaces** Watch certain namespaces in the Kubernetes cluster.
135+
If flag is omitted, or given an empty string, then the whole cluster will be watched.
136+
If the operator should watch multiple namespaces, provide them all separated by commas.
137+
(_string_ , defaults to _empty_)
138+
139+
* **--leader-elect** Whether or not to use leader election for the Solr Operator.
140+
If set to true, then only one operator pod will be functional for the namespaces given through `--watch-namespaces`.
141+
If multiple namespaces are provided, leader election will use the first namespace sorted alphabetically.
142+
(_true_ | _false_ , defaults to _true_)
143+
144+
* **--metrics-bind-address** The address to bind the metrics servlet on.
145+
If only a port is provided (e.g. `:8080`), then the metrics server will respond to requests with any Host header.
146+
(defaults to _:8080_)
147+
148+
* **--health-probe-bind-address=** The address to bind the health probe servlet on.
149+
If only a port is provided (e.g. `:8081`), then the metrics server will respond to requests with any Host header.
150+
(defaults to _:8081_)
133151

134152
## Client Auth for mTLS-enabled Solr clusters
135153

helm/solr-operator/Chart.yaml

+7
Original file line numberDiff line numberDiff line change
@@ -192,6 +192,13 @@ annotations:
192192
url: https://github.com/apache/solr-operator/pull/359
193193
- name: SolrBackup Documentation
194194
url: https://apache.github.io/solr-operator/docs/solr-backup#recurring-backups
195+
- kind: changed
196+
description: Solr Operator Leader Election enabled by default
197+
links:
198+
- name: Github Issue
199+
url: https://github.com/apache/solr-operator/issues/366
200+
- name: Github PR
201+
url: https://github.com/apache/solr-operator/pull/367
195202
artifacthub.io/images: |
196203
- name: solr-operator
197204
image: apache/solr-operator:v0.5.0-prerelease

helm/solr-operator/README.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -158,8 +158,9 @@ The command removes all the Kubernetes components associated with the chart and
158158
|-----|------|---------|-------------|
159159
| watchNamespaces | string | `""` | A comma-separated list of namespaces that the solr operator should watch. If empty, the solr operator will watch all namespaces in the cluster. If set to `true`, this will be populated with the namespace that the operator is deployed to. |
160160
| zookeeper-operator.install | boolean | `true` | This option installs the Zookeeper Operator as a helm dependency |
161-
| metrics.enable | boolean | `true` | Enable metrics for the Solr Operator. Will be available via the "metrics"/8080 port on the solr operator pods under the "/metrics" path. |
162161
| zookeeper-operator.use | boolean | `false` | This option enables the use of provided Zookeeper instances for SolrClouds via the Zookeeper Operator, without installing the Zookeeper Operator as a dependency. If `zookeeper-operator.install`=`true`, then this option is ignored. |
162+
| leaderElection.enable | boolean | `true` | Enable leader election for the Solr Operator. Will work across multiple `watchNamespaces`, as long as all deployments have the same list for `watchNamespaces`. |
163+
| metrics.enable | boolean | `true` | Enable metrics for the Solr Operator. Will be available via the "metrics"/8080 port on the solr operator pods under the "/metrics" path. |
163164
| mTLS.clientCertSecret | string | `""` | Name of a Kubernetes TLS secret, in the same namespace, that contains a Client certificate to load into the operator. If provided, this is used when communicating with Solr. |
164165
| mTLS.caCertSecretKey | string | `""` | Name of a Kubernetes secret, in the same namespace, that contains PEM encoded Root CA Certificate to use when connecting to Solr with Client Auth. |
165166
| mTLS.caCertSecret | string | `""` | Name of the key in the `caCertSecret` that contains the Root CA Cert as a value. |

helm/solr-operator/templates/deployment.yaml

+3
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,9 @@ spec:
7171
{{- if .Values.metrics.enable }}
7272
- "--metrics-bind-address=:8080"
7373
{{- end }}
74+
{{- if not .Values.leaderElection.enable }}
75+
- "--leader-elect=false"
76+
{{- end }}
7477

7578
env:
7679
- name: POD_NAMESPACE

helm/solr-operator/values.yaml

+3
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,9 @@ image:
2727
nameOverride: ""
2828
fullnameOverride: ""
2929

30+
leaderElection:
31+
enable: true
32+
3033
zookeeper-operator:
3134
# Include the zookeeper-operator as a helm chart dependency.
3235
# Setting this to true also tells the Solr Operator it is use to use ZookeeperCluster objects.

main.go

+26-20
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ import (
3232
"path/filepath"
3333
"runtime"
3434
"sigs.k8s.io/controller-runtime/pkg/cache"
35+
"sort"
3536
"strings"
3637

3738
// Import all Kubernetes client auth plugins (e.g. Azure, GCP, OIDC, etc.)
@@ -114,7 +115,7 @@ func main() {
114115
var probeAddr string
115116
flag.StringVar(&metricsAddr, "metrics-bind-address", ":8080", "The address the metric endpoint binds to.")
116117
flag.StringVar(&probeAddr, "health-probe-bind-address", ":8081", "The address the probe endpoint binds to.")
117-
flag.BoolVar(&enableLeaderElection, "leader-elect", false,
118+
flag.BoolVar(&enableLeaderElection, "leader-elect", true,
118119
"Enable leader election for controller manager. "+
119120
"Enabling this will ensure there is only one active controller manager.")
120121
opts := zap.Options{
@@ -135,33 +136,38 @@ func main() {
135136
setupLog.Info(fmt.Sprintf("Go Version: %v", runtime.Version()))
136137
setupLog.Info(fmt.Sprintf("Go OS/Arch: %s / %s", runtime.GOOS, runtime.GOARCH))
137138

138-
// When the operator is started to watch resources in a specific set of namespaces, we use the MultiNamespacedCacheBuilder cache.
139-
// In this scenario, it is also suggested to restrict the provided authorization to this namespace by replacing the default
140-
// ClusterRole and ClusterRoleBinding to Role and RoleBinding respectively
141-
// For further information see the kubernetes documentation about
142-
// Using [RBAC Authorization](https://kubernetes.io/docs/reference/access-authn-authz/rbac/).
143-
var managerWatchCache cache.NewCacheFunc
139+
operatorOptions := ctrl.Options{
140+
Scheme: scheme,
141+
MetricsBindAddress: metricsAddr,
142+
Port: 9443,
143+
HealthProbeBindAddress: probeAddr,
144+
LeaderElection: enableLeaderElection,
145+
LeaderElectionID: "88488bdc.solr.apache.org",
146+
}
147+
148+
/*
149+
When the operator is started to watch resources in a specific set of namespaces, we use the MultiNamespacedCacheBuilder cache.
150+
In this scenario, it is also suggested to restrict the provided authorization to this namespace by replacing the default
151+
ClusterRole and ClusterRoleBinding to Role and RoleBinding respectively
152+
For further information see the kubernetes documentation about
153+
Using [RBAC Authorization](https://kubernetes.io/docs/reference/access-authn-authz/rbac/).
154+
155+
When watching multiple namespaces with leader election enabled, the leader election will use the lowest-sorted namespace.
156+
It is likely safer to run individual solr operators per-namespace, to ensure leader election is working as expected no matter what.
157+
*/
144158
if watchNamespaces != "" {
145159
setupLog.Info(fmt.Sprintf("Managing for Namespaces: %s", watchNamespaces))
146160
ns := strings.Split(watchNamespaces, ",")
147161
for i := range ns {
148162
ns[i] = strings.TrimSpace(ns[i])
149163
}
150-
managerWatchCache = cache.MultiNamespacedCacheBuilder(ns)
151-
} else {
152-
setupLog.Info("Managing for the entire cluster.")
153-
managerWatchCache = (cache.NewCacheFunc)(nil)
164+
sort.Strings(ns)
165+
operatorOptions.NewCache = cache.MultiNamespacedCacheBuilder(ns)
166+
167+
operatorOptions.LeaderElectionNamespace = ns[0]
154168
}
155169

156-
mgr, err := ctrl.NewManager(ctrl.GetConfigOrDie(), ctrl.Options{
157-
Scheme: scheme,
158-
MetricsBindAddress: metricsAddr,
159-
Port: 9443,
160-
HealthProbeBindAddress: probeAddr,
161-
LeaderElection: enableLeaderElection,
162-
LeaderElectionID: "88488bdc.solr.apache.org",
163-
NewCache: managerWatchCache,
164-
})
170+
mgr, err := ctrl.NewManager(ctrl.GetConfigOrDie(), operatorOptions)
165171
if err != nil {
166172
setupLog.Error(err, "unable to start solr operator")
167173
os.Exit(1)

0 commit comments

Comments
 (0)