New to KubeDB? Please start here.

Run Solr with Custom PodTemplate

KubeDB supports providing custom configuration for Solr via PodTemplate. This tutorial will show you how to use KubeDB to run a Solr database with custom configuration using PodTemplate.

Before You Begin

  • At first, you need to have a Kubernetes cluster, and the kubectl command-line tool must be configured to communicate with your cluster. If you do not already have a cluster, you can create one by using kind.

  • Now, install KubeDB cli on your workstation and KubeDB operator in your cluster following the steps here.

  • To keep things isolated, this tutorial uses a separate namespace called demo throughout this tutorial.

    $ kubectl create ns demo
    namespace/demo created
    

Note: YAML files used in this tutorial are stored in docs/guides/solr/configuration/podtemplating/yamls folder in GitHub repository kubedb/docs.

Overview

KubeDB allows providing a template for leaf and aggregator pod through spec.topology.aggregator.podTemplate and spec.topology.leaf.podTemplate. KubeDB operator will pass the information provided in spec.topology.aggregator.podTemplate and spec.topology.leaf.podTemplate to the aggregator and leaf PetSet created for Solr database.

KubeDB accept following fields to set in spec.podTemplate:

  • metadata:
    • annotations (pod’s annotation)
    • labels (pod’s labels)
  • controller:
    • annotations (petset’s annotation)
    • labels (petset’s labels)
  • spec:
    • volumes
    • initContainers
    • containers
    • imagePullSecrets
    • nodeSelector
    • affinity
    • serviceAccountName
    • schedulerName
    • tolerations
    • priorityClassName
    • priority
    • securityContext
    • livenessProbe
    • readinessProbe
    • lifecycle

Read about the fields in details in PodTemplate concept,

CRD Configuration

Below is the YAML for the Solr created in this example.

apiVersion: kubedb.com/v1alpha2
kind: Solr
metadata:
  name: solr-misc-config
  namespace: demo
spec:
  version: "9.6.1"
  topology:
    data:
      replicas: 1
      podTemplate:
        spec:
          containers:
            - name: "solr"
              resources:
                requests:
                  cpu: "900m"
                limits:
                  cpu: "900m"
                  memory: "2.5Gi"
      storage:
        accessModes:
          - ReadWriteOnce
        resources:
          requests:
            storage: 1Gi
        storageClassName: standard
    overseer:
      replicas: 1
      podTemplate:
        spec:
          containers:
            - name: "solr"
              resources:
                requests:
                  cpu: "900m"
                limits:
                  cpu: "900m"
                  memory: "2.5Gi"
      storage:
        accessModes:
          - ReadWriteOnce
        resources:
          requests:
            storage: 1Gi
        storageClassName: standard
    coordinator:
      replicas: 1
      podTemplate:
        spec:
          containers:
            - name: "solr"
              resources:
                requests:
                  cpu: "900m"
                limits:
                  cpu: "900m"
                  memory: "2.5Gi"
      storage:
        accessModes:
          - ReadWriteOnce
        resources:
          requests:
            storage: 1Gi
        storageClassName: standard
$ kubectl create -f https://github.com/kubedb/docs/raw/v2025.1.9/docs/examples/solr/configuration/sl-custom-podtemplate.yaml
Solr.kubedb.com/solr-misc-config created

Now, wait a few minutes. KubeDB operator will create necessary PVC, petset, services, secret etc. If everything goes well, we will see that a pod with the name sdb-misc-config-aggregator-0 has been created.

Check that the petset’s pod is running

$ kubectl get pod -n demo -l app.kubernetes.io/instance=solr-misc-config
NAME                             READY   STATUS    RESTARTS   AGE
solr-misc-config-coordinator-0   1/1     Running   0          3m30s
solr-misc-config-data-0          1/1     Running   0          3m35s
solr-misc-config-overseer-0      1/1     Running   0          3m33s

Now, we will check if the database has started with the custom configuration we have provided.

$ kubectl get pod -n demo solr-misc-config-coordinator-0 -o json | jq '.spec.containers[].resources'
{
  "limits": {
    "cpu": "900m",
    "memory": "2560Mi"
  },
  "requests": {
    "cpu": "900m",
    "memory": "2560Mi"
  }
}

$ kubectl get pod -n demo solr-misc-config-data-0 -o json | jq '.spec.containers[].resources'
{
  "limits": {
    "cpu": "900m",
    "memory": "2560Mi"
  },
  "requests": {
    "cpu": "900m",
    "memory": "2560Mi"
  }
}

$ kubectl get pod -n demo solr-misc-config-overseer-0 -o json | jq '.spec.containers[].resources'
{
  "limits": {
    "cpu": "900m",
    "memory": "2560Mi"
  },
  "requests": {
    "cpu": "900m",
    "memory": "2560Mi"
  }
}

Using Node Selector

Here in this example we will use node selector to schedule our Solr pod to a specific node. Applying nodeSelector to the Pod involves several steps. We first need to assign a label to some node that will be later used by the nodeSelector . Let’s find what nodes exist in your cluster. To get the name of these nodes, you can run:

$ kubectl get nodes
NAME                                    STATUS   ROLES    AGE    VERSION
gke-pritam-default-pool-c682fe6e-59x3   Ready    <none>   110m   v1.30.5-gke.1443001
gke-pritam-default-pool-c682fe6e-rbtx   Ready    <none>   110m   v1.30.5-gke.1443001
gke-pritam-default-pool-c682fe6e-spdb   Ready    <none>   110m   v1.30.5-gke.1443001
gke-pritam-default-pool-cc96ce9b-049h   Ready    <none>   110m   v1.30.5-gke.1443001
gke-pritam-default-pool-cc96ce9b-b8p8   Ready    <none>   110m   v1.30.5-gke.1443001
gke-pritam-default-pool-cc96ce9b-vbpc   Ready    <none>   110m   v1.30.5-gke.1443001
gke-pritam-default-pool-dadbf4db-5fv5   Ready    <none>   110m   v1.30.5-gke.1443001
gke-pritam-default-pool-dadbf4db-5vkv   Ready    <none>   110m   v1.30.5-gke.1443001
gke-pritam-default-pool-dadbf4db-p039   Ready    <none>   110m   v1.30.5-gke.1443001

As you see, we have nine nodes in the cluster.

Let’s say we want pods to schedule to nodes with key topology.gke.io/zone and value us-central1-b

$ kubectl get nodes -n demo -l topology.gke.io/zone=us-central1-b
NAME                                    STATUS   ROLES    AGE    VERSION
gke-pritam-default-pool-c682fe6e-59x3   Ready    <none>   118m   v1.30.5-gke.1443001
gke-pritam-default-pool-c682fe6e-rbtx   Ready    <none>   118m   v1.30.5-gke.1443001
gke-pritam-default-pool-c682fe6e-spdb   Ready    <none>   118m   v1.30.5-gke.1443001

As you see, the gke-pritam-default-pool-c682fe6e-59x3 now has a new label topology.gke.io/zone=us-central1-b. To see all labels attached to the node, you can also run:

$ kubectl describe nodes gke-pritam-default-pool-c682fe6e-59x3
Name:               gke-pritam-default-pool-c682fe6e-59x3
Roles:              <none>
Labels:             beta.kubernetes.io/arch=amd64
                    beta.kubernetes.io/instance-type=e2-standard-2
                    beta.kubernetes.io/os=linux
                    cloud.google.com/gke-boot-disk=pd-balanced
                    cloud.google.com/gke-container-runtime=containerd
                    cloud.google.com/gke-cpu-scaling-level=2
                    cloud.google.com/gke-logging-variant=DEFAULT
                    cloud.google.com/gke-max-pods-per-node=110
                    cloud.google.com/gke-memory-gb-scaling-level=8
                    cloud.google.com/gke-nodepool=default-pool
                    cloud.google.com/gke-os-distribution=cos
                    cloud.google.com/gke-provisioning=standard
                    cloud.google.com/gke-stack-type=IPV4
                    cloud.google.com/machine-family=e2
                    cloud.google.com/private-node=false
                    disktype=ssd
                    failure-domain.beta.kubernetes.io/region=us-central1
                    failure-domain.beta.kubernetes.io/zone=us-central1-b
                    kubernetes.io/arch=amd64
                    kubernetes.io/hostname=gke-pritam-default-pool-c682fe6e-59x3
                    kubernetes.io/os=linux
                    node.kubernetes.io/instance-type=e2-standard-2
                    topology.gke.io/zone=us-central1-b
                    topology.kubernetes.io/region=us-central1
                    topology.kubernetes.io/zone=us-central1-b

Now let’s create a Solr with this new label as nodeSelector. Below is the yaml we are going to apply:

apiVersion: kubedb.com/v1alpha2
kind: Solr
metadata:
  name: solr-custom-nodeselector
  namespace: demo
spec:
  version: 9.6.1
  replicas: 2
  podTemplate:
    spec:
      nodeSelector:
        topology.gke.io/zone: us-central1-b
  zookeeperRef:
    name: zoo
    namespace: demo
  storage:
    accessModes:
      - ReadWriteOnce
    resources:
      requests:
        storage: 1Gi
$ kubectl create -f https://github.com/kubedb/docs/raw/v2025.1.9/docs/guides/solr/configuration/sl-custom-nodeselector.yaml
solr.kubedb.com/solr-node-selector created

Now, wait a few minutes. KubeDB operator will create necessary petset, services, secret etc. If everything goes well, we will see that a pod with the name sdb-node-selector-0 has been created.

Check that the petset’s pod is running

$ kubectl get pod -n demo -l app.kubernetes.io/instance=solr-custom-nodeselector
NAME                         READY   STATUS    RESTARTS   AGE
solr-custom-nodeselector-0   1/1     Running   0          3m18s
solr-custom-nodeselector-1   1/1     Running   0          2m54s

As we see the pod is running, you can verify that by running kubectl get pods -n demo sdb-node-selector-0 -o wide and looking at the “NODE” to which the Pod was assigned.

$ kubectl get pod -n demo -l app.kubernetes.io/instance=solr-custom-nodeselector -owide
NAME                         READY   STATUS    RESTARTS   AGE     IP          NODE                                    NOMINATED NODE   READINESS GATES
solr-custom-nodeselector-0   1/1     Running   0          3m52s   10.12.7.7   gke-pritam-default-pool-c682fe6e-spdb   <none>           <none>
solr-custom-nodeselector-1   1/1     Running   0          3m28s   10.12.8.9   gke-pritam-default-pool-c682fe6e-59x3   <none>           <none>

We can successfully verify that our pod was scheduled to our desired node.

Using Taints and Tolerations

Here in this example we will use Taints and Tolerations to schedule our Solr pod to a specific node and also prevent from scheduling to nodes. Applying taints and tolerations to the Pod involves several steps. Let’s find what nodes exist in your cluster. To get the name of these nodes, you can run:

$ kubectl get nodes
NAME                                    STATUS   ROLES    AGE    VERSION
gke-pritam-default-pool-c682fe6e-59x3   Ready    <none>   123m   v1.30.5-gke.1443001
gke-pritam-default-pool-c682fe6e-rbtx   Ready    <none>   123m   v1.30.5-gke.1443001
gke-pritam-default-pool-c682fe6e-spdb   Ready    <none>   123m   v1.30.5-gke.1443001
gke-pritam-default-pool-cc96ce9b-049h   Ready    <none>   123m   v1.30.5-gke.1443001
gke-pritam-default-pool-cc96ce9b-b8p8   Ready    <none>   123m   v1.30.5-gke.1443001
gke-pritam-default-pool-cc96ce9b-vbpc   Ready    <none>   123m   v1.30.5-gke.1443001
gke-pritam-default-pool-dadbf4db-5fv5   Ready    <none>   123m   v1.30.5-gke.1443001
gke-pritam-default-pool-dadbf4db-5vkv   Ready    <none>   123m   v1.30.5-gke.1443001
gke-pritam-default-pool-dadbf4db-p039   Ready    <none>   123m   v1.30.5-gke.1443001

As you see, we have nine nodes in the cluster

Next, we are going to taint these nodes.

$ kubectl taint nodes gke-pritam-default-pool-c682fe6e-59x3 key1=node1:NoSchedule
node/gke-pritam-default-pool-c682fe6e-59x3 tainted
$ kubectl taint nodes gke-pritam-default-pool-c682fe6e-rbtx key1=node2:NoSchedule
node/gke-pritam-default-pool-c682fe6e-rbtx tainted
$ kubectl taint nodes gke-pritam-default-pool-c682fe6e-spdb key1=node3:NoSchedule
node/gke-pritam-default-pool-c682fe6e-spdb tainted
$ kubectl taint nodes gke-pritam-default-pool-cc96ce9b-049h key1=node4:NoSchedule
node/gke-pritam-default-pool-cc96ce9b-049h tainted
$ kubectl taint nodes gke-pritam-default-pool-cc96ce9b-b8p8 key1=node5:NoSchedule
node/gke-pritam-default-pool-cc96ce9b-b8p8 tainted
$ kubectl taint nodes gke-pritam-default-pool-cc96ce9b-vbpc key1=node6:NoSchedule
node/gke-pritam-default-pool-cc96ce9b-vbpc tainted
$ kubectl taint nodes gke-pritam-default-pool-dadbf4db-5fv5 key1=node7:NoSchedule
node/gke-pritam-default-pool-dadbf4db-5fv5 tainted
$ kubectl taint nodes gke-pritam-default-pool-dadbf4db-5vkv key1=node8:NoSchedule
node/gke-pritam-default-pool-dadbf4db-5vkv tainted
$ kubectl taint nodes gke-pritam-default-pool-dadbf4db-p039 key1=node9:NoSchedule
node/gke-pritam-default-pool-dadbf4db-p039 tainted

Let’s see our tainted nodes here,

$ kubectl get nodes -o json | jq -r '.items[] | select(.spec.taints != null) | .metadata.name, .spec.taints'
gke-pritam-default-pool-c682fe6e-59x3
[
  {
    "effect": "NoSchedule",
    "key": "key1",
    "value": "node1"
  }
]
gke-pritam-default-pool-c682fe6e-rbtx
[
  {
    "effect": "NoSchedule",
    "key": "key1",
    "value": "node2"
  }
]
gke-pritam-default-pool-c682fe6e-spdb
[
  {
    "effect": "NoSchedule",
    "key": "key1",
    "value": "node3"
  }
]
gke-pritam-default-pool-cc96ce9b-049h
[
  {
    "effect": "NoSchedule",
    "key": "key1",
    "value": "node4"
  }
]
gke-pritam-default-pool-cc96ce9b-b8p8
[
  {
    "effect": "NoSchedule",
    "key": "key1",
    "value": "node5"
  }
]
gke-pritam-default-pool-cc96ce9b-vbpc
[
  {
    "effect": "NoSchedule",
    "key": "key1",
    "value": "node6"
  }
]
gke-pritam-default-pool-dadbf4db-5fv5
[
  {
    "effect": "NoSchedule",
    "key": "key1",
    "value": "node7"
  }
]
gke-pritam-default-pool-dadbf4db-5vkv
[
  {
    "effect": "NoSchedule",
    "key": "key1",
    "value": "node8"
  }
]
gke-pritam-default-pool-dadbf4db-p039
[
  {
    "effect": "NoSchedule",
    "key": "key1",
    "value": "node9"
  }
]

We can see that our taints were successfully assigned. Now let’s try to create a Solr without proper tolerations. Here is the yaml of Solr we are going to createc

apiVersion: kubedb.com/v1alpha2
kind: Solr
metadata:
  name: solr-without-toleration
  namespace: demo
spec:
  version: 9.6.1
  replicas: 2
  zookeeperRef:
    name: zoo
    namespace: demo
  storage:
    accessModes:
      - ReadWriteOnce
    resources:
      requests:
        storage: 1Gi
$ kubectl create -f https://github.com/kubedb/docs/raw/v2025.1.9/docs/guides/solr/configuration/solr-without-tolerations.yaml
solr.kubedb.com/solr-without-tolerations created

Now, wait a few minutes. KubeDB operator will create necessary petset, services, secret etc. If everything goes well, we will see that a pod with the name sdb-without-tolerations-0 has been created and running.

Check that the petset’s pod is running or not,

$ kubectl get pod -n demo -l app.kubernetes.io/instance=solr-without-toleration
NAME                        READY   STATUS    RESTARTS   AGE
solr-without-toleration-0   0/1     Pending   0          64s

Here we can see that the pod is not running. So let’s describe the pod,

$ kubectl describe pod -n demo solr-without-toleration-0
Name:             solr-without-toleration-0
Namespace:        demo
Priority:         0
Service Account:  default
Node:             <none>
Labels:           app.kubernetes.io/component=database
                  app.kubernetes.io/instance=solr-without-toleration
                  app.kubernetes.io/managed-by=kubedb.com
                  app.kubernetes.io/name=solrs.kubedb.com
                  apps.kubernetes.io/pod-index=0
                  controller-revision-hash=solr-without-toleration-7d4b4d4bcc
                  coordinator=set
                  data=set
                  overseer=set
                  statefulset.kubernetes.io/pod-name=solr-without-toleration-0
Annotations:      cloud.google.com/cluster_autoscaler_unhelpable_since: 2024-11-13T12:32:48+0000
                  cloud.google.com/cluster_autoscaler_unhelpable_until: Inf
Status:           Pending
IP:               
IPs:              <none>
Controlled By:    PetSet/solr-without-toleration
Init Containers:
  init-solr:
    Image:           ghcr.io/kubedb/solr-init:9.6.1@sha256:dbbee5c25da5666a90fbb5d90d146c3a8d54f04eefacd779b59a248c0972ef15
    Port:            <none>
    Host Port:       <none>
    SeccompProfile:  RuntimeDefault
    Limits:
      memory:  512Mi
    Requests:
      cpu:     200m
      memory:  512Mi
    Environment:
      SOLR_JAVA_MEM:           -Xms1g -Xmx3g
      SOLR_HOME:               /var/solr
      SOLR_PORT:               8983
      SOLR_NODE_PORT:          8983
      SOLR_logS_DIR:           /var/solr/logs
      SOLR_log_LEVEL:          DEBUG
      SOLR_PORT_ADVERTISE:     8983
      CLUSTER_NAME:            solr-without-toleration
      POD_HOSTNAME:            solr-without-toleration-0 (v1:metadata.name)
      POD_NAME:                solr-without-toleration-0 (v1:metadata.name)
      POD_IP:                   (v1:status.podIP)
      GOVERNING_SERVICE:       solr-without-toleration-pods
      POD_NAMESPACE:           demo (v1:metadata.namespace)
      SOLR_HOST:               $(POD_NAME).$(GOVERNING_SERVICE).$(POD_NAMESPACE)
      ZK_HOST:                 zoo-0.zoo-pods.demo.svc.cluster.local:2181,zoo-1.zoo-pods.demo.svc.cluster.local:2181,zoo-2.zoo-pods.demo.svc.cluster.local:2181/demosolr-without-toleration
      ZK_SERVER:               zoo-0.zoo-pods.demo.svc.cluster.local:2181,zoo-1.zoo-pods.demo.svc.cluster.local:2181,zoo-2.zoo-pods.demo.svc.cluster.local:2181
      ZK_CHROOT:               /demosolr-without-toleration
      SOLR_MODULES:            
      JAVA_OPTS:               
      CONNECTION_SCHEME:       http
      SECURITY_ENABLED:        true
      SOLR_USER:               <set to the key 'username' in secret 'solr-without-toleration-auth'>  Optional: false
      SOLR_PASSWORD:           <set to the key 'password' in secret 'solr-without-toleration-auth'>  Optional: false
      SOLR_OPTS:               -DhostPort=$(SOLR_NODE_PORT) -Dsolr.autoSoftCommit.maxTime=1000 -DzkACLProvider=org.apache.solr.common.cloud.DigestZkACLProvider -DzkCredentialsInjector=org.apache.solr.common.cloud.VMParamsZkCredentialsInjector -DzkCredentialsProvider=org.apache.solr.common.cloud.DigestZkCredentialsProvider -DzkDigestPassword=nwbnmVwBoJhW)eft -DzkDigestReadonlyPassword=7PxFSc)z~DWLL)Tt -DzkDigestReadonlyUsername=zk-digest-readonly -DzkDigestUsername=zk-digest
      ZK_CREDS_AND_ACLS:       -DzkACLProvider=org.apache.solr.common.cloud.DigestZkACLProvider -DzkCredentialsInjector=org.apache.solr.common.cloud.VMParamsZkCredentialsInjector -DzkCredentialsProvider=org.apache.solr.common.cloud.DigestZkCredentialsProvider -DzkDigestPassword=nwbnmVwBoJhW)eft -DzkDigestReadonlyPassword=7PxFSc)z~DWLL)Tt -DzkDigestReadonlyUsername=zk-digest-readonly -DzkDigestUsername=zk-digest
      SOLR_ZK_CREDS_AND_ACLS:  -DzkACLProvider=org.apache.solr.common.cloud.DigestZkACLProvider -DzkCredentialsInjector=org.apache.solr.common.cloud.VMParamsZkCredentialsInjector -DzkCredentialsProvider=org.apache.solr.common.cloud.DigestZkCredentialsProvider -DzkDigestPassword=nwbnmVwBoJhW)eft -DzkDigestReadonlyPassword=7PxFSc)z~DWLL)Tt -DzkDigestReadonlyUsername=zk-digest-readonly -DzkDigestUsername=zk-digest
    Mounts:
      /temp-config from default-config (rw)
      /var/run/secrets/kubernetes.io/serviceaccount from kube-api-access-n9nxh (ro)
      /var/security from auth-config (rw)
      /var/solr from slconfig (rw)
Containers:
  solr:
    Image:           ghcr.io/appscode-images/solr:9.6.1@sha256:b625c7e8c91c8070b23b367cc03a736f2c5c2cb9cfd7981f72c461e57df800a1
    Port:            8983/TCP
    Host Port:       0/TCP
    SeccompProfile:  RuntimeDefault
    Limits:
      memory:  2Gi
    Requests:
      cpu:     900m
      memory:  2Gi
    Environment:
      SOLR_JAVA_MEM:           -Xms1g -Xmx3g
      SOLR_HOME:               /var/solr
      SOLR_PORT:               8983
      SOLR_NODE_PORT:          8983
      SOLR_logS_DIR:           /var/solr/logs
      SOLR_log_LEVEL:          DEBUG
      SOLR_PORT_ADVERTISE:     8983
      CLUSTER_NAME:            solr-without-toleration
      POD_HOSTNAME:            solr-without-toleration-0 (v1:metadata.name)
      POD_NAME:                solr-without-toleration-0 (v1:metadata.name)
      POD_IP:                   (v1:status.podIP)
      GOVERNING_SERVICE:       solr-without-toleration-pods
      POD_NAMESPACE:           demo (v1:metadata.namespace)
      SOLR_HOST:               $(POD_NAME).$(GOVERNING_SERVICE).$(POD_NAMESPACE)
      ZK_HOST:                 zoo-0.zoo-pods.demo.svc.cluster.local:2181,zoo-1.zoo-pods.demo.svc.cluster.local:2181,zoo-2.zoo-pods.demo.svc.cluster.local:2181/demosolr-without-toleration
      ZK_SERVER:               zoo-0.zoo-pods.demo.svc.cluster.local:2181,zoo-1.zoo-pods.demo.svc.cluster.local:2181,zoo-2.zoo-pods.demo.svc.cluster.local:2181
      ZK_CHROOT:               /demosolr-without-toleration
      SOLR_MODULES:            
      JAVA_OPTS:               
      CONNECTION_SCHEME:       http
      SECURITY_ENABLED:        true
      SOLR_USER:               <set to the key 'username' in secret 'solr-without-toleration-auth'>  Optional: false
      SOLR_PASSWORD:           <set to the key 'password' in secret 'solr-without-toleration-auth'>  Optional: false
      SOLR_OPTS:               -DhostPort=$(SOLR_NODE_PORT) -Dsolr.autoSoftCommit.maxTime=1000 -DzkACLProvider=org.apache.solr.common.cloud.DigestZkACLProvider -DzkCredentialsInjector=org.apache.solr.common.cloud.VMParamsZkCredentialsInjector -DzkCredentialsProvider=org.apache.solr.common.cloud.DigestZkCredentialsProvider -DzkDigestPassword=nwbnmVwBoJhW)eft -DzkDigestReadonlyPassword=7PxFSc)z~DWLL)Tt -DzkDigestReadonlyUsername=zk-digest-readonly -DzkDigestUsername=zk-digest
      ZK_CREDS_AND_ACLS:       -DzkACLProvider=org.apache.solr.common.cloud.DigestZkACLProvider -DzkCredentialsInjector=org.apache.solr.common.cloud.VMParamsZkCredentialsInjector -DzkCredentialsProvider=org.apache.solr.common.cloud.DigestZkCredentialsProvider -DzkDigestPassword=nwbnmVwBoJhW)eft -DzkDigestReadonlyPassword=7PxFSc)z~DWLL)Tt -DzkDigestReadonlyUsername=zk-digest-readonly -DzkDigestUsername=zk-digest
      SOLR_ZK_CREDS_AND_ACLS:  -DzkACLProvider=org.apache.solr.common.cloud.DigestZkACLProvider -DzkCredentialsInjector=org.apache.solr.common.cloud.VMParamsZkCredentialsInjector -DzkCredentialsProvider=org.apache.solr.common.cloud.DigestZkCredentialsProvider -DzkDigestPassword=nwbnmVwBoJhW)eft -DzkDigestReadonlyPassword=7PxFSc)z~DWLL)Tt -DzkDigestReadonlyUsername=zk-digest-readonly -DzkDigestUsername=zk-digest
    Mounts:
      /var/run/secrets/kubernetes.io/serviceaccount from kube-api-access-n9nxh (ro)
      /var/solr from slconfig (rw)
      /var/solr/data from solr-without-toleration-data (rw)
Conditions:
  Type           Status
  PodScheduled   False 
Volumes:
  solr-without-toleration-data:
    Type:       PersistentVolumeClaim (a reference to a PersistentVolumeClaim in the same namespace)
    ClaimName:  solr-without-toleration-data-solr-without-toleration-0
    ReadOnly:   false
  default-config:
    Type:        Secret (a volume populated by a Secret)
    SecretName:  solr-without-toleration-config
    Optional:    false
  slconfig:
    Type:       EmptyDir (a temporary directory that shares a pod's lifetime)
    Medium:     
    SizeLimit:  <unset>
  auth-config:
    Type:        Secret (a volume populated by a Secret)
    SecretName:  solr-without-toleration-auth-config
    Optional:    false
  kube-api-access-n9nxh:
    Type:                    Projected (a volume that contains injected data from multiple sources)
    TokenExpirationSeconds:  3607
    ConfigMapName:           kube-root-ca.crt
    ConfigMapOptional:       <nil>
    DownwardAPI:             true
QoS Class:                   Burstable
Node-Selectors:              <none>
Tolerations:                 node.kubernetes.io/not-ready:NoExecute op=Exists for 300s
                             node.kubernetes.io/unreachable:NoExecute op=Exists for 300s
Events:
  Type     Reason             Age                  From                Message
  ----     ------             ----                 ----                -------
  Normal   NotTriggerScaleUp  106s                 cluster-autoscaler  pod didn't trigger scale-up:
  Warning  FailedScheduling   104s (x2 over 106s)  default-scheduler   0/9 nodes are available: 1 node(s) had untolerated taint {key1: node1}, 1 node(s) had untolerated taint {key1: node2}, 1 node(s) had untolerated taint {key1: node3}, 1 node(s) had untolerated taint {key1: node4}, 1 node(s) had untolerated taint {key1: node5}, 1 node(s) had untolerated taint {key1: node6}, 1 node(s) had untolerated taint {key1: node7}, 1 node(s) had untolerated taint {key1: node8}, 1 node(s) had untolerated taint {key1: node9}. preemption: 0/9 nodes are available: 9 Preemption is not helpful for scheduling.

Here we can see that the pod has no tolerations for the tainted nodes and because of that the pod is not able to scheduled.

So, let’s add proper tolerations and create another Solr. Here is the yaml we are going to apply,

apiVersion: kubedb.com/v1alpha2
kind: Solr
metadata:
  name: solr-with-toleration
  namespace: demo
spec:
  version: 9.6.1
  replicas: 2
  podTemplate:
    spec:
      tolerations:
        - key: "key1"
          operator: "Equal"
          value: "node7"
          effect: "NoSchedule"
        - key: "key1"
          operator: "Equal"
          value: "node8"
          effect: "NoSchedule"
  zookeeperRef:
    name: zoo
    namespace: demo
  storage:
    accessModes:
      - ReadWriteOnce
    resources:
      requests:
        storage: 1Gi
$ kubectl create -f https://github.com/kubedb/docs/raw/v2025.1.9/docs/guides/Solr/configuration/solr-with-tolerations.yaml
solr.kubedb.com/solr-with-tolerations created

Now, wait a few minutes. KubeDB operator will create necessary petset, services, secret etc. If everything goes well, we will see that a pod with the name sdb-with-tolerations-0 has been created.

Check that the petset’s pod is running

$ kubectl get pod -n demo -l app.kubernetes.io/instance=solr-with-toleration
NAME                     READY   STATUS    RESTARTS   AGE
solr-with-toleration-0   1/1     Running   0          2m12s
solr-with-toleration-1   1/1     Running   0          80s

As we see the pod is running, you can verify that by running kubectl get pods -n demo sdb-with-tolerations-0 -o wide and looking at the “NODE” to which the Pod was assigned.

$ kubectl get pod -n demo -l app.kubernetes.io/instance=solr-with-toleration -owide
NAME                     READY   STATUS    RESTARTS   AGE     IP          NODE                                    NOMINATED NODE   READINESS GATES
solr-with-toleration-0   1/1     Running   0          2m37s   10.12.3.7   gke-pritam-default-pool-dadbf4db-5fv5   <none>           <none>
solr-with-toleration-1   1/1     Running   0          105s    10.12.5.5   gke-pritam-default-pool-dadbf4db-5vkv   <none>           <none>

We can successfully verify that our pod was scheduled to the node which it has tolerations.

Cleaning up

To cleanup the Kubernetes resources created by this tutorial, run:

kubectl delete solr -n demo solr-misc-config solr-without-toleration solr-with-toleration

kubectl delete ns demo

If you would like to uninstall KubeDB operator, please follow the steps here.

Next Steps